1. 程式人生 > >小程式製作海報並分享到朋友圈

小程式製作海報並分享到朋友圈

新增畫布

首先,在小程式裡進行繪圖操作需要用到<canvas>元件,那我們就先在我們的wxml程式碼中放入如下的<canvas>

<canvas canvas-id="shareCanvas" style="width:600px;height:900px"></canvas>

這樣一來我們就有了一個600x900的繪圖區域。然後,我們要開始寫JS程式碼在這張畫布上進行繪圖操作。

步驟1:繪製背景圖

通過觀察《長城你造不造》合成的那張分享圖,我們可以分析得出它的組成主要有以下3個部分:一張大的背景圖,一段動態的文字(xxxx 喊你“一起來為修長城獻磚”),以及一個小程式碼圖片。

那麼我們就先找一張圖片來當做背景圖,將它畫到畫布上去,程式碼大致如下:

const wxGetImageInfo = promisify(wx.getImageInfo)

wxGetImageInfo({
    src: 'http://some-domain/bg.png'
}).then(res => {
    const ctx = wx.createCanvasContext('shareCanvas')
    ctx.drawImage(res.path, 0, 0, 600, 900)
    ctx.draw()
})

在這段程式碼中,我們通過使用wx.getImageInfo

這個API來下載一個網路圖片到本地(並可獲取該圖片的尺寸等其他資訊),然後呼叫ctx.drawImage方法將圖片繪製到畫布上,填滿畫布。

步驟2:繪製文字

接著,讓我們來在畫布上繼續繪製一段文字,一般這種宣傳用的分享圖,少不了文字描述,而且可能是根據場景內容不同而產生的動態資訊,比如可能是一篇文章的作者、文章的標題和內容。

我們嘗試下在畫布上新增一段居中顯示的文字:“作者:一斤程式碼”,還是基於前面的那段程式碼接著寫:

const wxGetImageInfo = promisify(wx.getImageInfo)

wxGetImageInfo({
    src: 'http://some-domain/bg.png'
}).then(res => { const ctx = wx.createCanvasContext('shareCanvas') // 底圖 ctx.drawImage(res.path, 0, 0, 600, 900) // 作者名稱 ctx.setTextAlign('center') // 文字居中 ctx.setFillStyle('#000000') // 文字顏色:黑色 ctx.setFontSize(22) // 文字字號:22px ctx.fillText(“作者:一斤程式碼”, 600 / 2, 500) ctx.stroke() ctx.draw() })

由於在canvas上繪製文字不會自動折行,如果要畫一段比較長的文字,可以考慮限制一行的字數,將長文字拆分成幾行來畫。

步驟3:繪製小程式碼

最後,我們在畫布最後新增一個小程式碼,可以是靜態的小程式碼,也可以是比如為每一篇文章動態生成的小程式碼(參考《微信小程式之生成自定義引數小程式二維碼》這篇文章),反正這個小程式碼也就是一張圖片,所以繪製方法跟繪製底圖差不多。最後的程式碼類似如此:

const wxGetImageInfo = promisify(wx.getImageInfo)

Promise.all([
    wxGetImageInfo({
        src: 'http://some-domain.com/background.png'
    }),
    wxGetImageInfo({
        src: 'http://some-domain.com/api/generate/qrcode'
    })
]).then(res => {
    const ctx = wx.createCanvasContext('shareCanvas')
    
    // 底圖
    ctx.drawImage(res[0].path, 0, 0, 600, 900)

    // 作者名稱
    ctx.setTextAlign('center')    // 文字居中
    ctx.setFillStyle('#000000')  // 文字顏色:黑色
    ctx.setFontSize(22)         // 文字字號:22px
    ctx.fillText(“作者:一斤程式碼”, 600 / 2, 500)

    // 小程式碼
    const qrImgSize = 180
    ctx.drawImage(res[1].path, (600 - qrImgSize) / 2, 530, qrImgSize, qrImgSize)

    ctx.stroke()
    ctx.draw()
})

這樣,差不多我們的分享圖就生成好了。

儲存到系統相簿

接著,我們要把它儲存進使用者的系統相簿中去,實現這個功能,我們主要靠wx.canvasToTempFilePathwx.saveImageToPhotosAlbum這兩個API。

主要的流程就是先通過wx.canvasToTempFilePath<canvas>上繪製的影象生成臨時檔案的形式,然後再通過wx.saveImageToPhotosAlbum進行儲存到系統相簿的操作。

const wxCanvasToTempFilePath = promisify(wx.canvasToTempFilePath)
const wxSaveImageToPhotosAlbum = promisify(wx.saveImageToPhotosAlbum)

wxCanvasToTempFilePath({
    canvasId: 'shareCanvas'
}, this).then(res => {
    return wxSaveImageToPhotosAlbum({
        filePath: res.tempFilePath
    })
}).then(res => {
    wx.showToast({
        title: '已儲存到相簿'
    })
})


其中promise.util.js如下:

/**
 * 將wx的callback形式的API轉換成支援Promise的形式
 */
module.exports = {

  promisify: api => {
    return (options, ...params) => {
      return new Promise((resolve, reject) => {
        const extras = {
          success: resolve,
          fail: reject
        }
        api({ ...options, ...extras }, ...params)
      })
    }
  }

}