I used uniApp to make H5 custom posters at the front end. I thought it was a simple canvas for common functions. Because I had done similar functions in H5 and small programs before, I directly started to do it.
Implementation function explanation:
The poster effect diagram, the above wechat avatar, nickname and the following copy, the TWO-DIMENSIONAL code is dynamic, that is to say, it needs to be generated at the back end, including the background picture, also need to be generated randomly, and then the user can save long press the interface, identify the TWO-DIMENSIONAL code, share the function.
Implementation:
Create a canvas and set the style to the canvas.
<canvas style="" canvas-id="myCanvas" id="myCanvas"></canvas>
canvas {
background-color: #fff;
border: 1px solid #d0d0d0;
width: 100vw;
height: 100vh;
position: absolute;
left: 100%;
}
Copy the code
Create a canvas instance and start painting.
var context = uni.createCanvasContext('myCanvas');
Copy the code
All the data is passed from the back end, and there are two scenarios for the image material,
1. If the image is allowed to cross domains, you can use uni.getImageInfo to get the image path and put it in the canvas.
2. If the image is not allowed to cross domains, convert it to base64 format and place it on the canvas using context.drawImage()
The above two pictures reported an error, is the picture does not support cross-domain caused by. The developer tool may be able to correctly canvas out and export as a picture, but mobile canvas export will report an error, this time you need to convert to Base64, first try to use code to transfer, if not, on the site online conversion, and then save base64 to a file.
Some people might think of putting images in the code package to make local images and get around the cross-domain problem, but WHEN I tried this method, I found that applet works, H5 does not, unfortunately. If you have successful H5 local picture canvas, you can also share your experience with me, thank you.
H5 long press to identify the picture function:
User long press the picture will have the system default popover to recognize or save the share, no code implementation.
Take a look at the complete implementation code below :(clean, annotated, like a like!)
<template>
<!--pages/poster/poster.wxml-->
<view class="page">
<canvas style="" canvas-id="myCanvas" id="myCanvas"></canvas>
<view class="container">
<image :src="tempFilePath" mode="widthFix"></image>
</view>
</view>
</template>
<script>
import {
IMAGES_PATH
} from '@/common/const.js';
import {
getPoster
} from '@/common/api.js';
import {
pathToBase64,
base64ToPath
} from '../../js_sdk/gsq-image-tools/image-tools/index.js'
import base64_img from "./base64_img.js";
export default {
data() {
return {
tempFilePath: ''
};
},
onLoad: function(options) {
var that = this;
var context = uni.createCanvasContext('myCanvas');
getPoster(options.id)
.then(res => {
var {
img,
avatar,
nickname,
avatar_content,
active_describe
} = res.data.data || {};
var right_bottom_txt_base64 =
""
pathToBase64(avatar)
.then(base64_avatar => {
// 背景底图
context.drawImage(base64_img[img], 0, 0, 200, 387, 400, 774)
// 用户昵称
context.setFillStyle('#233582');
context.fillText(nickname, 90, 132);
// 文字上面横线 1
context.setFillStyle('#233582')
context.fillRect(16, 146, 166, 1.2)
// 文字 判断长度换行处理
context.setFontSize(9)
if (active_describe.length > 18) {
let tt1 = active_describe.substr(0, 17);
let tt2 = active_describe.substr(17, active_describe.length - 1);
context.fillText(tt1, 16, 166);
context.fillText(tt2, 16, 181);
} else {
context.fillText(active_describe, 20, 166, 320);
}
// 文字下面横线 2
context.fillRect(16, 196, 166, 1.2)
// 右下角的文字用图片,因为在背景图的话会失真模糊
context.drawImage(right_bottom_txt_base64, 100, 230, 90, 30)
// 规定一个圆形的位置,里面放头像图片
let avatarurl_width = 40
let avatarurl_heigth = 40
let avatarurl_x = 80
let avatarurl_y = 80
context.arc(avatarurl_width / 2 + avatarurl_x, avatarurl_heigth / 2 + avatarurl_y,
avatarurl_width / 2,
0, Math.PI * 2, false)
context.clip()
context.drawImage(base64_avatar, avatarurl_x, avatarurl_y, avatarurl_width, avatarurl_heigth)
context.restore()
// 保存画布,生成图片指定大小的图片,并返回图片路径
// 有个坑 : H5端 Canvas 内绘制的图像需要支持跨域访问才能成功。(本代码的解决方案:把图片转成base64)
context.draw(true, () => {
uni.canvasToTempFilePath({
x: 0,
y: 0,
width: 200,
height: 387,
destWidth: getApp().windowWidth,
destHeight: getApp().windowHeight,
canvasId: 'myCanvas',
fileType: "jpg",
quality: 1,
success(res) {
console.log('绘制成功-------', res)
uni.hideLoading();
that.tempFilePath = res.tempFilePath;
},
fail(err) {
console.log('绘制失败', err)
}
});
});
})
})
},
methods: {
// 小程序的场景使用,小程序支持本地图片
getImageInfo(url) {
return new Promise(function(resolve, reject) {
uni.getImageInfo({
src: url,
success: function(res) {
console.log(res);
resolve(res);
},
fail: function(err) {
console.log(err);
uni.showToast({
title: '生成失败',
icon: 'none'
});
}
});
});
},
// 小程序的场景使用,长按保存图片
saveImg(e) {
let url = this.tempFilePath;
uni.saveFile({
tempFilePath: url,
success: function(res) {
var savedFilePath = res.savedFilePath;
uni.showToast({
title: "图片保存成功",
icon: "none"
});
}
});
},
// 小程序场景使用,获取图片信息,保存到相册
downImage() {
const url = this.images1;
uni.getImageInfo({
src: url,
success: res => {
let path = res.path;
uni.saveImageToPhotosAlbum({
filePath: path,
success: res => {
uni.showToast({
title: '保存成功',
icon: 'none'
});
},
fail: res => {
console.log(res);
}
});
},
fail: res => {
console.log(res);
}
});
}
}
};
</script>
<style>
canvas {
background-color: #fff;
border: 1px solid #d0d0d0;
width: 100vw;
height: 100vh;
position: absolute;
left: 100%;
}
.container {
width: 100vw;
align-items: center;
overflow: auto;
background: #fefefe;
}
.container image {
width: 100%;
position: absolute;
top: 0;
}
</style>
Copy the code
App.vue sets global constants, screen width and height
onLaunch: function(info) { var that=this; uni.getSystemInfo({ success: function (res) { that.windowWidth =res.windowWidth; that.windowHeight =res.windowHeight; console.log('----this---www',this.windowWidth); }}); // network(); console.log('App Launch'); },Copy the code
./base64_img.js code, which put base64 pictures, is the background of the array, the backend random generation of 1~5, take the subscript canvas on the line.
‘.. /.. /js_sdk/ gSQ-image-tools /image-tools/index.js’ code, image to base64 function package, no need to modify.
function getLocalFilePath(path) { if (path.indexOf('_www') === 0 || path.indexOf('_doc') === 0 || path.indexOf('_documents') === 0 || path.indexOf('_downloads') === 0) { return path } if (path.indexOf('file://') === 0) { return path } if (path.indexOf('/storage/emulated/0/') === 0) { return path } if (path.indexOf('/') === 0) { var localFilePath = plus.io.convertAbsoluteFileSystem(path) if (localFilePath ! == path) { return localFilePath } else { path = path.substr(1) } } return '_www/' + path } export function pathToBase64(path) { return new Promise(function(resolve, reject) { if (typeof window === 'object' && 'document' in window) { if (typeof FileReader === 'function') { var xhr = new XMLHttpRequest() xhr.open('GET', path, true) xhr.responseType = 'blob' xhr.onload = function() { if (this.status === 200) { let fileReader = new FileReader() fileReader.onload = function(e) { resolve(e.target.result) } fileReader.onerror = reject fileReader.readAsDataURL(this.response) } } xhr.onerror = reject xhr.send() return } var canvas = document.createElement('canvas') var c2x = canvas.getContext('2d') var img = new Image img.onload = function() { canvas.width = img.width canvas.height = img.height c2x.drawImage(img, 0, 0) resolve(canvas.toDataURL()) canvas.height = canvas.width = 0 } img.onerror = reject img.src = path return } if (typeof plus === 'object') { plus.io.resolveLocalFileSystemURL(getLocalFilePath(path), function(entry) { entry.file(function(file) { var fileReader = new plus.io.FileReader() fileReader.onload = function(data) { resolve(data.target.result) } fileReader.onerror = function(error) { reject(error) } fileReader.readAsDataURL(file) }, function(error) { reject(error) }) }, function(error) { reject(error) }) return } if (typeof wx === 'object' && wx.canIUse('getFileSystemManager')) { wx.getFileSystemManager().readFile({ filePath: path, encoding: 'base64', success: function(res) { resolve('data:image/png; base64,' + res.data) }, fail: function(error) { reject(error) } }) return } reject(new Error('not support')) }) } export function base64ToPath(base64) { return new Promise(function(resolve, reject) { if (typeof window === 'object' && 'document' in window) { base64 = base64.split(',') var type = base64[0].match(/:(.*?) ; /)[1] var str = atob(base64[1]) var n = str.length var array = new Uint8Array(n) while (n--) { array[n] = str.charCodeAt(n) } return resolve((window.URL || window.webkitURL).createObjectURL(new Blob([array], { type: type }))) } var extName = base64.match(/data\:\S+\/(\S+); /) if (extName) { extName = extName[1] } else { reject(new Error('base64 error')) } var fileName = Date.now() + '.' + extName if (typeof plus === 'object') { var bitmap = new plus.nativeObj.Bitmap('bitmap' + Date.now()) bitmap.loadBase64Data(base64, function() { var filePath = '_doc/uniapp_temp/' + fileName bitmap.save(filePath, {}, function() { bitmap.clear() resolve(filePath) }, function(error) { bitmap.clear() reject(error) }) }, function(error) { bitmap.clear() reject(error) }) return } if (typeof wx === 'object' && wx.canIUse('getFileSystemManager')) { var filePath = wx.env.USER_DATA_PATH + '/' + fileName wx.getFileSystemManager().writeFile({ filePath: filePath, data: base64.replace(/^data:\S+\/\S+; base64,/, ''), encoding: 'base64', success: function() { resolve(filePath) }, fail: function(error) { reject(error) } }) return } reject(new Error('not support')) }) }Copy the code
Complete!!!!!