最新公告
  • 欢迎您光临网站无忧模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • canvas炫技--抠图或颜色替换

    正文概述 掘金(阿明Beck)   2021-08-26   805

    演示地址 代码仓库

    关键技术

    canvas apielement-ui (el-color-picker、el-upload)

    原理解析

    1. 用户从本地上传一张图片

    我们拿到图片数据并在页面渲染出来,这一步用到了elementui的el-upload

    el-upload.avatar-uploader(
      ref="logoUpload",
      accept="image/*",
      action="#",
      :auto-upload="false",
      :on-change="handleStatusChange"
    )
      el-button(size="small", type="primary") 点击上传
    

    这里实际上不需要真正上传到服务器,只需要获取到图片的在内存中的url

    handleStatusChange(file) {
      // console.log(file);
      this.originImg = URL.createObjectURL(file.raw);
    },
    

    拿到url之后可以先把图片用img标签渲染在页面上,这样做的目的是为了获取图片的实际尺寸,方面我们等比例缩放在我们的canvas上。

    // 图片加载完成
        loadImg(e) {
          const img = e.target;
          const width = img.offsetWidth;
          const height = img.offsetHeight;
          this.imgWidth = width;
          this.imgHeight = height;
          this.canHeight = (height / width) * this.canWidth;
        }
    

    canvas的width可以根据外层容器来获取

    this.$nextTick(() => {
          const contentWidth = document.querySelector(".origin-box").offsetWidth;
          this.canWidth = Math.min(contentWidth - 12, this.canWidth);
    });
    

    2. 绘制canvas

    绘制图片到canvas,这一步比较简单,

    //这里需要缩放一下,因为我们的画布已经被缩放了
    this.originCtx.scale(this.canWidth / width, this.canWidth / width);
    this.originCtx.drawImage(img, 0, 0);
    

    3. 选中颜色

    点击canvas,我们可以拿到该点上的颜色值,获取方式

    ctx.getImageData(targetX,targetY,1,1)
    

    拿到的是imagedate对象,{data, width, height},data即为我们要的颜色值。

    4. 遍历原图片的颜色值,匹配到选中颜色之后,做对应颜色的替换即可

    核心api: getImageData putImageData

    //获取data
    const data = this.imageData.data || [];
    //遍历并替换
    for (let i = 0; i < data.length; i += 4) {
            const similar = this.isSimilar(_fromColor, data.slice(i, i + 4));
            if (similar) {
              data[i] = _toColor[0];
              data[i + 1] = _toColor[1];
              data[i + 2] = _toColor[2];
              data[i + 3] = _toColor[3] * 255;
            }
     }
     //绘制到目标容器上
     this.transCtx.putImageData(this.imageData, 0, 0);
    

    isSimilar方法用来判断两个颜色是否相似或相等,这个可以通过参数调整(类似于ps的容差概念),容差值越小,匹配越精准。

    5. undo&redo

    撤销、前进、后退功能还是很有必要的,重复替换操作,可返回历史操作步骤。 创建一个队列(这里用数组代替),每次有新的数据变化添加到队列里,用unshift表示入列pop从队列后面删除。可以设置上限10,队列过大会占用较大内存,不建议设置过大。 维护一个index,理解成指针,表示当前回退的数据在队列中的位置。

    undo() {
      this.index++;
      this.redrawImg();
    },
    redo() {
      this.index--;
      this.redrawImg();
    },
    redrawImg() {
      const preImageData = JSON.parse(this.imgStock[this.index]).data;
      this.imageData = this.transCtx.createImageData(
        this.canWidth,
        this.canHeight
      );
      for (let i = 0; i < this.imageData.data.length; i++) {
        this.imageData.data[i] = preImageData[i];
      }
      this.transCtx.putImageData(this.imageData, 0, 0);
    },
    

    至此,就完成了核心功能了~

    TODO LIST

    1. 增加边缘识别,去除毛边
    2. 增加容差选项
    3. 尝试视频抠图和替换
    4. 。。。

    演示地址 代码仓库

    欢迎提意见&star!


    下载网 » canvas炫技--抠图或颜色替换

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元