1132 lines
		
	
	
		
			43 KiB
		
	
	
	
		
			Vue
		
	
	
	
		
		
			
		
	
	
			1132 lines
		
	
	
		
			43 KiB
		
	
	
	
		
			Vue
		
	
	
	
|  | <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> |