utils.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. /**
  2. * 在矩形内绘制矩形
  3. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  4. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  5. * @param {number} startXPercent - 内部矩形起始点横坐标,范围从 0 到 100
  6. * @param {number} startYPercent - 内部矩形起始点纵坐标,范围从 0 到 100
  7. * @param {number} widthPercent - 内部矩形宽度,范围从 0 到 100
  8. * @param {number} heightPercent - 内部矩形高度,范围从 0 到 100
  9. * @param {string} color - 填充颜色,可选参数,默认为黑色
  10. */
  11. const drawRectInRect = (ctx, rect, startXPercent, startYPercent, widthPercent, heightPercent, color = 'black') => {
  12. // 计算内部矩形的起始坐标
  13. const startX = rect.x + (startXPercent / 100) * rect.width;
  14. const startY = rect.y + (startYPercent / 100) * rect.height;
  15. // 计算内部矩形的宽度和高度
  16. const width = (widthPercent / 100) * rect.width;
  17. const height = (heightPercent / 100) * rect.height;
  18. ctx.fillStyle = color; // 设置填充颜色
  19. ctx.fillRect(startX, startY, width, height); // 绘制矩形
  20. return { x:startX, y:startY, width, height };
  21. };
  22. /**
  23. * 在矩形内绘制文本
  24. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  25. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  26. * @param {string} text - 要绘制的文本
  27. * @param {number} startXPercent - 文字起始横坐标,范围从 0 到 100
  28. * @param {number} startYPercent - 文字起始纵坐标,范围从 0 到 100
  29. * @param {number} fontSizePercent - 文字大小,范围从 0 到 100,基于矩形高度
  30. */
  31. const drawTextInRect = (ctx, rect, text, startXPercent, startYPercent, fontSizePercent) => {
  32. // 计算文字起始坐标
  33. const startX = rect.x + (startXPercent / 100) * rect.width;
  34. const startY = rect.y + (startYPercent / 100) * rect.height;
  35. // 计算文字大小
  36. const fontSize = (fontSizePercent / 100) * rect.height;
  37. ctx.font = `${fontSize}px sans-serif`; // 设置字体样式
  38. ctx.fillText(text, startX, startY);
  39. };
  40. /**
  41. * 在矩形内绘制线
  42. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  43. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  44. * @param {number} startXPercent - 起始点横坐标,范围从 0 到 100
  45. * @param {number} startYPercent - 起始点纵坐标,范围从 0 到 100
  46. * @param {number} endXPercent - 结束点横坐标,范围从 0 到 100
  47. * @param {number} endYPercent - 结束点纵坐标,范围从 0 到 100
  48. * @param {string} color - 线条颜色,可选参数,默认为黑色
  49. * @param {number} lineWidth - 线条宽度,可选参数,默认为1像素
  50. */
  51. const drawLineInRect = (ctx, rect, startXPercent, startYPercent, endXPercent, endYPercent, color = 'black', lineWidth = 1) => {
  52. // 计算起始点坐标
  53. const startX = rect.x + (startXPercent / 100) * rect.width;
  54. const startY = rect.y + (startYPercent / 100) * rect.height;
  55. // 计算结束点坐标
  56. const endX = rect.x + (endXPercent / 100) * rect.width;
  57. const endY = rect.y + (endYPercent / 100) * rect.height;
  58. // 设置线条样式
  59. ctx.strokeStyle = color; // 设置线条颜色
  60. ctx.lineWidth = lineWidth; // 设置线条宽度
  61. // 绘制线
  62. ctx.beginPath();
  63. ctx.moveTo(startX, startY);
  64. ctx.lineTo(endX, endY);
  65. ctx.stroke();
  66. };
  67. /**
  68. * 在矩形内绘制点
  69. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  70. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  71. * @param {number} xPercent - 点的横坐标,范围从 0 到 100
  72. * @param {number} yPercent - 点的纵坐标,范围从 0 到 100
  73. * @param {string} color - 点的颜色,可选参数,默认为黑色
  74. * @param {number} radius - 点的半径,可选参数,默认为1像素
  75. */
  76. const drawPointInRect = (ctx, rect, xPercent, yPercent, color = 'black', radius = 1) => {
  77. // 计算点的坐标
  78. const x = rect.x + (xPercent / 100) * rect.width;
  79. const y = rect.y + (yPercent / 100) * rect.height;
  80. // 设置点的样式
  81. ctx.fillStyle = color; // 设置点的颜色
  82. // 开始绘制路径
  83. ctx.beginPath();
  84. ctx.arc(x, y, radius, 0, Math.PI * 2); // 绘制一个圆
  85. ctx.fill(); // 填充圆
  86. };
  87. /**
  88. * 在矩形内绘制图片
  89. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  90. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  91. * @param {HTMLImageElement} img - 要绘制的图片对象
  92. * @param {number} imgXPercent - 图片起始横坐标,范围从 0 到 100
  93. * @param {number} imgYPercent - 图片起始纵坐标,范围从 0 到 100
  94. * @param {number} imgWidthPercent - 图片宽度百分比,范围从 0 到 100
  95. * @param {number} imgHeightPercent - 图片高度百分比,范围从 0 到 100
  96. */
  97. const drawImageInRect = (ctx, rect, img, imgXPercent, imgYPercent, imgWidthPercent, imgHeightPercent) => {
  98. // 计算图片的起始坐标
  99. const imgX = rect.x + (imgXPercent / 100) * rect.width;
  100. const imgY = rect.y + (imgYPercent / 100) * rect.height;
  101. // 计算图片的宽度和高度
  102. const imgWidth = (imgWidthPercent / 100) * rect.width;
  103. const imgHeight = (imgHeightPercent / 100) * rect.height;
  104. // 开始绘制图片
  105. ctx.drawImage(img, imgX, imgY, imgWidth, imgHeight);
  106. };
  107. /**
  108. * 在矩形内绘制图片,带圆角和白色边框
  109. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  110. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  111. * @param {HTMLImageElement} img - 要绘制的图片对象
  112. * @param {number} imgXPercent - 图片起始横坐标,范围从 0 到 100
  113. * @param {number} imgYPercent - 图片起始纵坐标,范围从 0 到 100
  114. * @param {number} imgWidthPercent - 图片宽度百分比,范围从 0 到 100
  115. * @param {number} imgHeightPercent - 图片高度百分比,范围从 0 到 100
  116. * @param {number} borderRadius - 圆角半径,以像素为单位
  117. * @param {number} borderWidth - 边框宽度,以像素为单位
  118. */
  119. const drawBorderImageInRect = (ctx, rect, img, imgXPercent, imgYPercent, imgWidthPercent, imgHeightPercent, borderRadius, borderWidth) => {
  120. // 计算图片的起始坐标
  121. const imgX = rect.x + (imgXPercent / 100) * rect.width + borderWidth;
  122. const imgY = rect.y + (imgYPercent / 100) * rect.height + borderWidth;
  123. // 计算图片的宽度和高度
  124. const imgWidth = (imgWidthPercent / 100) * rect.width - 2 * borderWidth;
  125. const imgHeight = (imgHeightPercent / 100) * rect.height - 2 * borderWidth;
  126. // 保存画布状态
  127. ctx.save();
  128. // 开始绘制圆角矩形的边框
  129. ctx.beginPath();
  130. ctx.moveTo(imgX + borderRadius, imgY);
  131. ctx.lineTo(imgX + imgWidth - borderRadius, imgY);
  132. ctx.arcTo(imgX + imgWidth, imgY, imgX + imgWidth, imgY + borderRadius, borderRadius);
  133. ctx.lineTo(imgX + imgWidth, imgY + imgHeight - borderRadius);
  134. ctx.arcTo(imgX + imgWidth, imgY + imgHeight, imgX + imgWidth - borderRadius, imgY + imgHeight, borderRadius);
  135. ctx.lineTo(imgX + borderRadius, imgY + imgHeight);
  136. ctx.arcTo(imgX, imgY + imgHeight, imgX, imgY + imgHeight - borderRadius, borderRadius);
  137. ctx.lineTo(imgX, imgY + borderRadius);
  138. ctx.arcTo(imgX, imgY, imgX + borderRadius, imgY, borderRadius);
  139. ctx.closePath();
  140. // 设置边框颜色和宽度
  141. ctx.strokeStyle = 'white';
  142. ctx.lineWidth = borderWidth;
  143. ctx.stroke();
  144. // 开始绘制圆角矩形的填充(可选,如果需要背景)
  145. // ctx.fillStyle = 'transparent'; // 或者设置其他背景颜色
  146. // ctx.fill();
  147. // 开始裁剪
  148. ctx.clip();
  149. // 绘制图片
  150. ctx.drawImage(img, imgX, imgY, imgWidth, imgHeight);
  151. // 恢复画布状态
  152. ctx.restore();
  153. };
  154. /**
  155. * 在矩形内绘制横向文本列表
  156. * @param {CanvasRenderingContext2D} ctx - 画布的上下文
  157. * @param {Object} rect - 矩形范围,包含 { x, y, width, height },以像素为单位
  158. * @param {Array<string>} texts - 要绘制的文本数组
  159. * @param {number} fontSizePercent - 文字大小,范围从 0 到 100,基于矩形高度
  160. * @param {number} startXPercent - 文字起始横坐标,范围从 0 到 100,基于矩形宽度
  161. * @param {number} startYPercent - 文字起始纵坐标,范围从 0 到 100,基于矩形高度
  162. * @param {number} columnSpacingPercent - 列间距,范围从 0 到 100,基于矩形宽度
  163. * @param {string} columnSeparator - 列间隔字符,默认为空字符串
  164. * @param {number} columnSeparatorWidthPercent - 列间隔字符的宽度百分比,范围从 0 到 100,基于矩形宽度
  165. * @param {number} separatorFontSizePercent - 分隔符字号大小,范围从 0 到 100,基于矩形高度
  166. * @param {string} separatorColor - 分隔符颜色,默认为黑色
  167. * @param {number} separatorMarginLeftPercent - 分隔符左边的边距,范围从 0 到 100,基于矩形宽度
  168. * @param {number} separatorMarginRightPercent - 分隔符右边的边距,范围从 0 到 100,基于矩形宽度
  169. */
  170. const drawHorizontalTextList = (ctx, rect,color='white', texts, fontSizePercent, startXPercent = 0, startYPercent = 50, columnSpacingPercent = 0, columnSeparator = '', columnSeparatorWidthPercent = 0, separatorFontSizePercent = fontSizePercent, separatorColor = 'black', separatorMarginLeftPercent = 0, separatorMarginRightPercent = 0) => {
  171. // 计算文字大小
  172. const fontSize = (fontSizePercent / 100) * rect.height;
  173. ctx.font = `${fontSize}px sans-serif`; // 设置字体样式
  174. // 计算文字的总宽度
  175. let totalTextWidth = 0;
  176. texts.forEach(text => {
  177. totalTextWidth += ctx.measureText(text).width;
  178. });
  179. // 计算分隔符的字体大小
  180. const separatorFontSize = (separatorFontSizePercent / 100) * rect.height;
  181. // 计算分隔符的总宽度
  182. ctx.font = `${separatorFontSize}px sans-serif`; // 设置分隔符字体样式
  183. const separatorWidth = (columnSeparatorWidthPercent / 100) * rect.width;
  184. const totalSeparatorWidth = (texts.length - 1) * separatorWidth;
  185. // 计算分隔符的左右边距
  186. const separatorMarginLeft = (separatorMarginLeftPercent / 100) * rect.width;
  187. const separatorMarginRight = (separatorMarginRightPercent / 100) * rect.width;
  188. // 计算可用的总间距
  189. const totalSpacing = (columnSpacingPercent / 100) * rect.width;
  190. // 计算所有文字和间隔字符以及间距的总宽度
  191. const totalWidth = totalTextWidth + totalSeparatorWidth + totalSpacing + (texts.length - 1) * (separatorMarginLeft + separatorMarginRight);
  192. // 计算实际的起始坐标,使得文本和间隔字符居中于矩形
  193. const startX = rect.x + (startXPercent / 100) * rect.width + (rect.width - totalWidth) / 2;
  194. const startY = rect.y + (startYPercent / 100) * rect.height - (fontSize / 2); // 绘制文字,纵坐标居中
  195. // 绘制文本和间隔字符
  196. let currentX = startX;
  197. texts.forEach((text, index) => {
  198. ctx.font = `${fontSize}px sans-serif`; // 设置文字字体样式
  199. ctx.fillStyle = color; // 设置文字颜色
  200. ctx.fillText(text, currentX, startY + (fontSize / 2)); // 绘制文字,纵坐标居中
  201. currentX += ctx.measureText(text).width;
  202. // 如果不是最后一列,则绘制间隔字符
  203. if (index < texts.length - 1) {
  204. // 添加左边距
  205. currentX += separatorMarginLeft;
  206. // 设置分隔符颜色和字体样式
  207. ctx.fillStyle = separatorColor;
  208. ctx.font = `${separatorFontSize}px sans-serif`;
  209. // 绘制分隔符
  210. ctx.fillText(columnSeparator, currentX, startY + (separatorFontSize / 2));
  211. // 添加右边距
  212. currentX += separatorWidth + separatorMarginRight + columnSpacingPercent / 100 * rect.width;
  213. }
  214. });
  215. };
  216. export { drawTextInRect, drawLineInRect, drawPointInRect ,drawImageInRect,drawBorderImageInRect, drawRectInRect, drawHorizontalTextList};