feat(signature): improve signature clarity by devicePixelRatio (#11835)

* feat(signature): Enhance signature clarity by devicePixelRatio

* refactor: optimize image generation logic

* Update Signature.tsx

---------

Co-authored-by: neverland <jait.chen@foxmail.com>
This commit is contained in:
李江辰 2023-05-15 21:26:54 +08:00 committed by GitHub
parent 4b29f1006c
commit 467243d495
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 36 additions and 10 deletions

View File

@ -76,6 +76,7 @@ Use `line-width` prop to set the width of the line.
| type | Export image type | _string_ | `png` | | type | Export image type | _string_ | `png` |
| pen-color | Color of the brush stroke, default is black | _string_ | `#000` | | pen-color | Color of the brush stroke, default is black | _string_ | `#000` |
| line-width | Width of the line | _number_ | `3` | | line-width | Width of the line | _number_ | `3` |
| background-color | Background color | _string_ | - |
| tips | Text that appears when Canvas is not supported | _string_ | - | | tips | Text that appears when Canvas is not supported | _string_ | - |
| clear-button-text | Clear button text | _string_ | `Clear` | | clear-button-text | Clear button text | _string_ | `Clear` |
| confirm-button-text | Confirm button text | _string_ | `Confirm` | | confirm-button-text | Confirm button text | _string_ | `Confirm` |

View File

@ -76,6 +76,7 @@ export default {
| type | 导出图片类型 | _string_ | `png` | | type | 导出图片类型 | _string_ | `png` |
| pen-color | 笔触颜色,默认黑色 | _string_ | `#000` | | pen-color | 笔触颜色,默认黑色 | _string_ | `#000` |
| line-width | 线条宽度 | _number_ | `3` | | line-width | 线条宽度 | _number_ | `3` |
| background-color | 背景颜色 | _string_ | - |
| tips | 当不支持 Canvas 的时候出现的提示文案 | _string_ | - | | tips | 当不支持 Canvas 的时候出现的提示文案 | _string_ | - |
| clear-button-text | 清除按钮文案 | _string_ | `清空` | | clear-button-text | 清除按钮文案 | _string_ | `清空` |
| confirm-button-text | 确认按钮文案 | _string_ | `确认` | | confirm-button-text | 确认按钮文案 | _string_ | `确认` |

View File

@ -2,6 +2,7 @@ import {
ref, ref,
reactive, reactive,
onMounted, onMounted,
nextTick,
defineComponent, defineComponent,
type ExtractPropTypes, type ExtractPropTypes,
} from 'vue'; } from 'vue';
@ -23,6 +24,7 @@ export const signatureProps = {
penColor: makeStringProp('#000'), penColor: makeStringProp('#000'),
lineWidth: makeNumberProp(3), lineWidth: makeNumberProp(3),
clearButtonText: String, clearButtonText: String,
backgroundColor: makeStringProp(''),
confirmButtonText: String, confirmButtonText: String,
}; };
@ -48,6 +50,7 @@ export default defineComponent({
width: 0, width: 0,
height: 0, height: 0,
ctx: null as CanvasRenderingContext2D | null | undefined, ctx: null as CanvasRenderingContext2D | null | undefined,
ratio: inBrowser ? window.devicePixelRatio : 1,
}); });
let canvasRect: DOMRect; let canvasRect: DOMRect;
@ -59,7 +62,7 @@ export default defineComponent({
} }
state.ctx.beginPath(); state.ctx.beginPath();
state.ctx.lineWidth = props.lineWidth; state.ctx.lineWidth = props.lineWidth * state.ratio;
state.ctx.strokeStyle = props.penColor; state.ctx.strokeStyle = props.penColor;
canvasRect = useRect(canvasRef); canvasRect = useRect(canvasRef);
@ -74,8 +77,8 @@ export default defineComponent({
preventDefault(event); preventDefault(event);
const touch = event.touches[0]; const touch = event.touches[0];
const mouseX = touch.clientX - (canvasRect?.left || 0); const mouseX = (touch.clientX - (canvasRect?.left || 0)) * state.ratio;
const mouseY = touch.clientY - (canvasRect?.top || 0); const mouseY = (touch.clientY - (canvasRect?.top || 0)) * state.ratio;
state.ctx.lineCap = 'round'; state.ctx.lineCap = 'round';
state.ctx.lineJoin = 'round'; state.ctx.lineJoin = 'round';
@ -97,6 +100,13 @@ export default defineComponent({
return canvas.toDataURL() === empty.toDataURL(); return canvas.toDataURL() === empty.toDataURL();
}; };
const setCanvasBgColor = () => {
if (state.ctx && props.backgroundColor) {
state.ctx.fillStyle = props.backgroundColor;
state.ctx.fillRect(0, 0, state.width, state.height);
}
};
const submit = () => { const submit = () => {
const canvas = canvasRef.value; const canvas = canvasRef.value;
if (!canvas) { if (!canvas) {
@ -104,12 +114,15 @@ export default defineComponent({
} }
const isEmpty = isCanvasEmpty(canvas); const isEmpty = isCanvasEmpty(canvas);
const image = isEmpty
const image: string = isEmpty
? '' ? ''
: canvas.toDataURL( : (
`image/${props.type}`, {
props.type === 'jpg' ? 0.9 : null jpg: (): string => canvas.toDataURL('image/jpeg', 0.8),
); jpeg: (): string => canvas.toDataURL('image/jpeg', 0.8),
}[props.type] as () => string
)?.() || canvas.toDataURL(`image/${props.type}`);
emit('submit', { emit('submit', {
image, image,
@ -121,6 +134,7 @@ export default defineComponent({
if (state.ctx) { if (state.ctx) {
state.ctx.clearRect(0, 0, state.width, state.height); state.ctx.clearRect(0, 0, state.width, state.height);
state.ctx.closePath(); state.ctx.closePath();
setCanvasBgColor();
} }
emit('clear'); emit('clear');
}; };
@ -128,8 +142,13 @@ export default defineComponent({
onMounted(() => { onMounted(() => {
if (isRenderCanvas) { if (isRenderCanvas) {
state.ctx = canvasRef.value?.getContext('2d'); state.ctx = canvasRef.value?.getContext('2d');
state.width = wrapRef.value?.offsetWidth || 0; state.width = (wrapRef.value?.offsetWidth || 0) * state.ratio;
state.height = wrapRef.value?.offsetHeight || 0; state.height = (wrapRef.value?.offsetHeight || 0) * state.ratio;
// ensure canvas is rendered
nextTick(() => {
setCanvasBgColor();
});
} }
}); });

View File

@ -17,6 +17,11 @@
border: var(--van-signature-content-border); border: var(--van-signature-content-border);
border-radius: var(--van-radius-lg); border-radius: var(--van-radius-lg);
overflow: hidden; overflow: hidden;
canvas {
width: 100%;
height: 100%;
}
} }
&__footer { &__footer {