index.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496
  1. <template>
  2. <view class="CANVAS_DRAWER">
  3. <xinyu-cross-canvas :width.sync="widthTemp" :height.sync="heightTemp" :styleWidth="widthTemp"
  4. :styleHeight="heightTemp" id="CANVAS_DRAWER_TEMP" ref="CANVAS_DRAWER_TEMP"></xinyu-cross-canvas>
  5. <xinyu-cross-canvas :width.sync="width" :height.sync="height" :styleWidth="width" :styleHeight="height"
  6. id="CANVAS_DRAWER" ref="CANVAS_DRAWER"></xinyu-cross-canvas>
  7. </view>
  8. </template>
  9. <script>
  10. /**
  11. * xinyu-canvas-drawer Canvas绘制器
  12. * @description 本组件可用于所有需要进行Canvas绘图的场景。绘制的Canvas组件会被放置到屏幕外面,不会显示在屏幕上。调用draw方法后返回值为图片的base64。
  13. * 【注意】本组件需要使用ref调用draw方法并await!
  14. * @property {Number} width 待绘制的实际图片的宽度,不可以中途修改宽度与高度
  15. * @property {Number} height 待绘制的实际图片的高度,不可以中途修改宽度与高度
  16. * @example
  17. * <xinyu-canvas-drawer ref="poster" :width="750" :height="847"></xinyu-canvas-drawer>
  18. */
  19. import XinyuCrossCanvas from "./xinyu-cross-canvas/xinyu-cross-canvas.vue";
  20. import QRCode from "./qrcode/QRCode.js";
  21. export default {
  22. name: 'xinyu-canvas-drawer',
  23. props: {
  24. width: Number, //待绘制的实际画布的宽度数值,实际画布宽度以px计
  25. height: Number //待绘制的实际画布的高度数值,实际画布高度以px计
  26. },
  27. data() {
  28. return {
  29. widthTemp: 255,
  30. heightTemp: 255,
  31. backgroundColor: "", //背景色,请使用setBackgroundColor方法设置。如果该值为空,则表示该canvas没有画背景色。
  32. waitingList: [] //所有待渲染数据。只有在调用draw方法时才会进行渲染。
  33. };
  34. },
  35. components: {
  36. XinyuCrossCanvas
  37. },
  38. methods: {
  39. /**
  40. * init方法
  41. * @description 本方法为初始化海报绘制组件方法,初始化后本组件的所有方法才可使用
  42. * @example await init()
  43. * @return {VueComponent} 当前实例对象,以便链式调用。
  44. */
  45. async init() {
  46. let canvas = this.$refs.CANVAS_DRAWER;
  47. let canvasTemp = this.$refs.CANVAS_DRAWER_TEMP;
  48. while (!(canvas && canvasTemp)) { //兼容字节跳动小程序
  49. canvas = this.$refs.CANVAS_DRAWER;
  50. canvasTemp = this.$refs.CANVAS_DRAWER_TEMP;
  51. await new Promise((recv) => setTimeout(recv, 200));
  52. }
  53. await canvas.init();
  54. await canvasTemp.init();
  55. this.canvas = canvas;
  56. this.canvas_temp = canvasTemp;
  57. return this;
  58. },
  59. /**
  60. * setBackgroundColor方法
  61. * @description 本方法设置待绘制的背景色
  62. * @property {String} color 待绘制的背景颜色
  63. * @example setBackgroundColor("#FFFFFF")
  64. * @return {VueComponent} 当前实例对象,以便链式调用。
  65. */
  66. setBackgroundColor(color) {
  67. this.backgroundColor = color;
  68. return this;
  69. },
  70. /**
  71. * addImage方法
  72. * @description 本方法用于向Canvas上添加图片(添加的层级会按照链式顺序从下到上叠加,下同)
  73. * @property {String} image 待绘制的图片,网络图片与本地图片均可(本地图片需要使用require("@/static/...")方式引入),网络图片在微信小程序中必须是要在微信download中信任的域名下的图片。
  74. * @property {Number} x 待绘制的图片左上角的实际画布坐标X
  75. * @property {Number} y 待绘制的图片左上角的实际画布坐标Y
  76. * @property {Number} w 待绘制的图片在实际画布中的宽度
  77. * @property {Number} h 待绘制的图片在实际画布中的高度
  78. * @property {Boolean} isRound 是否是圆图,该项为真时直径为w,h参数将没有意义
  79. * @example addImage("https://www.baidu.com/img/flexible/logo/pc/result.png",0,0,500,700);
  80. * @return {VueComponent} 当前实例对象,以便链式调用。
  81. */
  82. addImage(image, x, y, w, h, isRound) {
  83. this.waitingList.push({
  84. type: "image",
  85. data: {
  86. image,
  87. x,
  88. y,
  89. w,
  90. h,
  91. isRound: !!isRound
  92. }
  93. });
  94. return this;
  95. },
  96. /**
  97. * addQRCode方法
  98. * @description 本方法用于向Canvas上添加二维码
  99. * @property {String} text 待生成二维码的文本。
  100. * @property {Number} x 待绘制的二维码左上角的实际画布坐标X
  101. * @property {Number} y 待绘制的二维码左上角的实际画布坐标Y
  102. * @property {Number} w 待绘制的二维码在实际画布中的宽度
  103. * @property {Number} h 待绘制的二维码在实际画布中的高度
  104. * @property {Object} extraConfig QRCode的额外配置项
  105. * @example addQRCode("测试生成",0,0,500,700);
  106. * @return {VueComponent} 当前实例对象,以便链式调用。
  107. */
  108. addQRCode(text, x, y, w, h, extraConfig) {
  109. if (!extraConfig)
  110. extraConfig = {};
  111. this.waitingList.push({
  112. type: "qrcode",
  113. data: {
  114. text,
  115. x,
  116. y,
  117. w,
  118. h,
  119. extraConfig
  120. }
  121. });
  122. return this;
  123. },
  124. /**
  125. * addText方法
  126. * @description 本方法用于向Canvas上添加文本
  127. * @property {String} text 待绘制的文本,如果带有换行符\n或宽度超过maxWidth且isWrap为真时会根据lineHeight进行换行。如果宽度超过maxWidth但isWrap为假时超出部分会被省略号...代替。
  128. * @property {Number} x 待绘制的文本左上角的实际画布坐标X
  129. * @property {Number} y 待绘制的文本左上角的实际画布坐标Y
  130. * @property {Number} size 待绘制的文本大小(单位与画布实际大小一致,为px)
  131. * @property {String} color 待绘制的文本颜色
  132. * @property {Number} maxWidth 待绘制的文本在实际画布中的限制宽度(单位与画布实际大小一致,为px),如果超过此宽度则根据isWrap进行换行或省略
  133. * @property {Boolean} isWrap 待绘制的文本在超出限制宽度时是否换行,为真时换行,否则省略。
  134. * @property {Number} lineHeight 待绘制的文本距离的顶部Y坐标下一行顶部Y坐标的距离(单位与画布实际大小一致,为px)。
  135. * @example addText("百度一下,你就知道", 19, 708, 34, "#1A59FE", 453, false, 40);
  136. * @return {VueComponent} 当前实例对象,以便链式调用。
  137. */
  138. addText(text, x, y, size, color, maxWidth, isWrap, lineHeight) {
  139. if (!maxWidth)
  140. maxWidth = 99999;
  141. if (!lineHeight)
  142. lineHeight = size;
  143. this.waitingList.push({
  144. type: "text",
  145. data: {
  146. text,
  147. x,
  148. y,
  149. size,
  150. color,
  151. maxWidth,
  152. isWrap: !!isWrap,
  153. lineHeight: lineHeight
  154. }
  155. });
  156. return this;
  157. },
  158. /**
  159. * addRect方法
  160. * @description 本方法用于向Canvas上添加矩形,通常用于设计图的背景色区域填充
  161. * @property {Number} x 待绘制的矩形左上角的实际画布坐标X
  162. * @property {Number} y 待绘制的矩形左上角的实际画布坐标Y
  163. * @property {Number} w 待绘制的矩形在实际画布中的宽度
  164. * @property {Number} h 待绘制的矩形在实际画布中的高度
  165. * @property {String} color 待绘制的矩形颜色
  166. * @example addRect(0, 690, 750, 158, "#FEFEFE")
  167. * @return {VueComponent} 当前实例对象,以便链式调用。
  168. */
  169. addRect(x, y, w, h, color) {
  170. this.waitingList.push({
  171. type: "rect",
  172. data: {
  173. x,
  174. y,
  175. w,
  176. h,
  177. color
  178. }
  179. });
  180. return this;
  181. },
  182. /**
  183. * addCustom方法
  184. * @description 本方法用于满足自定义绘制的需求。
  185. * 【注意】不要在该方法中进行绘制(调用draw方法),绘制应交给draw方法最终统一绘制。
  186. * @property {Function} func 回调函数,参数为当前Canvas的Context对象。考虑到异步情况,请为该方法返回Promise对象或将该方法设置为async方法。
  187. * @example addCustom(async (canvas)=>{
  188. await canvas.setContextProp("fillStyle", "#000000");
  189. await canvas.callContextMethod("moveTo", [10, 10]);
  190. await canvas.callContextMethod("rect", [10, 10, 100, 50]);
  191. await canvas.callContextMethod("lineTo", [110, 60]);
  192. await canvas.callContextMethod("stroke", []);
  193. });
  194. * @return {VueComponent} 当前实例对象,以便链式调用。
  195. */
  196. addCustom(func) {
  197. this.waitingList.push({
  198. type: "custom",
  199. data: func
  200. });
  201. return this;
  202. },
  203. /**
  204. * calcTextLinesWithNewLine方法
  205. * @description 本方法用于获取文本在实际绘制时的所有行数组,用户可根据行数乘以行高计算出完整高度。
  206. * @property {String} text 待绘制的文本,如果带有换行符\n或宽度超过maxWidth且isWrap为真时会根据lineHeight进行换行。如果宽度超过maxWidth但isWrap为假时超出部分会被省略号...代替。
  207. * @property {Number} size 待绘制的文本大小(单位与画布实际大小一致,为px)
  208. * @property {Number} maxWidth 待绘制的文本在实际画布中的限制宽度(单位与画布实际大小一致,为px),如果超过此宽度则根据isWrap进行换行或省略
  209. * @property {Boolean} isWrap 待绘制的文本在超出限制宽度时是否换行,为真时换行,否则省略。
  210. * @example addText("百度一下,你就知道", 34, 453, false);
  211. * @return {Array} 包含每行文字的数组。例:["百度一下,你","就知道"]。当isWrap为false时数组中的最后一个字符串可能会以...结尾。
  212. */
  213. async calcTextLinesWithNewLine(text, size, maxWidth, isWrap) {
  214. let lines = [];
  215. let arr = text.split("\n");
  216. for (let i = 0; i < arr.length; i++) {
  217. let line = arr[i];
  218. lines = lines.concat(await this.calcTextLines(line, size, maxWidth, isWrap));
  219. }
  220. return lines;
  221. },
  222. /**
  223. * calcTextLines方法
  224. * @description 本方法为不转换换行符时获取所有行。本方法为组件私有方法,请不要调用。
  225. * @property {String} text 待绘制的文本,如果带有换行符\n或宽度超过maxWidth且isWrap为真时会根据lineHeight进行换行。如果宽度超过maxWidth但isWrap为假时超出部分会被省略号...代替。
  226. * @property {Number} size 待绘制的文本大小(单位与画布实际大小一致,为px)
  227. * @property {Number} maxWidth 待绘制的文本在实际画布中的限制宽度(单位与画布实际大小一致,为px),如果超过此宽度则根据isWrap进行换行或省略
  228. * @property {Boolean} isWrap 待绘制的文本在超出限制宽度时是否换行,为真时换行,否则省略。
  229. * @example calcTextLines("百度一下,你就知道", 34, 453, false);
  230. * @return {Array} 包含每行文字的数组。例:["百度一下,你","就知道"]。当isWrap为false时数组中的最后一个字符串可能会以...结尾。
  231. */
  232. async calcTextLines(text, size, maxWidth, isWrap) {
  233. await this.canvas.setContextProp('font', size + 'px sans-serif');
  234. let charArr = text.split("");
  235. let ret = [];
  236. if (!isWrap)
  237. maxWidth -= (await this.canvas.callContextMethod("measureText", ["..."])).width;
  238. while (charArr.length != 0) {
  239. let i;
  240. for (i = 0; i < charArr.length; i++) {
  241. let w = (await this.canvas.callContextMethod("measureText", [charArr.slice(0, i + 1).join(
  242. "")]))
  243. .width;
  244. if (w > maxWidth) {
  245. break;
  246. }
  247. }
  248. ret.push(charArr.splice(0, i + 1).join(""));
  249. if (!isWrap) {
  250. if (charArr.length != 0)
  251. return ret + "...";
  252. else
  253. return ret;
  254. }
  255. }
  256. return ret;
  257. },
  258. /**
  259. * calcTextWidth方法
  260. * @description 可以通过本方法获取对应文本绘制后的宽度。
  261. * @property {String} text 待绘制的文本,如果带有换行符\n或宽度超过maxWidth且isWrap为真时会根据lineHeight进行换行。如果宽度超过maxWidth但isWrap为假时超出部分会被省略号...代替。
  262. * @property {Number} size 待绘制的文本大小(单位与画布实际大小一致,为px)
  263. * @example calcTextWidth("百度一下,你就知道", 34);
  264. * @return {Number} 文字的宽度。
  265. */
  266. async calcTextWidth(text, size) {
  267. await this.canvas.setContextProp('font', size + 'px sans-serif');
  268. return (await this.canvas.callContextMethod("measureText", [text])).width;
  269. },
  270. /**
  271. * clear方法
  272. * @description 本方法用于重置Canvas。
  273. * @example clear();
  274. * @return {VueComponent} 当前实例对象,以便链式调用。
  275. */
  276. async clear() {
  277. await this.canvas.callContextMethod("clearRect", [0, 0, this.width, this.height]);
  278. this.backgroundColor = "";
  279. this.waitingList = [];
  280. this.src = "";
  281. return this;
  282. },
  283. /**
  284. * getImageInfo方法
  285. * @description 本方法用于获取图片的宽高等信息,本地图片不能使用该方法。
  286. * @property {String} src 图片完整url。
  287. * @example getImageInfo("https://www.baidu.com/img/flexible/logo/pc/result.png");
  288. * @return {Promise} Promise对象,成功时返回uni.getImageInfo的success情况下的回调对象,否则throw错误。
  289. */
  290. async getImageInfo(src) {
  291. let that = this;
  292. return await new Promise((recv, recj) => {
  293. uni.getImageInfo({
  294. src: src,
  295. success: (res) => {
  296. if (res.errMsg == 'getImageInfo:ok') {
  297. res.key = src;
  298. recv(JSON.parse(JSON.stringify(res)));
  299. } else
  300. recj(res.errMsg);
  301. },
  302. fail(e) {
  303. recj(e);
  304. }
  305. });
  306. });
  307. },
  308. async loadImage(src) {
  309. return await this.canvas.loadImage(src);
  310. },
  311. /**
  312. * draw方法
  313. * @description 本方法用于实际加载网络图片及异步绘制。可以await该方法实现体验优化。
  314. * @example draw();
  315. * @return {Promise} 返回Promise对象,当图片下载错误或渲染错误时该方法会throw错误,请务必使用try catch来捕获!
  316. */
  317. async draw() {
  318. let sid = 1;
  319. let list = [];
  320. for (let wid = 0; wid < this.waitingList.length; wid++) {
  321. let item = this.waitingList[wid];
  322. if (item.type == "image") {
  323. let ret = JSON.parse(JSON.stringify(item));
  324. if (ret.data.isRound) {
  325. let d = Math.min(ret.data.w, ret.data.h);
  326. let r = Math.floor(d / 2);
  327. await this.canvas_temp.refreshWidthHeight(d, d);
  328. await this.canvas_temp.callContextMethod('save', []);
  329. await this.canvas_temp.callContextMethod('clearRect', [0, 0, d, d]);
  330. await this.canvas_temp.callContextMethod('arc', [r, r, r, 0, 2 * Math.PI]);
  331. await this.canvas_temp.callContextMethod('fill', []);
  332. await this.canvas_temp.callContextMethod('clip', []);
  333. await this.canvas_temp.callContextMethod('drawImage', [ret.data.image, 0, 0, d, d]);
  334. ret.data.image = await this.canvas_temp.callContextMethod('toDataURL', []);
  335. await this.canvas_temp.callContextMethod('restore', []);
  336. }
  337. list.push(ret);
  338. } else if (item.type == "custom") {
  339. let t = JSON.parse(JSON.stringify(item));
  340. t.data = item.data;
  341. list.push(t);
  342. } else if (item.type == "qrcode") {
  343. let config = {
  344. x: 0,
  345. y: 0,
  346. width: 256,
  347. height: 256
  348. };
  349. for (let i in item.data.extraConfig)
  350. config[i] = item.data.extraConfig[i];
  351. await this.canvas_temp.refreshWidthHeight(256, 256);
  352. await this.canvas_temp.callContextMethod('clearRect', [0, 0, 256, 256]);
  353. await this.canvas_temp.setContextProp('fillStyle', "#FFFFFF");
  354. await this.canvas_temp.callContextMethod('fillRect', [0, 0, 256, 256]);
  355. let wh = await (new QRCode(this.canvas_temp, config)).calcCode(item.data.text);
  356. await this.canvas_temp.refreshWidthHeight(wh.width, wh.height);
  357. await this.canvas_temp.callContextMethod('clearRect', [0, 0, 256, 256]);
  358. let ret = JSON.parse(JSON.stringify(item));
  359. await (new QRCode(this.canvas_temp, config)).makeCode(item.data.text);
  360. ret.data.image = await this.canvas_temp.callContextMethod('toDataURL', []);
  361. list.push(ret);
  362. } else
  363. list.push(JSON.parse(JSON.stringify(item)));
  364. };
  365. if (this.backgroundColor != "") {
  366. await this.canvas.setContextProp('fillStyle', this.backgroundColor);
  367. await this.canvas.callContextMethod('fillRect', [0, 0, this.width, this.height]);
  368. }
  369. for (let itemIndex = 0; itemIndex < list.length; itemIndex++) {
  370. let item = list[itemIndex];
  371. if (item.type == "image") {
  372. await this.canvas.callContextMethod('drawImage', [item.data.image, item.data.x, item.data.y,
  373. item.data.w, item.data.h
  374. ]);
  375. } else if (item.type == "text") {
  376. await this.canvas.setContextProp('textBaseline', 'top');
  377. await this.canvas.setContextProp('font', item.data.size + 'px sans-serif');
  378. await this.canvas.setContextProp('fillStyle', item.data.color);
  379. let textArr = await this.calcTextLinesWithNewLine(item.data.text, item.data.size, item.data
  380. .maxWidth, item.data.isWrap);
  381. for (let line = 0; line < textArr.length; line++)
  382. await this.canvas.callContextMethod('fillText', [textArr[line], item.data.x, item.data.y +
  383. line *
  384. item.data.lineHeight
  385. ]);
  386. } else if (item.type == "rect") {
  387. await this.canvas.setContextProp('fillStyle', item.data.color);
  388. await this.canvas.callContextMethod('fillRect', [item.data.x, item.data.y, item.data.w, item
  389. .data.h
  390. ]);
  391. } else if (item.type == "qrcode") {
  392. await this.canvas.callContextMethod('drawImage', [item.data.image, item.data.x, item.data.y,
  393. item.data.w, item.data.h
  394. ]);
  395. } else if (item.type == "custom")
  396. await item.data(this.canvas);
  397. };
  398. return await this.canvas.callContextMethod('toDataURL', []);
  399. },
  400. /**
  401. * saveImageToPhotosAlbum方法
  402. * @description 本方法用于将本实例的src保存到本地相册种。
  403. * @example saveImageToPhotosAlbum(src);
  404. * @return {Promise} 返回Promise对象,当发生错误时该方法会throw错误,请务必使用try catch来捕获!
  405. */
  406. saveImageToPhotosAlbum(src) {
  407. // #ifndef H5
  408. return new Promise(async (recv, recj) => {
  409. // #ifdef MP
  410. if (src.startsWith("data:image")) {
  411. let base64 = src.substring(src.indexOf(",") + 1);
  412. // #ifdef MP-WEIXIN
  413. let tmpFile = wx.env.USER_DATA_PATH + "/" + Date.now() + ".png";
  414. // #endif
  415. // #ifdef MP-ALIPAY
  416. let tmpFile = my.env.USER_DATA_PATH + "/" + Date.now() + ".png";
  417. // #endif
  418. // #ifdef MP-TOUTIAO
  419. let tmpFile = tt.env.USER_DATA_PATH + "/" + Date.now() + ".png";
  420. // #endif
  421. await new Promise((recv1) => {
  422. uni.getFileSystemManager().writeFile({
  423. filePath: tmpFile,
  424. data: base64,
  425. encoding: 'base64',
  426. success: recv1
  427. })
  428. });
  429. src = tmpFile;
  430. }
  431. // #endif
  432. // #ifdef APP-PLUS
  433. if (src.startsWith("data:image")) {
  434. const url = "_doc/" + Date.now() + ".png";
  435. const bitmap = new plus.nativeObj.Bitmap("base64");
  436. await new Promise((recv1) => bitmap.loadBase64Data(src, recv1));
  437. await new Promise((recv1) => bitmap.save(url, {
  438. overwrite: true
  439. }, () => {
  440. bitmap.clear();
  441. recv1();
  442. }, () => {
  443. bitmap.clear();
  444. recj();
  445. }));
  446. src = url;
  447. }
  448. // #endif
  449. uni.saveImageToPhotosAlbum({
  450. filePath: src,
  451. success: () => {
  452. recv();
  453. },
  454. fail: (e) => {
  455. recj(e);
  456. }
  457. });
  458. });
  459. // #endif
  460. // #ifdef H5
  461. return new Promise((recv, recj) => {
  462. let base64 = src;
  463. let arr = base64.split(',');
  464. let bytes = atob(arr[1]);
  465. let ab = new ArrayBuffer(bytes.length);
  466. let ia = new Uint8Array(ab);
  467. for (let i = 0; i < bytes.length; i++) {
  468. ia[i] = bytes.charCodeAt(i);
  469. }
  470. let blob = new Blob([ab], {
  471. type: 'application/octet-stream'
  472. });
  473. let url = URL.createObjectURL(blob);
  474. let a = document.createElement('a');
  475. a.href = url;
  476. a.download = new Date().valueOf() + ".png";
  477. let e = document.createEvent('MouseEvents');
  478. e.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0,
  479. null);
  480. a.dispatchEvent(e);
  481. URL.revokeObjectURL(url);
  482. recv();
  483. });
  484. // #endif
  485. }
  486. }
  487. }
  488. </script>
  489. <style scoped>
  490. .CANVAS_DRAWER {
  491. position: fixed;
  492. left: 750rpx;
  493. top: 0rpx;
  494. }
  495. </style>