my-canvas.vue 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. <template>
  2. <view class=" hflex acenter jcenter" style="margin: 188rpx 0 0;">
  3. <canvas canvas-id="canvas" id="canvas" :style="{'width': width + 'px', 'minHeight': height + 'px'}"></canvas>
  4. </view>
  5. </template>
  6. <script>
  7. var context = {}
  8. export default {
  9. name: "drawImage",
  10. props: {
  11. // 绘制图片的尺寸
  12. imageSize: {
  13. type: Object,
  14. default: () => {},
  15. },
  16. // canvas绘制的数据
  17. canvasData: {
  18. type: Array,
  19. default: () => [],
  20. },
  21. // 是否开始绘制
  22. isDraw: {
  23. type: Boolean,
  24. default: false,
  25. },
  26. },
  27. data() {
  28. return {
  29. // 屏幕宽度
  30. screenWidth: 0,
  31. // canvas画布的宽度
  32. width: 305,
  33. // canvas画布的高度
  34. height: 324,
  35. // 当前图片放大倍数 - 清晰度
  36. count: 2,
  37. };
  38. },
  39. mounted() {
  40. // 这段代码主要是为了防止uni-app在app端IOS的问题,因为在IOS 画布过大可能会导致绘制空白 - 可以自行调整
  41. // #ifdef APP-PLUS
  42. if(uni.getSystemInfoSync().platform === 'ios') {
  43. this.count = 1;
  44. }
  45. // #endif
  46. },
  47. watch: {
  48. // 监听是否开始绘制
  49. isDraw: async function(newVal) {
  50. if(newVal) {
  51. this.getSystemInfo();
  52. this.getImageByCanvasData(this.canvasData);
  53. }
  54. }
  55. },
  56. onReady() {
  57. context = uni.createCanvasContext("canvas", this);
  58. console.log(typeof(context));
  59. this.getSystemInfo();
  60. this.getImageByCanvasData(this.canvasData);
  61. },
  62. methods: {
  63. /** 获取系统信息 */
  64. getSystemInfo() {
  65. const { screenWidth } = uni.getSystemInfoSync();
  66. // this.width = this.imageSize.width * this.count;
  67. // this.height = this.imageSize.height * this.count;
  68. this.screenWidth = screenWidth;
  69. },
  70. /**
  71. * 通过数据绘制图片
  72. * @param {array} data canvas绘制的数组
  73. * 格式:每一项的数据
  74. * { type: 'rect', attr: { color: '', x, y, width, height, radian_1, radian_2, radian_3, radian_4 } }
  75. * { type: 'image', attr: { image: '', x, y, width, height, radian_1, radian_2, radian_3, radian_4 } }
  76. * { type: 'text', attr: { text: '', x, y, color, size, weight, writingMode } }
  77. * */
  78. async getImageByCanvasData(data) {
  79. // 获取canvas上下文对象
  80. console.log('开始绘制',context);
  81. console.log('创建canvas');
  82. // 清空画布
  83. // context.clearRect(0, 0, this.width * this.count, this.height * this.count);
  84. for(const item of data) {
  85. // 判断类型
  86. if(item.type === 'rect') {
  87. // 绘制圆边矩形
  88. this.drawRoundRectangular(
  89. context,
  90. item.attr.color,
  91. item.attr.x * this.count,
  92. item.attr.y * this.count,
  93. item.attr.width * this.count,
  94. item.attr.height * this.count,
  95. item.attr.radian_1 ? item.attr.radian_1 * this.count : 0,
  96. item.attr.radian_2 ? item.attr.radian_2 * this.count : -1,
  97. item.attr.radian_3 ? item.attr.radian_3 * this.count : -1,
  98. item.attr.radian_4 ? item.attr.radian_4 * this.count : -1
  99. );
  100. }
  101. else if(item.type === 'image' && item.attr.image) {
  102. // 绘制圆边图片
  103. await this.drawRoundImageToCanvas(
  104. context,
  105. item.attr.image,
  106. item.attr.x * this.count,
  107. item.attr.y * this.count,
  108. item.attr.width * this.count,
  109. item.attr.height * this.count,
  110. item.attr.radian_1 ? item.attr.radian_1 * this.count : 0,
  111. item.attr.radian_2 ? item.attr.radian_2 * this.count : -1,
  112. item.attr.radian_3 ? item.attr.radian_3 * this.count : -1,
  113. item.attr.radian_4 ? item.attr.radian_4 * this.count : -1
  114. );
  115. console.log("image",item.attr.image);
  116. }
  117. else if(item.type === 'text' && item.attr.text) {
  118. // 绘制文本
  119. this.drawTextToCanvas(
  120. context,
  121. item.attr.text,
  122. item.attr.x * this.count,
  123. item.attr.y * this.count,
  124. item.attr.color,
  125. parseInt(item.attr.size ? item.attr.size * this.count : 16 * this.count),
  126. item.attr.weight,
  127. item.attr.writingMode ? item.attr.writingMode : 'initial'
  128. );
  129. }
  130. else if(item.type === 'line') {
  131. // 画线
  132. this.drawLine(
  133. context,
  134. item.attr.color,
  135. item.attr.x,
  136. item.attr.y,
  137. item.attr.width,
  138. item.attr.height,
  139. );
  140. }
  141. }
  142. // 绘制图片
  143. context.draw(false, () => {
  144. uni.canvasToTempFilePath({
  145. canvasId: 'canvas',
  146. x: 0,
  147. y: 0,
  148. width: this.width,
  149. height: this.height,
  150. destWidth: this.width,
  151. height: this.height,
  152. success: res => {
  153. this.$emit("generateImageSuccessful", res.tempFilePath);
  154. },
  155. }, this);
  156. });
  157. },
  158. /**
  159. * 绘制文本
  160. * @param {string} context Canvase的实例
  161. * @param {string} text 文本内容
  162. * @param {number} x 矩形的x坐标
  163. * @param {number} y 矩形的y坐标
  164. * @param {number} color 文本颜色
  165. * @param {number} size 字体的大小
  166. * @param {string} weight 字体的粗细
  167. * @param {string} writingMode 字体的排列方式 - initial 水平 tb 垂直
  168. * */
  169. drawTextToCanvas(context, text, x, y, color = '#000', size = 16, weight = '400', writingMode = 'initial') {
  170. console.log('绘制文本',text);
  171. context.fillStyle = color;
  172. context.font = `normal ${weight} ${size}px sans-serif`;
  173. if(writingMode === 'tb') {
  174. const temp = text.split("");
  175. for(let i = 0; i < temp.length; i++) {
  176. context.fillText(temp[i], x, i * size + y);
  177. }
  178. }
  179. else {
  180. // 判断是否有换行符
  181. const temp = text.split("\n");
  182. for(let i = 0; i < temp.length; i++) {
  183. context.fillText(temp[i], x, i * size + y + i * size * 0.2); // i * size * 0.2 增加换行的间距
  184. }
  185. }
  186. },
  187. /**
  188. * 绘制圆边矩形
  189. * @param {string} context Canvase的实例
  190. * @param {string} color 填充的颜色
  191. * @param {number} x 矩形的x坐标
  192. * @param {number} y 矩形的y坐标
  193. * @param {number} width 矩形的宽度
  194. * @param {number} height 矩形的高度
  195. * @param {number} height 图片的高度
  196. * @param {number} radian_1 弧度大小 - radian_1 右上 的弧度, 1个参数代表全部
  197. * @param {number} radian_2 弧度大小 - radian_2 右下 的弧度
  198. * @param {number} radian_3 弧度大小 - radian_3 左下 的弧度
  199. * @param {number} radian_4 弧度大小 - radian_4 左上 的弧度
  200. * */
  201. drawRoundRectangular(context, color, x, y, width, height, radian_1 = 0, radian_2 = -1, radian_3 = -1, radian_4 = -1) {
  202. context.save();
  203. this.drawRoundPath(context, x, y, width, height, radian_1, radian_2, radian_3, radian_4);
  204. context.setFillStyle(color);
  205. context.fill();
  206. context.restore();
  207. },
  208. /**
  209. * 绘制圆角图片
  210. * @param {string} context Canvase的实例
  211. * @param {string} image 图片地址
  212. * @param {number} x 图片的x坐标
  213. * @param {number} y 图片的y坐标
  214. * @param {number} width 图片的宽度
  215. * @param {number} height 图片的高度
  216. * @param {number} radian_1 弧度大小 - radian_1 右上 的弧度, 1个参数代表全部
  217. * @param {number} radian_2 弧度大小 - radian_2 右下 的弧度
  218. * @param {number} radian_3 弧度大小 - radian_3 左下 的弧度
  219. * @param {number} radian_4 弧度大小 - radian_4 左上 的弧度
  220. * */
  221. async drawRoundImageToCanvas(context, image, x, y, width, height, radian_1 = 0, radian_2 = -1, radian_3 = -1, radian_4 = -1) {
  222. context.save();
  223. this.drawRoundPath(context, x, y, width, height, radian_1, radian_2, radian_3, radian_4);
  224. context.drawImage(await this.handleNetworkImgaeTransferTempImage(image), x, y, width, height);
  225. context.restore();
  226. },
  227. /**
  228. * 绘制圆边路径
  229. * @param {string} context Canvase的实例
  230. * @param {number} x 图片的x坐标
  231. * @param {number} y 图片的y坐标
  232. * @param {number} width 图片的宽度
  233. * @param {number} height 图片的高度
  234. * @param {number} radian_1 弧度大小 - radian_1 右上 的弧度, 1个参数代表全部
  235. * @param {number} radian_2 弧度大小 - radian_2 右下 的弧度
  236. * @param {number} radian_3 弧度大小 - radian_3 左下 的弧度
  237. * @param {number} radian_4 弧度大小 - radian_4 左上 的弧度
  238. * */
  239. drawRoundPath(context, x, y, width, height, radian_1 = 0, radian_2 = -1, radian_3 = -1, radian_4 = -1) {
  240. // 设置弧度
  241. radian_2 = radian_2 === -1 ? radian_1 : radian_2;
  242. radian_3 = radian_3 === -1 ? radian_1 : radian_3;
  243. radian_4 = radian_4 === -1 ? radian_1 : radian_4;
  244. // 创建路径 - 绘制带圆边的矩形
  245. context.beginPath();
  246. context.moveTo(x + width / 2, y);
  247. context.arcTo(x + width, y, x + width, y + height, radian_1);
  248. context.arcTo(x + width, y + height, x, y + height, radian_2);
  249. context.arcTo(x, y + height, x, y, radian_3);
  250. context.arcTo(x, y, x + width, y, radian_4);
  251. // 关闭路径 - 结束绘制
  252. context.closePath();
  253. context.strokeStyle = "transparent";
  254. context.stroke();
  255. context.clip();
  256. },
  257. drawLine(context, x, y, width, height,color) {
  258. context.beginPath();
  259. context.moveTo(x,x);
  260. context.lineTo(y, y);
  261. context.stroke();
  262. },
  263. /** 将网络图片变成临时图片 */
  264. handleNetworkImgaeTransferTempImage(url) {
  265. return new Promise(resolve => {
  266. if(url.indexOf('http') === 0) {
  267. uni.downloadFile({
  268. url,
  269. success: res => {
  270. resolve(res.tempFilePath);
  271. }
  272. });
  273. }
  274. else {
  275. resolve(url);
  276. }
  277. });
  278. },
  279. }
  280. }
  281. </script>
  282. <style scoped lang="scss">
  283. .canvas {
  284. position: absolute;
  285. top: 188rpx;
  286. left: 50rpx;
  287. width: 650rpx;
  288. min-height: 768rpx;
  289. background: #FFFFFF;
  290. border-radius: 20rpx;
  291. }
  292. </style>