<template> <view class="uqrcode" :class="{ 'uqrcode-hide': hide }" :style="{ width: `${templateOptions.width}px`, height: `${templateOptions.height}px` }"> <view class="uqrcode-canvas-wrapper"> <!-- 画布 --> <!-- #ifndef APP-NVUE --> <canvas class="uqrcode-canvas" :id="canvasId" :canvas-id="canvasId" :type="canvasType" :style="{ width: `${templateOptions.canvasWidth}px`, height: `${templateOptions.canvasHeight}px`, transform: templateOptions.canvasTransform }" v-if="templateOptions.canvasDisplay" @click="onClick"></canvas> <!-- #endif --> <!-- nvue用gcanvas --> <!-- #ifdef APP-NVUE --> <gcanvas class="uqrcode-canvas" ref="gcanvas" :style="{ width: `${templateOptions.canvasWidth}px`, height: `${templateOptions.canvasHeight}px` }" v-if="templateOptions.canvasDisplay" @click="onClick"></gcanvas> <!-- #endif --> </view> <!-- 加载效果 --> <view class="uqrcode-makeing" v-if="loading === undefined ? makeing : loading"> <slot name="loading"> <image class="uqrcode-makeing-image" :style="{ width: `${templateOptions.size / 4}px`, height: `${templateOptions.size / 4}px` }" src=""> </image> </slot> </view> <!-- 错误处理 --> <view class="uqrcode-error" v-if="isError" @click="onClick"> <slot name="error" :error="error"> <text class="uqrcode-error-message">{{ error.errMsg }}</text> </slot> </view> <!-- H5保存提示 --> <!-- #ifdef H5 --> <view class="uqrcode-h5-save" v-if="isH5Save"> <slot name="h5save" :tempFilePath="tempFilePath"> <image class="uqrcode-h5-save-image" :src="tempFilePath"></image> <text class="uqrcode-h5-save-text">{{ h5SaveIsDownload ? '若保存失败,' : '' }}请长按二维码进行保存</text> </slot> <view class="uqrcode-h5-save-close" @click.stop="isH5Save = false"> <view class="uqrcode-h5-save-close-before"></view> <view class="uqrcode-h5-save-close-after"></view> </view> </view> <!-- #endif --> </view> </template> <script> // #ifdef VUE3 import { toRaw } from 'vue'; // #endif /* 引入uQRCode核心js */ import UQRCode from '../../js_sdk/uqrcode/uqrcode'; /* 引入nvue所需模块 */ // #ifdef APP-NVUE import { enable, WeexBridge } from '../../js_sdk/gcanvas'; const modal = weex.requireModule('modal'); // #endif /* 引入队列 */ import { queueDraw, queueLoadImage } from '../../common/queue'; /* 引入缓存图片 */ import { cacheImageList } from '../../common/cache'; let instance = null; export default { name: 'uqrcode', props: { /** * canvas组件id */ canvasId: { type: String, required: true // canvasId在微信小程序初始值不能为空,created中赋值也不行,必须给一个值,否则挂载组件后无法绘制。不考虑用随机id,uuid }, /** * 二维码内容 */ value: { type: [String, Number] }, /** * 选项 */ options: { type: Object, default: () => { return {}; } }, /** * 二维码大小 */ size: { type: [String, Number], default: 200 }, /** * 二维码尺寸单位 */ sizeUnit: { type: String, default: 'px' }, /** * 导出的文件类型 */ fileType: { type: String, default: 'png' }, /** * 是否初始化组件后就开始生成 */ start: { type: Boolean, default: true }, /** * 是否数据发生改变自动重绘 */ auto: { type: Boolean, default: true }, /** * 隐藏组件 */ hide: { type: Boolean, default: false }, /** * canvas 类型,微信小程序默认使用2d,非2d微信官方已放弃维护,问题比较多 * 注意:微信小程序type2d手机上正常,PC上微信内打开小程序toDataURL报错,看后期微信官方团队会不会做兼容,不兼容的话只能在自行判断在PC使用非2d,或者直接提示用户请在手机上操作,微信团队的海报中心小程序就是这么做的 */ type: { type: String, default: () => { // #ifdef MP-WEIXIN return '2d'; // #endif // #ifndef MP-WEIXIN return 'normal'; // #endif } }, /** * 队列绘制,主要针对NVue端 */ queue: { type: Boolean, default: false }, /** * 是否队列加载图片,可减少canvas发起的网络资源请求,节省服务器资源 */ isQueueLoadImage: { type: Boolean, default: false }, /** * loading态 */ loading: { type: Boolean, default: undefined }, /** * H5保存即自动下载(在支持的环境下),默认false为仅弹层提示用户需要长按图片保存,不会自动下载 */ h5SaveIsDownload: { type: Boolean, default: false }, /** * H5下载名称 */ h5DownloadName: { type: String, default: 'uQRCode' } }, data() { return { canvas: undefined, canvasType: undefined, canvasContext: undefined, makeDelegate: undefined, drawDelegate: undefined, toTempFilePathDelegate: undefined, makeExecuted: false, makeing: false, drawing: false, isError: false, error: undefined, isH5Save: false, tempFilePath: '', templateOptions: { size: 0, width: 0, // 组件宽度 height: 0, canvasWidth: 0, // canvas宽度 canvasHeight: 0, canvasTransform: '', canvasDisplay: false }, uqrcodeOptions: { data: '' }, plugins: [], makeingPattern: [ [ [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true] ], [ [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, false, false, false], [true, true, true, true, true, true, false, true, true, true], [true, true, true, true, true, true, false, true, true, true], [true, true, true, true, true, true, false, true, true, true] ], [ [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true], [true, true, true, true, true, true, true, false, false, false], [true, true, true, true, true, true, true, false, false, false], [true, true, true, true, true, true, true, false, false, false], [true, true, true, false, false, false, false, true, true, true], [true, true, true, false, false, false, false, true, true, true] ], [ [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, false, false, false, false, false, false, false], [true, true, true, false, false, false, false, false, false, false], [true, true, true, false, false, false, false, false, false, false], [true, true, true, false, false, false, false, false, false, false], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true], [true, true, true, true, true, true, true, true, true, true] ] ] }; }, watch: { type: { handler(val) { const types = ['2d']; if (types.includes(val)) { this.canvasType = val; } else { this.canvasType = undefined; } }, immediate: true }, value: { handler() { if (this.auto) { this.remake(); } } }, size: { handler() { if (this.auto) { this.remake(); } } }, options: { handler() { if (this.auto) { this.remake(); } }, deep: true }, makeing: { handler(val) { if (!val) { if (typeof this.toTempFilePathDelegate === 'function') { this.toTempFilePathDelegate(); } } } } }, mounted() { this.templateOptions.size = this.sizeUnit == 'rpx' ? uni.upx2px(this.size) : this.size; this.templateOptions.width = this.templateOptions.size; this.templateOptions.height = this.templateOptions.size; this.templateOptions.canvasWidth = this.templateOptions.size; this.templateOptions.canvasHeight = this.templateOptions.size; if (this.canvasType == '2d') { // #ifndef MP-WEIXIN this.templateOptions.canvasTransform = `scale(${this.templateOptions.size / this.templateOptions.canvasWidth}, ${this.templateOptions.size / this.templateOptions.canvasHeight})`; // #endif } else { this.templateOptions.canvasTransform = `scale(${this.templateOptions.size / this.templateOptions.canvasWidth}, ${this.templateOptions.size / this.templateOptions.canvasHeight})`; } if (this.start) { this.make(); } }, methods: { /** * 获取模板选项 */ getTemplateOptions() { var size = this.sizeUnit == 'rpx' ? uni.upx2px(this.size) : this.size; return deepReplace(this.templateOptions, { size, width: size, height: size }); }, /** * 获取插件选项 */ getUqrcodeOptions() { return deepReplace(this.options, { data: String(this.value), size: Number(this.templateOptions.size) }); }, /** * 重置画布 */ resetCanvas(callback) { this.templateOptions.canvasDisplay = false; this.$nextTick(() => { this.templateOptions.canvasDisplay = true; this.$nextTick(() => { callback && callback(); }); }); }, /** * 绘制二维码 */ async draw(callback = {}, isDrawDelegate = false) { if (typeof callback.success != 'function') { callback.success = () => {}; } if (typeof callback.fail != 'function') { callback.fail = () => {}; } if (typeof callback.complete != 'function') { callback.complete = () => {}; } if (this.drawing) { if (!isDrawDelegate) { this.drawDelegate = () => { this.draw(callback, true); }; return; } } else { this.drawing = true; } if (!this.canvasId) { console.error('[uQRCode]: canvasId must be set!'); this.isError = true; this.drawing = false; callback.fail({ errMsg: '[uQRCode]: canvasId must be set!' }); callback.complete({ errMsg: '[uQRCode]: canvasId must be set!' }); return; } if (!this.value) { console.error('[uQRCode]: value must be set!'); this.isError = true; this.drawing = false; callback.fail({ errMsg: '[uQRCode]: value must be set!' }); callback.complete({ errMsg: '[uQRCode]: value must be set!' }); return; } /* 组件数据 */ this.templateOptions = this.getTemplateOptions(); /* uQRCode选项 */ this.uqrcodeOptions = this.getUqrcodeOptions(); /* 纠错等级兼容字母写法 */ if (typeof this.uqrcodeOptions.errorCorrectLevel === 'string') { this.uqrcodeOptions.errorCorrectLevel = UQRCode.errorCorrectLevel[this.uqrcodeOptions.errorCorrectLevel]; } /* nvue不支持动态修改gcanvas尺寸,除nvue外,默认使用useDynamicSize */ // #ifndef APP-NVUE if (typeof this.options.useDynamicSize === 'undefined') { this.uqrcodeOptions.useDynamicSize = true; } // #endif // #ifdef APP-NVUE if (typeof this.options.useDynamicSize === 'undefined') { this.uqrcodeOptions.useDynamicSize = false; } // if (typeof this.options.drawReserve === 'undefined') { // this.uqrcodeOptions.drawReserve = true; // } // #endif /* 获取uQRCode实例 */ const qr = instance = new UQRCode(); /* 注册扩展 */ this.plugins.forEach(p => qr.register(p.plugin)); /* 设置uQRCode选项 */ qr.setOptions(this.uqrcodeOptions); /* 调用制作二维码方法 */ qr.make(); /* 获取canvas上下文 */ let canvasContext = null; // #ifndef APP-NVUE if (this.canvasType === '2d') { // #ifdef MP-WEIXIN /* 微信小程序获取canvas2d上下文方式 */ const canvas = (this.canvas = await new Promise(resolve => { uni .createSelectorQuery() .in(this) // 在组件内使用需要 .select(`#${this.canvasId}`) .fields({ node: true, size: true }) .exec(res => { resolve(res[0].node); }); })); canvasContext = this.canvasContext = canvas.getContext('2d'); /* 2d的组件设置宽高与实际canvas绘制宽高不是一个,打个比方,组件size=200,canvas.width设置为100,那么绘制出来就是100=200,组件size=400,canvas.width设置为800,绘制大小还是800=400,所以无需理会下方返回的dynamicSize是多少,按dpr重新赋值给canvas即可 */ this.templateOptions.canvasWidth = qr.size; this.templateOptions.canvasHeight = qr.size; this.templateOptions.canvasTransform = ''; /* 使用dynamicSize+scale,可以解决小块间出现白线问题,dpr可以解决模糊问题 */ const dpr = uni.getSystemInfoSync().pixelRatio; canvas.width = qr.dynamicSize * dpr; canvas.height = qr.dynamicSize * dpr; canvasContext.scale(dpr, dpr); /* 微信小程序获取图像方式 */ qr.loadImage = this.getLoadImage(function(src) { /* 小程序下获取网络图片信息需先配置download域名白名单才能生效 */ return new Promise((resolve, reject) => { const img = canvas.createImage(); img.src = src; img.onload = () => { resolve(img); }; img.onerror = err => { reject(err); }; }); }); // #endif // #ifndef MP-WEIXIN /* 非微信小程序不支持2d,切换回uniapp获取canvas上下文方式 */ canvasContext = this.canvasContext = uni.createCanvasContext(this.canvasId, this); /* 使用dynamicSize,可以解决小块间出现白线问题,再通过scale缩放至size,使其达到所设尺寸 */ this.templateOptions.canvasWidth = qr.dynamicSize; this.templateOptions.canvasHeight = qr.dynamicSize; this.templateOptions.canvasTransform = `scale(${this.templateOptions.size / this.templateOptions.canvasWidth}, ${this.templateOptions.size / this.templateOptions.canvasHeight})`; /* uniapp获取图像方式 */ qr.loadImage = this.getLoadImage(function(src) { return new Promise((resolve, reject) => { if (src.startsWith('http')) { uni.getImageInfo({ src, success: res => { resolve(res.path); }, fail: err => { reject(err); } }); } else { if (src.startsWith('.')) { console.error('[uQRCode]: 本地图片路径仅支持绝对路径!'); throw new Error('[uQRCode]: local image path only supports absolute path!'); } else { resolve(src); } } }); }); // #endif } else { /* uniapp获取canvas上下文方式 */ canvasContext = this.canvasContext = uni.createCanvasContext(this.canvasId, this); /* 使用dynamicSize,可以解决小块间出现白线问题,再通过scale缩放至size,使其达到所设尺寸 */ this.templateOptions.canvasWidth = qr.dynamicSize; this.templateOptions.canvasHeight = qr.dynamicSize; this.templateOptions.canvasTransform = `scale(${this.templateOptions.size / this.templateOptions.canvasWidth}, ${this.templateOptions.size / this.templateOptions.canvasHeight})`; /* uniapp获取图像方式 */ qr.loadImage = this.getLoadImage(function(src) { return new Promise((resolve, reject) => { /* getImageInfo在微信小程序的bug:本地路径返回路径会把开头的/或../移除,导致路径错误,解决方法:限制只能使用绝对路径 */ if (src.startsWith('http')) { uni.getImageInfo({ src, success: res => { resolve(res.path); }, fail: err => { reject(err); } }); } else { if (src.startsWith('.')) { console.error('[uQRCode]: 本地图片路径仅支持绝对路径!'); throw new Error('[uQRCode]: local image path only supports absolute path!'); } else { resolve(src); } } }); }); } // #endif // #ifdef APP-NVUE /* NVue获取canvas上下文方式 */ const gcanvas = this.$refs['gcanvas']; const canvas = enable(gcanvas, { bridge: WeexBridge }); canvasContext = this.canvasContext = canvas.getContext('2d'); /* NVue获取图像方式 */ qr.loadImage = this.getLoadImage(function(src) { return new Promise((resolve, reject) => { /* getImageInfo在nvue的bug:获取同一个路径的图片信息,同一时间第一次获取成功,后续失败,猜测是写入本地时产生文件写入冲突,所以没有返回,特别是对于网络资源 --- 已实现队列绘制,已解决此问题 */ if (src.startsWith('.')) { console.error('[uQRCode]: 本地图片路径仅支持绝对路径!'); throw new Error('[uQRCode]: local image path only supports absolute path!'); } else { uni.getImageInfo({ src, success: res => { resolve(res.path); }, fail: err => { reject(err); } }); } }); }); // #endif /* 设置uQRCode实例的canvas上下文 */ qr.canvasContext = canvasContext; /* 延时等待页面重新绘制完毕 */ setTimeout(() => { /* 从插件获取具体要调用哪一个扩展函数 */ var plugin = this.plugins.find(p => p.name == qr.style); var drawCanvasName = plugin ? plugin.drawCanvas : 'drawCanvas'; /* 虽然qr[drawCanvasName]是直接返回Promise的,但由于js内部this指向问题,故不能直接exec(qr[drawCanvasName])此方式执行,需要改成exec(() => qr[drawCanvasName]())才能正确获取this */ var drawCanvas; if (this.queue) { drawCanvas = () => queueDraw.exec(() => qr[drawCanvasName]()); // drawCanvas = () => queueDraw.exec(() => new Promise((resolve, reject) => { // setTimeout(() => { // qr[drawCanvasName]().then(resolve).catch(reject); // }, 1000); // })); } else { drawCanvas = () => qr[drawCanvasName](); } /* 调用绘制方法将二维码图案绘制到canvas上 */ drawCanvas() .then(() => { if (this.drawDelegate) { /* 高频重绘纠正 */ let delegate = this.drawDelegate; this.drawDelegate = undefined; delegate(); } else { this.drawing = false; callback.success(); } }) .catch(err => { console.log(err); if (this.drawDelegate) { /* 高频重绘纠正 */ let delegate = this.drawDelegate; this.drawDelegate = undefined; delegate(); } else { this.drawing = false; this.isError = true; callback.fail(err); } }) .finally(() => { callback.complete(); }); }, 300); }, /** * 生成二维码 */ make(callback = {}) { this.makeExecuted = true; this.makeing = true; this.isError = false; if (typeof callback.success != 'function') { callback.success = () => {}; } if (typeof callback.fail != 'function') { callback.fail = () => {}; } if (typeof callback.complete != 'function') { callback.complete = () => {}; } this.resetCanvas(() => { clearTimeout(this.makeDelegate); this.makeDelegate = setTimeout(() => { this.draw({ success: () => { setTimeout(() => { callback.success(); this.complete(true); }, 300); }, fail: err => { callback.fail(err); this.error = err; this.complete(false, err.errMsg); }, complete: () => { callback.complete(); this.makeing = false; } }); }, 300); }); }, /** * 重新生成 */ remake(callback) { this.$emit('change'); this.make(callback); }, /** * 生成完成 */ complete(success = true, errMsg = '') { if (success) { this.$emit('complete', { success }); } else { this.$emit('complete', { success, errMsg }); } }, /** * 导出临时路径 */ toTempFilePath(callback = {}) { if (typeof callback.success != 'function') { callback.success = () => {}; } if (typeof callback.fail != 'function') { callback.fail = () => {}; } if (typeof callback.complete != 'function') { callback.complete = () => {}; } if (!this.makeExecuted) { console.error('[uQRCode]: make() 方法从未调用!请先成功调用 make() 后再进行操作。'); var err = { errMsg: '[uQRCode]: make() method has never been executed! please execute make() successfully before operating.' }; callback.fail(err); callback.complete(err); return; } if (this.isError) { callback.fail(this.error); callback.complete(this.error); return; } if (this.makeing) { /* 如果还在生成状态,那当前操作将托管到委托,监听生成完成后再通过委托复调当前方法 */ this.toTempFilePathDelegate = () => { this.toTempFilePath(callback); }; return; } else { this.toTempFilePathDelegate = null; } // #ifndef APP-NVUE if (this.canvasType === '2d') { // #ifdef MP-WEIXIN try { let dataURL = null; // #ifdef VUE3 dataURL = toRaw(this.canvas) .toDataURL(); // #endif // #ifndef VUE3 dataURL = this.canvas.toDataURL(); // #endif callback.success({ tempFilePath: dataURL }); callback.complete({ tempFilePath: dataURL }); } catch (e) { callback.fail(e); callback.complete(e); } // #endif } else { uni.canvasToTempFilePath({ canvasId: this.canvasId, fileType: this.fileType, width: Number(this.templateOptions.canvasWidth), height: Number(this.templateOptions.canvasHeight), destWidth: Number(this.templateOptions.size), destHeight: Number(this.templateOptions.size), success: res => { callback.success(res); }, fail: err => { callback.fail(err); }, complete: () => { callback.complete(); } }, this ); } // #endif // #ifdef APP-NVUE const dpr = uni.getSystemInfoSync().pixelRatio; this.canvasContext.toTempFilePath( 0, 0, this.templateOptions.canvasWidth * dpr, this.templateOptions.canvasHeight * dpr, this.templateOptions.size * dpr, this.templateOptions.size * dpr, '', 1, res => { callback.success(res); callback.complete(res); } ); // #endif }, /** * 保存 */ save(callback = {}) { if (typeof callback.success != 'function') { callback.success = () => {}; } if (typeof callback.fail != 'function') { callback.fail = () => {}; } if (typeof callback.complete != 'function') { callback.complete = () => {}; } this.toTempFilePath({ success: res => { // #ifndef H5 if (this.canvasType === '2d') { // #ifdef MP-WEIXIN /* 需要将 data:image/png;base64, 这段去除 writeFile 才能正常打开文件,否则是损坏文件,无法打开 */ const reg = new RegExp('^data:image/png;base64,', 'g'); const dataURL = res.tempFilePath.replace(reg, ''); const fs = wx.getFileSystemManager(); const tempFilePath = `${wx.env.USER_DATA_PATH}/${new Date().getTime()}${ Math.random() .toString() .split('.')[1] }.png`; fs.writeFile({ filePath: tempFilePath, // 要写入的文件路径 (本地路径) data: dataURL, // base64图片 encoding: 'base64', success: res1 => { uni.saveImageToPhotosAlbum({ filePath: tempFilePath, success: res2 => { callback.success(res2); }, fail: err2 => { callback.fail(err2); }, complete: () => { callback.complete(); } }); }, fail: err => { callback.fail(err); }, complete: () => { callback.complete(); } }); // #endif } else { uni.saveImageToPhotosAlbum({ filePath: res.tempFilePath, success: res1 => { callback.success(res1); }, fail: err1 => { callback.fail(err1); }, complete: () => { callback.complete(); } }); } // #endif // #ifdef H5 /* 可以在电脑浏览器下载,移动端iOS不行,安卓微信浏览器不行,安卓外部浏览器可以 */ this.isH5Save = true; this.tempFilePath = res.tempFilePath; if (this.h5SaveIsDownload) { const aEle = document.createElement('a'); aEle.download = this.h5DownloadName; // 设置下载的文件名,默认是'下载' aEle.href = res.tempFilePath; document.body.appendChild(aEle); aEle.click(); aEle.remove(); // 下载之后把创建的元素删除 } callback.success({ errMsg: 'ok' }); callback.complete({ errMsg: 'ok' }); // #endif }, fail: err => { callback.fail(err); callback.complete(err); } }); }, /** * 注册click事件 */ onClick(e) { this.$emit('click', e); }, /** * 获取实例 */ getInstance() { return instance; }, /** * 注册扩展,组件仅支持注册type为style的drawCanvas扩展 * @param {Object} plugin */ registerStyle(plugin) { if (plugin.Type != 'style') { console.warn('[uQRCode]: registerStyle 仅支持注册 style 类型的扩展!'); return { errMsg: 'registerStyle 仅支持注册 style 类型的扩展!' }; } if (typeof plugin === 'function') { this.plugins.push({ plugin, name: plugin.Name, drawCanvas: plugin.DrawCanvas }); } }, getLoadImage(loadImage) { var that = this; if (typeof loadImage == 'function') { return function(src) { /* 判断是否是队列加载图片的 */ if (that.isQueueLoadImage) { /* 解决iOS APP||NVUE同时绘制多个二维码导致图片丢失需使用队列 */ return queueLoadImage.exec(() => { return new Promise((resolve, reject) => { setTimeout(() => { const cache = cacheImageList.find(x => x.src == src); if (cache) { resolve(cache.img); } else { loadImage(src) .then(img => { cacheImageList.push({ src, img }); resolve(img); }) .catch(err => { reject(err); }); } }, 10); }); }); } else { return loadImage(src); } }; } else { return function(src) { return Promise.resolve(src); }; } } } }; /** * 对象属性深度替换 * @param {Object} o 原始对象/默认对象/被替换的对象 * @param {Object} r 从这个对象里取值替换到o对象里 * @return {Object} 替换后的新对象 */ function deepReplace(o = {}, r = {}, c = false) { let obj; if (c) { // 从源替换 obj = o; } else { // 不替换源,copy一份备份来替换 obj = { ...o }; } for (let k in r) { var vr = r[k]; if (vr != undefined) { if (vr.constructor == Object) { obj[k] = this.deepReplace(obj[k], vr); } else if (vr.constructor == String && !vr) { obj[k] = obj[k]; } else { obj[k] = vr; } } } return obj; } </script> <style scoped> .uqrcode { position: relative; } .uqrcode-hide { position: fixed; left: 7500rpx; } .uqrcode-canvas { transform-origin: top left; } .uqrcode-makeing { position: absolute; top: 0; right: 0; bottom: 0; left: 0; z-index: 10; /* #ifndef APP-NVUE */ display: flex; /* #endif */ justify-content: center; align-items: center; } .uqrcode-makeing-image { /* #ifndef APP-NVUE */ display: block; max-width: 120px; max-height: 120px; /* #endif */ } .uqrcode-error { position: absolute; top: 0; right: 0; bottom: 0; left: 0; /* #ifndef APP-NVUE */ display: flex; /* #endif */ justify-content: center; align-items: center; } .uqrcode-error-message { font-size: 12px; color: #939291; } /* #ifdef H5 */ .uqrcode-h5-save { position: fixed; top: 0; right: 0; bottom: 0; left: 0; z-index: 100; background-color: rgba(0, 0, 0, 0.68); display: flex; flex-direction: column; justify-content: center; align-items: center; } .uqrcode-h5-save-image { width: 512rpx; height: 512rpx; padding: 32rpx; } .uqrcode-h5-save-text { margin-top: 20rpx; font-size: 32rpx; font-weight: 700; color: #ffffff; } .uqrcode-h5-save-close { position: relative; margin-top: 72rpx; width: 60rpx; height: 60rpx; border: 2rpx solid #ffffff; border-radius: 60rpx; padding: 10rpx; } .uqrcode-h5-save-close-before { position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%) rotate(45deg); width: 40rpx; height: 4rpx; background: #ffffff; } .uqrcode-h5-save-close-after { position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%) rotate(-45deg); width: 40rpx; height: 4rpx; background: #ffffff; } /* #endif */ </style>