如何使用canvas完成图片合成下载

最近在群里面看到一个很有意思的提问:
用户希望上传一张海报与另外一个小图片,最后可以下载下来,图片,小图片位于大图片的右下角

这种功能其实还挺常见的,比如生成一个带二维码的海报之类的,就抽时间做了一下

首先分析一下思路

  1. 要做图片合成,那么必定会使用到canvas
  2. 怎么进行合成?这个就很简单了?将canvas设置成大图的宽高,绘制大图,再在对应位置绘制小图,那么就得到我们所需要的图案
  3. 最后需要做的是如何下载,可以使用a标签来完成

具体代码

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>Document</title>
  <script src="https://cdn.bootcss.com/rxjs/6.5.2/rxjs.umd.js"></script>
</head>

<body>
  <canvas id="combination"></canvas>
  <a download id="download">下载</a>
  <script>
    const src0 = 'http://img5.imgtn.bdimg.com/it/u=3108685336,1951350863&fm=26&gp=0.jpg'
    const src1 = 'http://img5.imgtn.bdimg.com/it/u=2221756694,2517784243&fm=26&gp=0.jpg'
    const img0 = new Image()
    const img1 = new Image()
    img0.src = src0
    img1.src = src1
    // 当两张图都加载好了的时候进行canvas设置
    const img0Load$ = rxjs.fromEvent(img0, 'load')
    const img1Load$ = rxjs.fromEvent(img1, 'load')
    rxjs.zip(img0Load$, img1Load$).subscribe(([e0, e1]) => {
      console.log(e0);
      const img0 = e0.path[0]
      const img1 = e1.path[0]
      const imgSrc0 = img0.currentSrc
      const imgSrc1 = img1.currentSrc
      const img0Width = img0.naturalWidth
      const img1Width = img1.naturalWidth
      const img0Height = img0.naturalHeight
      const img1Height = img1.naturalHeight
      const minWidth = Math.min(img0Width, img1Width)
      const maxWidth = Math.max(img0Width, img1Width)
      const minHeight = Math.min(img0Height, img1Height)
      const maxHeight = Math.max(img0Height, img1Height)
      const canvas = document.getElementById('combination')
      canvas.width = maxWidth
      canvas.height = maxHeight
      const ctx = canvas.getContext('2d')
      let background = new Image()
      let minImage = new Image()
      // 如果不设置会导致报错
      // Tainted canvases may not be exported
      // https://stackoverflow.com/questions/22710627/tainted-canvases-may-not-be-exported
      background.setAttribute('crossOrigin', 'anonymous')
      minImage.setAttribute('crossOrigin', 'anonymous')
      if (maxHeight === img0Height) {
        background.src = imgSrc0
        minImage.src = imgSrc1
      } else {
        background.src = imgSrc1
        minImage.src = imgSrc0
      }
      rxjs.zip(rxjs.fromEvent(background, 'load'), rxjs.fromEvent(minImage, 'load')).subscribe(() => {
        // 绘制背景
        ctx.drawImage(background, 0, 0, maxWidth, maxHeight)
        // 绘制小图
        ctx.drawImage(minImage, maxWidth - minWidth, maxHeight - minHeight, minWidth, minHeight);
        const downloadImageSrc = canvas.toDataURL("image/png")
        document.getElementById('download').href = downloadImageSrc
      })
    })
  </script>
</body>

</html>

下面对代码进行解析

  1. 请无视rxjs的代码,只要知道他们的作用是:当两张图片都加载完成的时候,执行里面的内容,类似Promise.all
  2. 使用drawImage来绘制图片,注意第一个参数不是src而是一个image对象
  3. 注意minImage.setAttribute('crossOrigin', 'anonymous') 这句代码,如果没有这段代码会导致跨域加载图片的问题出现导致 canvas.toDataURL("image/png") 无法执行

可见还是相对比较简单的,在实际开发中并不需要去识别那个是大图那个是小图,一般在上传的时候就决定了,而且一般图片大小都是固定的,所以并不需要使用这么麻烦的rxjs代码

项目地址

图片合成

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容