小程序之canvas图片及文本适配

场景需求

小程序目前不支持直接分享到朋友圈,所以对于有分享到朋友圈的需求,一般是生成一张图片,例如,生成一张带有小程序码的图片,用户可以将这张图片保存到手机本地,然后将这张图片分享到朋友圈。这张图片需要使用canvas画出来。今天我们不讲怎么生成码,这个一般是后端封装一个API,前端通过调用API得到一个小程序码的url,通过image去画到canvas上,跟在canvas上画一个image是一样的逻辑。这篇文章主要是讲怎么在canvas上适配图片和文字,也就是怎么将图片和文本画到canvas上的正确的位置,能在不同的手机上都能正确的展示。

效果图展示(以下图片是在微信开发者工具中显示的)

这里演示的Demo是:
“选择图片”button选择一张图片,可以居中显示在以下图中的黄色区域,即canvas中,并在canvas的顶部居中展示文本“你若盛开,蝴蝶自来”。

以下是两个不同尺寸的图片画在canvas上,分别在iphone5、iphone6、iphone6 plus上的展示效果图。


这里的猫图片的原始尺寸:658*658

这里的girl图片的原始尺寸是:700*699

代码说明

$1. 小程序尺寸单位
小程序尺寸单位
$2. wxml
<button bindtap="chooseImg">选择图片</button>
<view class="canvas-box">
    <canvas class="canvas" canvas-id="shareCanvas" bindlongtap="saveShareImage"></canvas>
</view>
$3. scss(wxss)

这里我用的是scss去写样式代码,在webStorm中可以通过配置,将scss解析成wxss,至于具体怎么配置,可以自行百度。
PS1:这里定义了一个scss rpx function,是因为在webStrom中格式化代码的时候会将数字和单位rpx中间添加一个空格,导致编译报错,所以用一个function去处理
PS2:样式说明(设计稿是基于iphone6视觉稿标准):
button宽高:700rpx * 92rpx (350px * 46px)
包裹canvas的容器的view(.canvas-box)position:fixed,占据除顶部button的高度之外,填满剩余全部空间,canvas居中显示,宽高:700rpx, 100%

@function rpx($value) {
  @return $value*1rpx;
}

page {
  background-color: #fff6f6;
  button {
    width: rpx(700);
    height: rpx(92);
    background-color: #ffddab !important;
  }
  .canvas-box {
    position: fixed;
    top: rpx(92);
    left: 0;
    right: 0;
    bottom: 0;
    canvas {
      position: absolute;
      width: rpx(700);
      height: 100%;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
    }
  }
}
$4. js

1、在data中定义三个变量

data: {
        windowWidth: 0, // 可使用窗口宽度
        windowHeight: 0, // 可使用窗口高度
        ratio: 0 // 根据尺寸动态计算 1px换算成多少rpx
    }

2、通过wx.getSystemInfo获得系统信息,并且计算ratio

    onReady: function (e) {
        // 获取系统信息
        wx.getSystemInfo({
            success: res => {
                console.log(res);
                this.setData({
                    windowWidth: res.windowWidth,
                    windowHeight: res.windowHeight,
                });
                this.setData({
                    // 屏幕宽度 375px = 750rpx,1px=2rpx
                    // 1px = (750 / 屏幕宽度)rpx;
                    // 1rpx = (屏幕宽度 / 750)px;
                    ratio: 750 / this.data.windowWidth
                });
            }
        });
    },

3、button的触发事件chooseImg,通过wx.chooseImage选择图片,通过wx.getImageInfo获取选择的图片的大小

chooseImg: function () {
        wx.chooseImage({
            count: 1,
            success: res => {
                let imgUrl = res.tempFilePaths[0];
                // 获取图片大小
                wx.getImageInfo({
                    src: imgUrl,
                    success: data => {
                        let imgWidth = data.width;
                        let imgHeight = data.height;

                        // 创建canvas,根据选择的图片大小,在canvas上绘制图片和文字
                        this.createCanvasShareImage(imgUrl, imgWidth, imgHeight);
                    }
                });
            }
        });
    }

4、创建canvas并在canvas上添加图片和文本

 createCanvasShareImage: function (imgUrl, imgW, imgH) {
        // 使用wx.createCanvasContext获取绘图上下文 context
        let context = wx.createCanvasContext('shareCanvas');
        console.log('context: ', context);

        // 获取canvas的宽度:
        // 750的设计稿基于iphone6的尺寸(屏幕宽度: 375px)在小程序中的比例是: 1px = 2rpx ==> 375px = 750rpx ==> 屏幕宽度(px) = 750rpx
        // 所以 1rpx = 屏幕宽度 / 750
        // 我们这里css中设置的 canvas 的width:700rpx, 所以 canvas的宽度计算是: [(屏幕宽度 / 750)* 700]rpx, 这样就可以做到在不同手机上都可以适配
        let canvasWidthPx = 700 / this.data.ratio;

        // 设置 canvas 的背景并填充canvas
        context.fillStyle = '#ffdc22';
        context.fillRect(0, 0, canvasWidthPx, this.data.windowHeight);

        // 绘制图片:图片居中显示在 canvas 中
        let imgX = (700 - imgW) / 2;
        let imgY = (this.data.windowHeight * this.data.ratio - 46 * this.data.ratio - imgH) / 2;
        let clipWidth = imgW * this.data.ratio;
        let clipHeight = imgH * this.data.ratio;
        context.drawImage(imgUrl, -imgX, -imgY, clipWidth, clipHeight, 0, 0, imgW, imgH);

        // 设置字体大小、文本颜色
        context.setFontSize(20);
        context.fillStyle = "#000";

        // 计算文本的宽度:measureText() 取到文本的 width
        let txtWidth = context.measureText('你若盛开,蝴蝶自来').width;

        // 绘制居中文本:这个地方的 (x, y)的坐标是在文本的左下角位置
        context.fillText("你若盛开,蝴蝶自来", (canvasWidthPx - txtWidth) / 2, 20 * this.data.ratio);

        context.draw();
    }

5、长按bindlongtapcanva保存图片或者直接预览图片,先通过wx.canvasToTempFilePath将当前画布指定区域的内容导出生成指定大小的的图片,再通过wx.saveImageToPhotosAlbum将图片保存到本地,或者通过wx.previewImage直接预览图片。

    saveShareImage() {
        wx.canvasToTempFilePath({
            canvasId: 'shareCanvas',
            success: (res) => {
                console.log('canvasToTempFilePath: ', res);
                // 将图片保存到相册
                wx.saveImageToPhotosAlbum({
                    filePath: res.tempFilePath,
                    success: (data) => {
                        console.log(data);
                    }
                });

                // 直接预览图片
                wx.previewImage({
                    urls: [res.tempFilePath]
                })
            }
        })
    }

真机效果图显示(一加六手机)

一加6手机显示的小程序效果图.png

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

推荐阅读更多精彩内容

  • 转载链接 注:本文转载知乎上的回答 作者:初雪 链接:https://www.zhihu.com/question...
    pengshuangta阅读 28,522评论 9 295
  • 开发前准备本文首先假定开发者已经粗略阅读过微信小程序的开发文档,所以注册小程序的流程就不介绍了。不过需要注意,小程...
    人类进化又没带我阅读 1,845评论 0 14
  • 给提问的开发者的建议:提问之前先查询 文档、通过社区右上角搜索搜索已经存在的问题。 写一个简明扼要的标题,并且...
    极乐叔阅读 13,414评论 0 3
  • 作者蔡骏 借来看这本书看的原因是,它是一个悬疑小说,是想学习它制造悬念,引发好奇的描写手法,结果看完之后在心...
    花色春秋阅读 680评论 0 0
  • 突然要一个人面对所有的一切,说实话很不舍,很希望能够在一起,至少有个情感的寄托!但是一个人总要独立面对生活。就算对...
    妖艳贱客阅读 428评论 0 0