微信分享图片组件(小程序 )

效果图.png

需求:点击分享,弹出遮罩层,可分享给好友,可保存图片分享,canvas中的图片自适应(缩放、裁剪)
技术:canvas 2,vant-weapp-ui
思路:计划将整个分享都写成一个自定义组件,由于一些bug,没能实现,部分还是需要写在页面中。

share组件:
share.wxml

<view class="wrapper shareWindow">
    <view class="other">
        <view class="flex_between">
            <button style="line-height: 100rpx;background:unset;" class="shareButton" open-type="share">
                <image class="share-icon" src="../../images/icon/wechat.png"></image>
                分享好友
            </button>
            <view class="pointer" catch:tap="savePic">
                <van-icon name="down" class="share-icon" /> 保存图片
            </view>
        </view>
    </view>
</view>

share.js 含有canvas绘制白色背景,canvas文本换行,绘制图片,图片原比例缩小裁剪,图片自适应,保存图片等方法

import {request} from "../../common/httpService"; 

const app = getApp()

Component({
    /**
     * 组件的属性列表
     */
    properties: {
        type: {
            type: String,
            value: ''
        },
        shareWords: {
            type: String,
            value: ''
        },
        imgSrc: {
            type: String,
            value: ''
        },
        title: {
            type: String,
            value: ''
        },
        userName: {
            type: String,
            value: ''
        },
        product: {
            type: Object,
            value: {}
        },
        dec: {
            type: String,
            value: ''
        },
        link: {
            type: String,
            value: ''
        },
        sceneStr: {
            type: String,
            value: ''
        },
        articleName: {
            type: String,
            value: ''
        }
    },

    /**
     * 组件的初始数据
     */
    data: {
        showShare: false,
        referrer: wx.getStorageSync('userId'),
        canvas: null
    },
    ready: function () {
        if (this.properties.type === 'article') {
            this.createArticlePic()
        } else if (this.properties.type === 'invited') {
            this.createInvitePic()
        } else if (this.properties.type === 'product') {
            this.createProductPic()
        }
        // this.getCodeImg()
    },


    /**
     * 组件的方法列表
     */
    methods: {
        initCanvas() {
            return new Promise((resolve, reject) => {
                const query = wx.createSelectorQuery()
                query.select('#myCanvas')
                    .fields({
                        node: true,
                        size: true,
                        id: true
                    })
                    .exec((res) => {
                        const canvas = res[0].node
                        const ctx = canvas.getContext('2d')
                        this.setData({
                            canvas: canvas
                        })
                        const dpr = wx.getSystemInfoSync().pixelRatio
                        canvas.width = res[0].width * dpr
                        canvas.height = res[0].height * dpr
                        ctx.scale(dpr, dpr)
                        let imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
                        console.log('imageData', imageData)
                        for (let i = 0; i < imageData.data.length; i += 4) {
                            if (imageData.data[i + 3] === 0) {
                                imageData.data[i] = 255;
                                imageData.data[i + 1] = 255;
                                imageData.data[i + 2] = 255;
                                imageData.data[i + 3] = 255;
                            }
                        }
                        ctx.putImageData(imageData, 0, 0);
                        let data = {
                            ctx,
                            canvas: res[0]
                        }
                        return resolve(data)
                    })
            })
        },
        createArticlePic() {
            let that = this
            that.initCanvas().then((data) => {
                const ctx = data.ctx
                const canvasInfo = data.canvas
                const canvas = that.data.canvas
                let pic = that.properties.imgSrc
                if (!that.properties.imgSrc) {
                    pic = '../../images/share-article-bg.png'
                }
                let cW = canvasInfo.width * 0.91
                let cH = canvasInfo.height * 0.56
                console.log('cW', cW)
                console.log('cH', cH)
                that.drawArticlePic(ctx, canvas, cW, cH, pic).then(res => {
                    console.log('画入成功')
                    ctx.fillStyle = 'rgba(0,0,0,0.5)'
                    ctx.fillRect(15, 95, cW, cH)
                    ctx.fillStyle = '#333333'
                    ctx.font = '18px PingFang'
                    let title = that.properties.articleName
                    console.log('文章title', title)
                    if (!title) {
                        if (that.properties.dec.length < 10) {
                            title = that.properties.dec
                        } else {
                            title = that.properties.dec.slice(0, 10) + '...'
                        }
                    }
                    ctx.fillText(title, 15, 50)
                    ctx.fillStyle = '#aaa'
                    ctx.font = '14px PingFang'
                    ctx.fillText(`来自${that.properties.userName}的邀请`, 15, 80)
                    ctx.fillStyle = '#ffffff'
                    ctx.font = '14px PingFang'
                    ctx.fillText(`123人已浏览`, 30, 288)
                    wx.canvasToTempFilePath({
                        canvasId: 'myCanvas',
                        canvas: canvas,
                        fileType: 'png',
                        success: function (res) {
                            console.log('canvas生成图片')
                            console.log(res.tempFilePath)
                            that.setData({
                                canvasPicPath: res.tempFilePath
                            })
                        },
                        fail: function (res) {
                            console.log('canvas生成图片失败')
                            console.log(res);
                        }
                    }, that)
                })
            })
        },
        createProductPic() {
            let that = this
            that.initCanvas().then(data => {
                const ctx = data.ctx
                const canvasInfo = data.canvas
                const canvas = that.data.canvas
                let borderW = canvasInfo.width * 0.89
                let borderH = canvasInfo.height * 0.77
                let textW = canvasInfo.width * 0.81
                this.initCanvasBg(canvas, ctx, "../../images/logo.png", 25, 25, 15, 28)
                ctx.fillStyle = '#333333'
                ctx.font = '12px PingFang'
                ctx.fillText('绿植物', 49, 46)
                ctx.fillStyle = '#8E8E8E'
                ctx.font = '14px PingFang'
                ctx.fillText(`${this.properties.userName}给你推荐了一个好东西`, 20, 79)
                ctx.strokeStyle = '#EEEEEE'
                ctx.strokeRect(20, 89, borderW, borderH)
                that.initCanvasBg(canvas, ctx, this.properties.imgSrc, 163, 163, 84, 98)
                ctx.fillStyle = '#CF1322'
                ctx.font = '24px PingFang'
                ctx.fillText('¥' + this.properties.product.vipPrice, 27, 288)
                ctx.fillStyle = '#AAAAAA'
                ctx.font = '14px PingFang'
                ctx.fillText('¥' + this.properties.product.originPrice, 105, 288)
                //原价删除线
                ctx.beginPath()
                ctx.lineWidth = 1;
                ctx.strokeStyle = "#AAAAAA";
                ctx.moveTo(103, 283)
                ctx.lineTo(149, 283)
                ctx.stroke()
                ctx.fillStyle = '#333333'
                ctx.font = '14px PingFang'
                // ctx.fillText(that.properties.product.name, 27, 318)
                that.drawText(ctx, that.properties.product.name, 29, 293, textW);
                ctx.fillStyle = '#AAAAAA'
                ctx.font = '12px PingFang'
                // ctx.fillText(that.properties.product.fullDescription, 27, 340)
                that.drawText(ctx, that.properties.product.fullDescription, 29, 340, textW);
                // TODO 二维码
                wx.canvasToTempFilePath({
                    canvasId: 'myCanvas',
                    canvas: canvas,
                    fileType: 'png',
                    success: function (res) {
                        console.log('canvas生成图片')
                        console.log(res.tempFilePath)
                        that.setData({
                            canvasPicPath: res.tempFilePath
                        })
                    },
                    fail: function (res) {
                        console.log('canvas生成图片失败')
                        console.log(res);
                    }
                }, that)
            })
        },
        createInvitePic() {
            let that = this
            that.initCanvas().then(data => {
                const ctx = data.ctx
                const canvasInfo = data.canvas
                const canvas = that.data.canvas
                let systemWidth = app.globalData.systemWidth
                let picWidth = systemWidth * 0.8
                let picHeight = picWidth * 0.97
                this.initCanvasBg(canvas, ctx, "../../images/logo.png", 25, 25, 15, 36).then((logoSuc) => {
                    console.log('logoSuc', logoSuc)
                    this.initCanvasBg(canvas, ctx, "../../images/share-img.png", picWidth, picHeight).then((res1) => {
                        console.log('res1', res1)
                        ctx.textAlign = 'left'
                        ctx.fillStyle = '#333333'
                        ctx.font = '14px PingFang'       // 文字字号:22px
                        ctx.fillText('绿植物', 49, 56)
                        ctx.textAlign = 'center'
                        ctx.fillStyle = '#E2A02B'
                        ctx.font = '20px PingFang'       // 文字字号:22px
                        ctx.fillText('邀您成为VIP会员', canvasInfo.width / 2, 137)
                        ctx.textAlign = 'center'
                        ctx.fillStyle = '#ffffff'
                        ctx.font = '14px PingFang'       // 文字字号:22px
                        ctx.fillText(`来自${this.properties.userName}的邀请`, canvasInfo.width / 2, 160)
                        this.roundRectColor(ctx, 119, 250, 92, 22, 16, canvasInfo.width)
                        ctx.textAlign = 'center'
                        ctx.fillStyle = '#E2A02B'
                        ctx.font = '20px PingFang'       // 文字字号:22px
                        ctx.fillText('最高可享3折优惠', canvasInfo.width / 2, 325)
                        wx.canvasToTempFilePath({
                            canvasId: 'myCanvas',
                            canvas: canvas,
                            fileType: 'png',
                            success: function (res) {
                                console.log('canvas生成图片')
                                console.log(res.tempFilePath)
                                that.setData({
                                    canvasPicPath: res.tempFilePath
                                })
                            },
                            fail: function (res) {
                                console.log('canvas生成图片失败')
                                console.log(res);
                            }
                        }, that)
                    })
                })
            })
        },
        initCanvasBg(canvas, ctx, imgPath, w, h, x, y) {
            return new Promise((resolve, reject) => {
                let bgPicturePath = imgPath;//图片路径不要出错
                const img = canvas.createImage()
                img.src = bgPicturePath
                let oX = 15
                let oY = 76
                if (x) {
                    oX = x
                }
                if (y) {
                    oY = y
                }
                img.onload = () => {
                    ctx.drawImage(img, oX, oY, w, h);
                    return resolve()
                }
            })
        },
        drawArticlePic(ctx, canvas, canvas_width, canvas_height, imgPath) {
            return new Promise((resolve, reject) => {
                wx.getImageInfo({
                    src: imgPath,
                    success: function (res) {

                        let img_width = res.width,   //图片宽
                            img_height = res.height;  //图片高
                        let clip_left, clip_top, //左偏移值,上偏移值,
                            clip_width, clip_height, //截取宽度,截取高度
                            dWidth, dHeight,
                            dx, dy;
                        //  ----------------裁剪图片----------------
                        if (img_width > canvas_width && img_height > canvas_height) {    //图片宽高都大于框框
                            if (img_width / img_height > 1 || img_width / img_height === 1) {   //宽图 或正方形
                                //将图高度自适应
                                let factor = img_height / canvas_height
                                clip_width = factor * canvas_width
                                clip_height = img_height
                                clip_top = 0
                                clip_left = (img_width - clip_width) / 2  //clip_left 取中间值
                            } else if (img_width / img_height < 1) {   //长图
                                let factor = img_width / canvas_width  // 图片宽/框宽= 裁剪长度/框长
                                // img_width = canvas_width
                                clip_width = img_width
                                clip_height = factor * canvas_height
                                clip_top = (img_height - clip_height) / 2
                                clip_left = 0  //clip_left 取中间值
                            }
                            dWidth = canvas_width
                            dHeight = canvas_height
                            dx = 15
                            dy = 95
                        } else if (img_width <= canvas_width) {   //宽度小于等于框
                            dWidth = img_width
                            if (img_height >= canvas_height) {
                                dHeight = canvas_height
                                clip_top = (img_height - canvas_height) / 2  //裁掉多余的高
                                dy = 0    //起始位置
                            } else {
                                dHeight = img_height
                                dy = (canvas_height - img_height) / 2
                                clip_top = 0
                            }
                            dx = 0
                            clip_left = 0

                        } else if (img_height <= canvas_height) {  //高度小于等于框
                            dHeight = img_height
                            if (img_width >= canvas_width) {
                                dWidth = canvas_width
                                dx = 0
                                clip_left = (img_width - canvas_width) / 2
                            } else {
                                dWidth = img_width
                                dx = (canvas_width - img_width) / 2
                                clip_left = 0
                            }
                            dy = 0
                            clip_top = 0
                        }
                        const img = canvas.createImage()
                        img.src = imgPath
                        img.onload = (res) => {
                            console.log('img_load', res)
                            ctx.drawImage(img, clip_left, clip_top, clip_width, clip_height, dx, dy, dWidth, dHeight);
                            return resolve()
                        }
                    }
                })
            })
        },
        roundRectColor(context, x, y, w, h, r, canvasWidth) {  //绘制圆角矩形(纯色填充)
            context.save();
            context.fillStyle = '#3B3B3B';
            context.strokeStyle = '#3B3B3B'
            context.lineJoin = 'round';  //交点设置成圆角
            context.lineWidth = r;
            context.strokeRect(x + r / 2, y + r / 2, w - r, h - r);
            context.fillRect(x + r, y + r, w - r * 2, h - r * 2);
            context.stroke();
            context.closePath();
            context.textAlign = 'center'
            context.fillStyle = '#ffffff'
            context.font = '10px PingFang'       // 文字字号:22px
            context.fillText('长按扫码加入', canvasWidth / 2, 265)
        },
        savePic() {
            let canvasPicPath = this.data.canvasPicPath
            console.log('canvasPicPath', canvasPicPath)
            wx.saveImageToPhotosAlbum({
                filePath: canvasPicPath,
                success(res) {
                    console.log('保存成功', res)
                    wx.showToast({
                        title: '保存成功'
                    })
                }
            })
        },
        closeWindow: function () {
            console.log('ad')
            wx.switchTab({
                url: 'pages/user/user'
            })
        },
        getCodeImg() {
            let link = this.properties.link
            let sceneStr = this.properties.sceneStr
            console.log('link', link)
            console.log('sceneStr', sceneStr)
            let randomStr = this.randomString(32)
            console.log('randomStr', randomStr)
            // return
            request(`/item/share/getVxQRCode?sceneStr=${sceneStr}&url=${link}`, {}).then(res => {
                console.log('res', res)
            })
        },
        randomString(len) {
            //默认去掉了容易混淆的字符oOLl,9gq,Vv,Uu,I1
            let chars = 'ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678';
            let tempLen = chars.length, tempStr = '';
            for (let i = 0; i < len; ++i) {
                tempStr += chars.charAt(Math.floor(Math.random() * tempLen));
            }
            return tempStr;
        },
        drawText(context, t, x, y, w) {    //canvas中文本换行
            let chr = t.split("");
            let temp = "";
            let row = [];
            console.log('chr', chr)
            for (let a = 0; a < chr.length; a++) {
                if (context.measureText(temp).width < w) {
                    ;
                } else {
                    row.push(temp);
                    temp = "";
                }
                temp += chr[a];
            }

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