forked from natuka/web.puabadge.com
8 changed files with 989 additions and 56 deletions
@ -0,0 +1,586 @@
@@ -0,0 +1,586 @@
|
||||
// textHelper.ts
|
||||
/** |
||||
* 文字定位助手模块 |
||||
* @module TextPosition |
||||
*/ |
||||
|
||||
/** |
||||
* 创建文字定位助手实例 |
||||
* @param {Object} options - 配置选项 |
||||
* @param {Function} options.setData - 设置数据的方法(用于更新响应式数据) |
||||
* @returns {Object} 包含renderText方法的对象 |
||||
*/ |
||||
export function createTextPosition({ setData }) { |
||||
if (!setData) { |
||||
throw new Error('setData is required'); |
||||
} |
||||
|
||||
// 默认样式配置
|
||||
const defaultStyle = { |
||||
fontSize: 16, // 字体大小(px)
|
||||
fontFamily: 'Arial', // 字体
|
||||
color: '#000000', // 文字颜色
|
||||
textAlign: 'left', // 文本对齐方式: left, center, right
|
||||
fontWeight: 'normal', // 字体粗细: normal, bold
|
||||
lineHeight: 1.5, // 行高
|
||||
opacity: 1, // 透明度
|
||||
backgroundColor: 'transparent', // 背景颜色
|
||||
padding: 0, // 内边距(px)
|
||||
borderRadius: 0, // 边框圆角(px)
|
||||
}; |
||||
|
||||
/** |
||||
* 渲染文字到指定位置 |
||||
* @param {Object} options - 文字配置 |
||||
* @param {string} options.text - 要显示的文字 |
||||
* @param {number} options.x - 文字左上角X坐标(基于原始容器尺寸) |
||||
* @param {number} options.y - 文字左上角Y坐标(基于原始容器尺寸) |
||||
* @param {number} [options.width] - 文字容器宽度(基于原始容器尺寸) |
||||
* @param {number} [options.height] - 文字容器高度(基于原始容器尺寸) |
||||
* @param {Object} [options.style] - 文字样式配置 |
||||
* @param {string} [options.bindKey] - 绑定到页面数据的键名 |
||||
*/ |
||||
|
||||
function roundText({ shapeImage, x, y, width, height, radius, style = {}, bindKey = 'roundTextConfig', text = '', isFront = true, cachedImageInfo = null, onImageInfoCached = null }) { |
||||
// 合并默认样式和用户自定义样式
|
||||
const mergedStyle = { ...defaultStyle, ...style }; |
||||
|
||||
// 获取设备信息,计算缩放比例
|
||||
const screenWidth = 320; |
||||
|
||||
// 计算宽度和高度的缩放比例
|
||||
const scale = 0.6; |
||||
|
||||
// 计算字符位置的函数
|
||||
const calculateTextPosition = (imageInfo) => { |
||||
// 计算图片宽高比
|
||||
const imageRatio = imageInfo.width / imageInfo.height; |
||||
const containerRatio = 1; // 容器是正方形,宽高比为1
|
||||
|
||||
let ratio; |
||||
|
||||
// 根据图片和容器的比例关系决定如何缩放
|
||||
if (imageRatio > containerRatio) { |
||||
// 图片更宽,以宽度为基准缩放
|
||||
ratio = screenWidth / imageInfo.width; |
||||
} else { |
||||
// 图片更高或等比,以高度为基准缩放
|
||||
ratio = screenWidth / imageInfo.height; |
||||
} |
||||
|
||||
// 计算缩放后的参数
|
||||
const scaledX = x * ratio * scale; |
||||
const scaledY = y * ratio * scale; |
||||
const scaledWidth = width * ratio * scale; |
||||
const scaledHeight = height * ratio * scale; |
||||
const scaledRadius = radius * ratio * scale; |
||||
const scaledFontSize = mergedStyle.fontSize * ratio * scale; |
||||
|
||||
// 计算圆心位置(基于容器的中心点)
|
||||
const centerX = scaledX + scaledWidth / 2; |
||||
const centerY = scaledY + scaledHeight / 2; |
||||
|
||||
// 计算每个字符的位置信息
|
||||
// 根据正面/背面使用不同的计算方法,互不影响
|
||||
const charList = isFront |
||||
? calculateFrontRoundTextChars({ |
||||
text: text, |
||||
centerX: scaledWidth / 2, // 相对于容器的中心
|
||||
centerY: scaledHeight / 2, |
||||
radius: scaledRadius, |
||||
fontSize: scaledFontSize, |
||||
fontWeight: mergedStyle.fontWeight || 'normal', |
||||
color: mergedStyle.color || '#000000' |
||||
}) |
||||
: calculateBackRoundTextChars({ |
||||
text: text, |
||||
centerX: scaledWidth / 2, // 相对于容器的中心
|
||||
centerY: scaledHeight / 2, |
||||
radius: scaledRadius, |
||||
fontSize: scaledFontSize, |
||||
fontWeight: mergedStyle.fontWeight || 'normal', |
||||
color: mergedStyle.color || '#000000' |
||||
}); |
||||
|
||||
// 构建圆形文字配置对象
|
||||
const roundTextConfig = { |
||||
x: scaledX, |
||||
y: scaledY, |
||||
width: scaledWidth, |
||||
height: scaledHeight, |
||||
centerX: centerX, |
||||
centerY: centerY, |
||||
radius: scaledRadius, |
||||
text: text, |
||||
charList: charList, // 字符位置列表
|
||||
style: { |
||||
...mergedStyle, |
||||
fontSize: scaledFontSize, |
||||
} |
||||
}; |
||||
|
||||
console.log('roundTextConfig===>', roundTextConfig); |
||||
|
||||
// 更新页面数据
|
||||
setData({ |
||||
[bindKey]: roundTextConfig |
||||
}); |
||||
|
||||
// 返回配置对象,以便后续使用
|
||||
return roundTextConfig; |
||||
}; |
||||
|
||||
// 如果已有缓存的图片信息,直接使用
|
||||
if (cachedImageInfo && cachedImageInfo.src === shapeImage) { |
||||
calculateTextPosition(cachedImageInfo); |
||||
return; |
||||
} |
||||
|
||||
// 否则获取图片信息
|
||||
const img = new Image(); |
||||
img.onload = () => { |
||||
const res = { |
||||
width: img.width, |
||||
height: img.height, |
||||
path: shapeImage, |
||||
src: shapeImage |
||||
}; |
||||
|
||||
// 如果提供了缓存回调,调用它
|
||||
if (typeof onImageInfoCached === 'function') { |
||||
onImageInfoCached(res); |
||||
} |
||||
|
||||
calculateTextPosition(res); |
||||
}; |
||||
img.onerror = (err) => { |
||||
console.error('Failed to get image info:', err); |
||||
}; |
||||
img.src = shapeImage; |
||||
} |
||||
|
||||
/** |
||||
* 计算正面圆形文字中每个字符的位置和样式 |
||||
* @param {Object} options - 配置选项 |
||||
* @param {string} options.text - 文字内容 |
||||
* @param {number} options.centerX - 圆心X坐标(相对于容器) |
||||
* @param {number} options.centerY - 圆心Y坐标(相对于容器) |
||||
* @param {number} options.radius - 半径 |
||||
* @param {number} options.fontSize - 字体大小 |
||||
* @param {string} options.fontWeight - 字体粗细 |
||||
* @param {string} options.color - 文字颜色 |
||||
* @returns {Array} 字符位置信息数组 |
||||
*/ |
||||
function calculateFrontRoundTextChars({ text, centerX, centerY, radius, fontSize, fontWeight, color }) { |
||||
if (!text) { |
||||
return []; |
||||
} |
||||
|
||||
// 计算字符宽度
|
||||
// 使用更精确的估算方法
|
||||
const charWidths = []; |
||||
let totalWidth = 0; |
||||
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
let charWidth; |
||||
|
||||
// 判断字符类型,使用不同的宽度估算
|
||||
if (/[\u4e00-\u9fa5]/.test(char)) { |
||||
// 中文字符:宽度约为 fontSize
|
||||
charWidth = fontSize; |
||||
} else if (/[0-9]/.test(char)) { |
||||
// 数字:宽度约为 fontSize * 0.6
|
||||
charWidth = fontSize * 0.6; |
||||
} else if (/[a-zA-Z]/.test(char)) { |
||||
// 英文字母:宽度约为 fontSize * 0.6(大写)或 0.5(小写)
|
||||
charWidth = /[A-Z]/.test(char) ? fontSize * 0.65 : fontSize * 0.5; |
||||
} else if (/[\u3000-\u303f\uff00-\uffef]/.test(char)) { |
||||
// 全角符号:宽度约为 fontSize
|
||||
charWidth = fontSize; |
||||
} else { |
||||
// 其他字符(半角符号等):宽度约为 fontSize * 0.4
|
||||
charWidth = fontSize * 0.4; |
||||
} |
||||
|
||||
// 如果字体是粗体,稍微增加宽度
|
||||
if (fontWeight === 'bold' || fontWeight === '700') { |
||||
charWidth *= 1.1; |
||||
} |
||||
|
||||
charWidths.push(charWidth); |
||||
totalWidth += charWidth; |
||||
} |
||||
|
||||
// 为文字之间添加2px间距(除了最后一个字符)
|
||||
const spacing = 2; |
||||
const totalSpacing = (text.length > 1) ? (text.length - 1) * spacing : 0; |
||||
const totalWidthWithSpacing = totalWidth + totalSpacing; |
||||
|
||||
// 计算文字总弧长和每个字符的角度(正面使用原始半径)
|
||||
const circumference = 2 * Math.PI * radius; |
||||
const arcLength = totalWidthWithSpacing; |
||||
const totalAngle = (arcLength / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算起始角度(使文字居中于底部)
|
||||
// 从底部中心开始,顺时针排列,起始角度 = 底部中心 + 总角度的一半
|
||||
const startAngle = Math.PI / 2 + totalAngle / 2; |
||||
|
||||
// 创建并定位每个字符(逆时针排列,所以角度递减)
|
||||
let currentAngle = startAngle; |
||||
const charList = []; |
||||
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
const charWidth = charWidths[i]; |
||||
|
||||
// 计算字符角度增量(包括字符宽度和间距)
|
||||
// 最后一个字符不需要添加后面的间距
|
||||
const spacingAfter = (i < text.length - 1) ? spacing : 0; |
||||
const charAngleIncrement = ((charWidth + spacingAfter) / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算字符角度位置(逆时针排列,所以角度递减)
|
||||
// 字符中心位置 = 当前角度 - 字符宽度的一半对应的角度
|
||||
const charWidthAngle = (charWidth / circumference) * 2 * Math.PI; |
||||
const charAngle = currentAngle - charWidthAngle / 2; |
||||
|
||||
// 计算字符位置(正面使用原始半径)
|
||||
const x = centerX + radius * Math.cos(charAngle); |
||||
const y = centerY + radius * Math.sin(charAngle) - fontSize / 3; |
||||
|
||||
// 计算旋转角度(正面:文字底部向外,顶部朝向圆心)
|
||||
const rotateAngle = charAngle + Math.PI * 1.5; |
||||
// 转换为度数(用于CSS transform)
|
||||
const rotateDeg = (rotateAngle * 180) / Math.PI; |
||||
|
||||
charList.push({ |
||||
char: char, |
||||
x: x, |
||||
y: y, |
||||
rotate: rotateAngle, |
||||
rotateDeg: rotateDeg, // 度数版本
|
||||
fontSize: fontSize, |
||||
fontWeight: fontWeight, |
||||
color: color |
||||
}); |
||||
|
||||
// 更新当前角度(逆时针排列,所以角度递减)
|
||||
currentAngle -= charAngleIncrement; |
||||
} |
||||
|
||||
return charList; |
||||
} |
||||
|
||||
/** |
||||
* 计算背面圆形文字中每个字符的位置和样式 |
||||
* @param {Object} options - 配置选项 |
||||
* @param {string} options.text - 文字内容 |
||||
* @param {number} options.centerX - 圆心X坐标(相对于容器) |
||||
* @param {number} options.centerY - 圆心Y坐标(相对于容器) |
||||
* @param {number} options.radius - 半径 |
||||
* @param {number} options.fontSize - 字体大小 |
||||
* @param {string} options.fontWeight - 字体粗细 |
||||
* @param {string} options.color - 文字颜色 |
||||
* @returns {Array} 字符位置信息数组 |
||||
*/ |
||||
function calculateBackRoundTextChars({ text, centerX, centerY, radius, fontSize, fontWeight, color }) { |
||||
if (!text) { |
||||
return []; |
||||
} |
||||
|
||||
// 计算字符宽度
|
||||
// 使用更精确的估算方法
|
||||
const charWidths = []; |
||||
let totalWidth = 0; |
||||
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
let charWidth; |
||||
|
||||
// 判断字符类型,使用不同的宽度估算
|
||||
if (/[\u4e00-\u9fa5]/.test(char)) { |
||||
// 中文字符:宽度约为 fontSize
|
||||
charWidth = fontSize; |
||||
} else if (/[0-9]/.test(char)) { |
||||
// 数字:宽度约为 fontSize * 0.6
|
||||
charWidth = fontSize * 0.6; |
||||
} else if (/[a-zA-Z]/.test(char)) { |
||||
// 英文字母:宽度约为 fontSize * 0.6(大写)或 0.5(小写)
|
||||
charWidth = /[A-Z]/.test(char) ? fontSize * 0.65 : fontSize * 0.5; |
||||
} else if (/[\u3000-\u303f\uff00-\uffef]/.test(char)) { |
||||
// 全角符号:宽度约为 fontSize
|
||||
charWidth = fontSize; |
||||
} else { |
||||
// 其他字符(半角符号等):宽度约为 fontSize * 0.4
|
||||
charWidth = fontSize * 0.4; |
||||
} |
||||
|
||||
// 如果字体是粗体,稍微增加宽度
|
||||
if (fontWeight === 'bold' || fontWeight === '700') { |
||||
charWidth *= 1.1; |
||||
} |
||||
|
||||
charWidths.push(charWidth); |
||||
totalWidth += charWidth; |
||||
} |
||||
|
||||
// 为文字之间添加2px间距(除了最后一个字符)
|
||||
const spacing = 1; |
||||
const totalSpacing = (text.length > 1) ? (text.length - 1) * spacing : 0; |
||||
const totalWidthWithSpacing = totalWidth + totalSpacing; |
||||
|
||||
// 计算文字总弧长和每个字符的角度
|
||||
// 背面:文字到圆心的距离 = 半径 + 文字高度的一半
|
||||
const adjustedRadius = radius + fontSize / 2; |
||||
const circumference = 2 * Math.PI * adjustedRadius; |
||||
const arcLength = totalWidthWithSpacing; |
||||
const totalAngle = (arcLength / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算起始角度(使文字居中于底部)
|
||||
// 从底部中心开始,顺时针排列,起始角度 = 底部中心 - 总角度的一半
|
||||
const startAngle = Math.PI / 2 - totalAngle / 2; |
||||
|
||||
// 创建并定位每个字符(顺时针排列,所以角度递增)
|
||||
let currentAngle = startAngle; |
||||
const charList = []; |
||||
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
const charWidth = charWidths[i]; |
||||
|
||||
// 计算字符角度增量(包括字符宽度和间距)
|
||||
// 最后一个字符不需要添加后面的间距
|
||||
const spacingAfter = (i < text.length - 1) ? spacing : 0; |
||||
const charAngleIncrement = ((charWidth + spacingAfter) / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算字符角度位置(顺时针排列,所以角度递增)
|
||||
// 字符中心位置 = 当前角度 + 字符宽度的一半对应的角度
|
||||
const charWidthAngle = (charWidth / circumference) * 2 * Math.PI; |
||||
const charAngle = currentAngle + charWidthAngle / 2; |
||||
|
||||
// 计算字符位置(背面:文字到圆心的距离 = 半径 + 文字高度的一半)
|
||||
const x = centerX + adjustedRadius * Math.cos(charAngle); |
||||
const y = centerY + adjustedRadius * Math.sin(charAngle); |
||||
|
||||
// 计算旋转角度(背面:文字底部朝向圆心,顶部向外)
|
||||
const rotateAngle = charAngle + Math.PI * 0.5; |
||||
// 转换为度数(用于CSS transform)
|
||||
const rotateDeg = (rotateAngle * 180) / Math.PI; |
||||
|
||||
charList.push({ |
||||
char: char, |
||||
x: x, |
||||
y: y, |
||||
rotate: rotateAngle, |
||||
rotateDeg: rotateDeg, // 度数版本
|
||||
fontSize: fontSize, |
||||
fontWeight: fontWeight, |
||||
color: color |
||||
}); |
||||
|
||||
// 更新当前角度(顺时针排列,所以角度递增)
|
||||
currentAngle += charAngleIncrement; |
||||
} |
||||
|
||||
return charList; |
||||
} |
||||
|
||||
/** |
||||
* 初始化圆形文字 Canvas |
||||
* @param {string} canvasId - Canvas 的 ID 或选择器 |
||||
* @param {Object} config - 配置对象,包含 width 和 height |
||||
* @returns {Promise} 返回包含 canvas 和 ctx 的对象 |
||||
*/ |
||||
function initRoundTextCanvas(canvasId, config) { |
||||
return new Promise((resolve) => { |
||||
const tryInit = () => { |
||||
// 如果是选择器字符串,去掉 # 或 . 前缀
|
||||
const selector = canvasId.startsWith('#') ? canvasId.slice(1) : |
||||
canvasId.startsWith('.') ? canvasId.slice(1) : canvasId; |
||||
|
||||
// 尝试通过 ID 查找
|
||||
let canvasElement: HTMLCanvasElement | null = document.getElementById(selector) as HTMLCanvasElement | null; |
||||
|
||||
// 如果找不到,尝试通过类名查找(取第一个)
|
||||
if (!canvasElement && canvasId.startsWith('.')) { |
||||
const elements = document.getElementsByClassName(selector); |
||||
if (elements.length > 0) { |
||||
canvasElement = elements[0] as HTMLCanvasElement; |
||||
} |
||||
} |
||||
|
||||
// 如果还是找不到,尝试直接使用 canvasId 作为选择器
|
||||
if (!canvasElement) { |
||||
canvasElement = document.querySelector(canvasId) as HTMLCanvasElement | null; |
||||
} |
||||
|
||||
if (canvasElement && canvasElement instanceof HTMLCanvasElement) { |
||||
const canvas = canvasElement; |
||||
const ctx = canvas.getContext('2d'); |
||||
if (!ctx) { |
||||
throw new Error('无法获取 Canvas 2D 上下文'); |
||||
} |
||||
const dpr = window.devicePixelRatio || 1; |
||||
|
||||
// 使用配置中的尺寸,如果没有则使用 Canvas 的实际尺寸或默认值
|
||||
const width = (config && config.width) || canvas.offsetWidth || 700; |
||||
const height = (config && config.height) || canvas.offsetHeight || 700; |
||||
|
||||
canvas.width = width * dpr; |
||||
canvas.height = height * dpr; |
||||
ctx.scale(dpr, dpr); |
||||
|
||||
resolve({ canvas, ctx }); |
||||
} else { |
||||
// 如果 Canvas 还未渲染,延迟重试
|
||||
setTimeout(() => { |
||||
tryInit(); |
||||
}, 100); |
||||
} |
||||
}; |
||||
tryInit(); |
||||
}); |
||||
} |
||||
|
||||
/** |
||||
* 在 Canvas 上绘制圆形文字 |
||||
* @param {Object} options - 绘制选项 |
||||
* @param {Object} options.ctx - Canvas 2D 上下文 |
||||
* @param {Object} options.config - 圆形文字配置对象 |
||||
* @param {Function} options.setData - 设置数据的方法 |
||||
* @param {HTMLCanvasElement} options.canvas - Canvas 元素 |
||||
* @param {string} options.imageKey - 图片数据键名,用于存储生成的图片路径 |
||||
* @returns {Promise} 返回生成的图片路径(Data URL) |
||||
*/ |
||||
function drawRoundTextOnCanvas({ ctx, config, setData, canvas, imageKey = 'roundTextImage' }) { |
||||
return new Promise((resolve, reject) => { |
||||
if (!ctx || !config) { |
||||
console.log('drawRoundTextOnCanvas: ctx 或 config 不存在', { ctx: !!ctx, config: !!config }); |
||||
if (setData) { |
||||
setData({ |
||||
[imageKey]: '' |
||||
}); |
||||
} |
||||
resolve(''); |
||||
return; |
||||
} |
||||
|
||||
const text = config.text; |
||||
if (!text) { |
||||
console.log('drawRoundTextOnCanvas: 文字为空'); |
||||
// 如果没有文字,清空图片
|
||||
if (setData) { |
||||
setData({ |
||||
[imageKey]: '' |
||||
}); |
||||
} |
||||
resolve(''); |
||||
return; |
||||
} |
||||
|
||||
console.log('drawRoundTextOnCanvas: 开始绘制文字', { text, config }); |
||||
|
||||
// 使用配置中的宽高
|
||||
const canvasWidth = config.width || 700; |
||||
const canvasHeight = config.height || 700; |
||||
// 圆心位置:根据需求,圆心应该在容器的中心
|
||||
// 由于容器就是 Canvas,所以圆心就是 Canvas 的中心
|
||||
const centerX = canvasWidth / 2; |
||||
const centerY = canvasHeight / 2; |
||||
const radius = config.radius; |
||||
const fontSize = config.style.fontSize; |
||||
const color = config.style.color || '#000000'; |
||||
const fontWeight = config.style.fontWeight || 'normal'; |
||||
|
||||
// 清空画布(注意:Canvas 的实际尺寸已经乘以了 dpr,但绘制时使用的是逻辑尺寸)
|
||||
ctx.clearRect(0, 0, canvasWidth, canvasHeight); |
||||
|
||||
// 设置字体样式
|
||||
ctx.font = `${fontWeight} ${fontSize}px sans-serif`; |
||||
ctx.fillStyle = color; |
||||
ctx.textAlign = 'center'; |
||||
ctx.textBaseline = 'middle'; // 使用 middle 基线,文字中心对齐,更适合圆形排列
|
||||
|
||||
// 计算每个字符的角度
|
||||
const circumference = 2 * Math.PI * radius; |
||||
const charWidths = []; |
||||
let totalWidth = 0; |
||||
|
||||
// 测量每个字符的宽度
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
const metrics = ctx.measureText(char); |
||||
const width = metrics.width; |
||||
charWidths.push(width); |
||||
totalWidth += width; |
||||
} |
||||
|
||||
// 计算文字总弧长和每个字符的角度
|
||||
const arcLength = totalWidth; |
||||
const totalAngle = (arcLength / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算起始角度(使文字居中于底部)
|
||||
// 从底部中心开始,顺时针排列,起始角度 = 底部中心 + 总角度的一半
|
||||
const startAngle = Math.PI / 2 + totalAngle / 2; |
||||
|
||||
// 创建并定位每个字符(逆时针排列,所以角度递减)
|
||||
let currentAngle = startAngle; |
||||
|
||||
for (let i = 0; i < text.length; i++) { |
||||
const char = text[i]; |
||||
const charWidth = charWidths[i]; |
||||
|
||||
// 计算字符角度增量
|
||||
const charAngleIncrement = (charWidth / circumference) * 2 * Math.PI; |
||||
|
||||
// 计算字符角度位置(逆时针排列,所以角度递减)
|
||||
const charAngle = currentAngle - charAngleIncrement / 2; |
||||
|
||||
// 计算字符位置
|
||||
const x = centerX + radius * Math.cos(charAngle); |
||||
const y = centerY + radius * Math.sin(charAngle); |
||||
|
||||
// 保存上下文
|
||||
ctx.save(); |
||||
|
||||
// 移动到字符位置
|
||||
ctx.translate(x, y); |
||||
|
||||
// 旋转文字,使其沿着圆形路径排列
|
||||
// 文字底部向外,顶部朝向圆心,所以旋转角度为字符角度 + π * 1.5
|
||||
ctx.rotate(charAngle + Math.PI * 1.5); |
||||
|
||||
// 绘制文字(使用 0, 0 作为中心点,因为已经 translate 到字符位置)
|
||||
ctx.fillText(char, 0, 0); |
||||
|
||||
// 恢复上下文
|
||||
ctx.restore(); |
||||
|
||||
// 更新当前角度(逆时针排列,所以角度递减)
|
||||
currentAngle -= charAngleIncrement; |
||||
} |
||||
|
||||
// 将 Canvas 转换为图片(Data URL)
|
||||
try { |
||||
const dataURL = canvas.toDataURL('image/png'); |
||||
if (setData) { |
||||
setData({ |
||||
[imageKey]: dataURL |
||||
}); |
||||
} |
||||
resolve(dataURL); |
||||
} catch (err) { |
||||
console.error('Canvas 转图片失败:', err); |
||||
reject(err); |
||||
} |
||||
}); |
||||
} |
||||
|
||||
return { |
||||
roundText, |
||||
initRoundTextCanvas, |
||||
drawRoundTextOnCanvas, |
||||
calculateFrontRoundTextChars, |
||||
calculateBackRoundTextChars |
||||
}; |
||||
} |
||||
Loading…
Reference in new issue