mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
feat(signature): add Signature component (#11733)
* feat(signature): add signature component * feat(signature): fix md naming error * feat(signature): refactor and adjust code
This commit is contained in:
parent
dc1531084b
commit
f2b1b3156e
105
packages/vant/src/signature/README.md
Normal file
105
packages/vant/src/signature/README.md
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
# Signature
|
||||||
|
|
||||||
|
### Intro
|
||||||
|
|
||||||
|
Component for signature scene, based on Canvas.
|
||||||
|
|
||||||
|
### Install
|
||||||
|
|
||||||
|
Register component globally via `app.use`, refer to [Component Registration](#/en-US/advanced-usage#zu-jian-zhu-ce) for more registration ways.
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { createApp } from 'vue';
|
||||||
|
import { Space } from 'vant';
|
||||||
|
|
||||||
|
const app = createApp();
|
||||||
|
app.use(Space);
|
||||||
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
### Basic Usage
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature @submit="onSubmit" @clear="onClear" />
|
||||||
|
<van-image v-if="demoUrl" :src="demoUrl" />
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { ref } from 'vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
setup() {
|
||||||
|
const demoUrl = ref('');
|
||||||
|
|
||||||
|
const onSubmit = (data) => {
|
||||||
|
const { filePath, canvas } = data;
|
||||||
|
demoUrl.value = filePath;
|
||||||
|
|
||||||
|
console.log('submit', canvas, filePath);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onClear = () => console.log('clear');
|
||||||
|
|
||||||
|
return {
|
||||||
|
onSubmit,
|
||||||
|
onClear,
|
||||||
|
demoUrl,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### Pen Color
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature pen-color="#ff0000" @submit="onSubmit" @clear="onClear" />
|
||||||
|
```
|
||||||
|
|
||||||
|
### LineWidth
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature :line-width="6" @submit="onSubmit" @clear="onClear" />
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### Props
|
||||||
|
|
||||||
|
| Parameter | Description | Type | Default |
|
||||||
|
| --- | --- | --- | --- |
|
||||||
|
| type | Export image type | _string_ | `png` |
|
||||||
|
| penColor | Color of the brush stroke, default is black | _string_ | `#000` |
|
||||||
|
| lineWidth | Width of the line | _number_ | `3` |
|
||||||
|
| tips | Text that appears when Canvas is not supported | _string_ | - |
|
||||||
|
|
||||||
|
### Events
|
||||||
|
|
||||||
|
| Event Name | Description | Callback Parameters |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| start | Callback for start of signature | - |
|
||||||
|
| end | Callback for end of signature | - |
|
||||||
|
| signing | Callback for signature in progress | _event: TouchEvent_ |
|
||||||
|
| submit | submit button click | _data: {canvas: HTMLCanvasElement, filePath: string}_ |
|
||||||
|
| clear | clear button click | - |
|
||||||
|
|
||||||
|
### Types
|
||||||
|
|
||||||
|
The component exports the following type definitions:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import type { SignatureProps } from 'vant';
|
||||||
|
```
|
||||||
|
|
||||||
|
## Theming
|
||||||
|
|
||||||
|
### CSS Variables
|
||||||
|
|
||||||
|
The component provides the following CSS variables, which can be used to customize styles. Please refer to [ConfigProvider component](#/en-US/config-provider).
|
||||||
|
|
||||||
|
| Name | Default Value | Description |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| --van-signature-padding | _var(--van-padding-xs)_ | - |
|
||||||
|
| --van-signature-content-height | _160px_ | Height of the canvas |
|
||||||
|
| --van-signature-content-background | _var(--van-background-2)_ | Background color of the canvas |
|
||||||
|
| --van-signature-content-border | _1px dotted #dadada_ | Border style of the canvas |
|
105
packages/vant/src/signature/README.zh-CN.md
Normal file
105
packages/vant/src/signature/README.zh-CN.md
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
# Signature 签名
|
||||||
|
|
||||||
|
### 介绍
|
||||||
|
|
||||||
|
用于签名场景的组件,基于 Canvas。
|
||||||
|
|
||||||
|
### 引入
|
||||||
|
|
||||||
|
通过以下方式来全局注册组件,更多注册方式请参考[组件注册](#/zh-CN/advanced-usage#zu-jian-zhu-ce)。
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { createApp } from 'vue';
|
||||||
|
import { Signature } from 'vant';
|
||||||
|
|
||||||
|
const app = createApp();
|
||||||
|
app.use(Signature);
|
||||||
|
```
|
||||||
|
|
||||||
|
## 代码演示
|
||||||
|
|
||||||
|
### 基础用法
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature @submit="onSubmit" @clear="onClear" />
|
||||||
|
<van-image v-if="demoUrl" :src="demoUrl" />
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { ref } from 'vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
setup() {
|
||||||
|
const demoUrl = ref('');
|
||||||
|
|
||||||
|
const onSubmit = (data) => {
|
||||||
|
const { filePath, canvas } = data;
|
||||||
|
demoUrl.value = filePath;
|
||||||
|
|
||||||
|
console.log('submit', canvas, filePath);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onClear = () => console.log('clear');
|
||||||
|
|
||||||
|
return {
|
||||||
|
onSubmit,
|
||||||
|
onClear,
|
||||||
|
demoUrl,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### 自定义颜色
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature pen-color="#ff0000" @submit="onSubmit" @clear="onClear" />
|
||||||
|
```
|
||||||
|
|
||||||
|
### 自定义线宽
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-signature :line-width="6" @submit="onSubmit" @clear="onClear" />
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### Props
|
||||||
|
|
||||||
|
| 参数 | 说明 | 类型 | 默认值 |
|
||||||
|
| --------- | ------------------------------------ | -------- | ------ |
|
||||||
|
| type | 导出图片类型 | _string_ | `png` |
|
||||||
|
| penColor | 笔触颜色,默认黑色。 | _string_ | `#000` |
|
||||||
|
| lineWidth | 线条宽度 | _number_ | `3` |
|
||||||
|
| tips | 当不支持 Canvas 的时候出现的提示文案 | _string_ | - |
|
||||||
|
|
||||||
|
### Events
|
||||||
|
|
||||||
|
| 事件名 | 说明 | 回调参数 |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| start | 签名开始事件回调 | - |
|
||||||
|
| end | 签名结束事件回调 | - |
|
||||||
|
| signing | 签名过程事件回调 | _event: TouchEvent_ |
|
||||||
|
| submit | 确定按钮事件回调 | _data: {canvas: HTMLCanvasElement, filePath: string}_ |
|
||||||
|
| clear | 取消按钮事件回调 | - |
|
||||||
|
|
||||||
|
### 类型定义
|
||||||
|
|
||||||
|
组件导出以下类型定义:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import type { SignatureProps } from 'vant';
|
||||||
|
```
|
||||||
|
|
||||||
|
## 主题定制
|
||||||
|
|
||||||
|
### 样式变量
|
||||||
|
|
||||||
|
组件提供了下列 CSS 变量,可用于自定义样式,使用方法请参考 [ConfigProvider 组件](#/zh-CN/config-provider)。
|
||||||
|
|
||||||
|
| 名称 | 默认值 | 描述 |
|
||||||
|
| --- | --- | --- |
|
||||||
|
| --van-signature-padding | _var(--van-padding-xs)_ | - |
|
||||||
|
| --van-signature-content-height | _160px_ | 画布高度 |
|
||||||
|
| --van-signature-content-background | _var(--van-background-2)_ | 画布背景色 |
|
||||||
|
| --van-signature-content-border | _1px dotted #dadada_ | 画布边框样式 |
|
153
packages/vant/src/signature/Signature.tsx
Normal file
153
packages/vant/src/signature/Signature.tsx
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
import {
|
||||||
|
defineComponent,
|
||||||
|
reactive,
|
||||||
|
ref,
|
||||||
|
onMounted,
|
||||||
|
type ExtractPropTypes,
|
||||||
|
} from 'vue';
|
||||||
|
import { createNamespace, makeNumberProp, makeStringProp } from '../utils';
|
||||||
|
import { Button } from '../button';
|
||||||
|
|
||||||
|
const [name, bem, t] = createNamespace('signature');
|
||||||
|
|
||||||
|
export const signatureProps = {
|
||||||
|
type: makeStringProp('png'),
|
||||||
|
lineWidth: makeNumberProp(3),
|
||||||
|
penColor: makeStringProp('#000'),
|
||||||
|
tips: String,
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SignatureProps = ExtractPropTypes<typeof signatureProps>;
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name,
|
||||||
|
|
||||||
|
props: signatureProps,
|
||||||
|
|
||||||
|
emits: ['submit', 'clear', 'start', 'end', 'signing'],
|
||||||
|
|
||||||
|
setup(props, { emit }) {
|
||||||
|
const canvasRef = ref<HTMLCanvasElement | null>(null);
|
||||||
|
const wrapRef = ref<HTMLElement | null>(null);
|
||||||
|
|
||||||
|
const state = reactive({
|
||||||
|
width: 0,
|
||||||
|
height: 0,
|
||||||
|
ctx: null as any,
|
||||||
|
isSupportTouch: 'ontouchstart' in window,
|
||||||
|
});
|
||||||
|
|
||||||
|
const hasCanvasSupport = () => {
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
|
return !!(canvas.getContext && canvas.getContext('2d'));
|
||||||
|
};
|
||||||
|
|
||||||
|
const touchMove = (event: TouchEvent) => {
|
||||||
|
if (!state.ctx) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const evt = event.changedTouches
|
||||||
|
? event.changedTouches[0]
|
||||||
|
: event.targetTouches[0];
|
||||||
|
|
||||||
|
emit('signing', evt);
|
||||||
|
let mouseX = evt.clientX;
|
||||||
|
let mouseY = evt.clientY;
|
||||||
|
|
||||||
|
if (!state.isSupportTouch) {
|
||||||
|
const coverPos = canvasRef.value?.getBoundingClientRect();
|
||||||
|
mouseX = evt.clientX - (coverPos?.left || 0);
|
||||||
|
mouseY = evt.clientY - (coverPos?.top || 0);
|
||||||
|
}
|
||||||
|
state.ctx.lineCap = 'round';
|
||||||
|
state.ctx.lineJoin = 'round';
|
||||||
|
state.ctx?.lineTo(mouseX, mouseY);
|
||||||
|
state.ctx?.stroke();
|
||||||
|
};
|
||||||
|
|
||||||
|
const touchEnd = (event: { preventDefault: () => void }) => {
|
||||||
|
event.preventDefault();
|
||||||
|
emit('end');
|
||||||
|
};
|
||||||
|
|
||||||
|
const touchStart = () => {
|
||||||
|
if (!state.ctx) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
emit('start');
|
||||||
|
state.ctx.beginPath();
|
||||||
|
state.ctx.lineWidth = props.lineWidth;
|
||||||
|
state.ctx.strokeStyle = props.penColor;
|
||||||
|
};
|
||||||
|
|
||||||
|
const isCanvasEmpty = (canvas: HTMLCanvasElement) => {
|
||||||
|
const empty: HTMLCanvasElement = document.createElement('canvas');
|
||||||
|
empty.width = canvas.width;
|
||||||
|
empty.height = canvas.height;
|
||||||
|
return canvas.toDataURL() === empty.toDataURL();
|
||||||
|
};
|
||||||
|
|
||||||
|
const submit = () => {
|
||||||
|
const canvas = canvasRef.value;
|
||||||
|
if (!canvas) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const isEmpty = isCanvasEmpty(canvas);
|
||||||
|
const _canvas = isEmpty ? null : canvas;
|
||||||
|
const _filePath = isEmpty
|
||||||
|
? ''
|
||||||
|
: canvas.toDataURL(
|
||||||
|
`image/${props.type}`,
|
||||||
|
props.type === 'jpg' ? 0.9 : null
|
||||||
|
);
|
||||||
|
|
||||||
|
const data = {
|
||||||
|
canvas: _canvas,
|
||||||
|
filePath: _filePath,
|
||||||
|
};
|
||||||
|
|
||||||
|
emit('submit', data);
|
||||||
|
};
|
||||||
|
|
||||||
|
const clear = () => {
|
||||||
|
state.ctx.clearRect(0, 0, state.width, state.height);
|
||||||
|
state.ctx.closePath();
|
||||||
|
emit('clear');
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
if (hasCanvasSupport()) {
|
||||||
|
state.ctx = canvasRef.value?.getContext('2d');
|
||||||
|
state.width = wrapRef.value?.offsetWidth || 0;
|
||||||
|
state.height = wrapRef.value?.offsetHeight || 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => (
|
||||||
|
<div class={bem()}>
|
||||||
|
<div class={bem('content')} ref={wrapRef}>
|
||||||
|
{(hasCanvasSupport() && (
|
||||||
|
<canvas
|
||||||
|
ref={canvasRef}
|
||||||
|
width={state.width}
|
||||||
|
height={state.height}
|
||||||
|
onTouchstartPassive={touchStart}
|
||||||
|
onTouchmovePassive={touchMove}
|
||||||
|
onTouchend={touchEnd}
|
||||||
|
/>
|
||||||
|
)) || <p>{props.tips}</p>}
|
||||||
|
</div>
|
||||||
|
<div class={bem('footer')}>
|
||||||
|
<p>{props.tips}</p>
|
||||||
|
<Button size="small" onClick={clear}>
|
||||||
|
{t('cancel')}
|
||||||
|
</Button>
|
||||||
|
<Button type="primary" size="small" onClick={submit}>
|
||||||
|
{t('confirm')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
67
packages/vant/src/signature/demo/index.vue
Normal file
67
packages/vant/src/signature/demo/index.vue
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
<script setup>
|
||||||
|
import { ref } from 'vue';
|
||||||
|
import VanSignature from '..';
|
||||||
|
import VanImage from '../../image';
|
||||||
|
|
||||||
|
import { useTranslate } from '../../../docs/site';
|
||||||
|
|
||||||
|
const t = useTranslate({
|
||||||
|
'zh-CN': {
|
||||||
|
basic: '基础用法',
|
||||||
|
penColor: '自定义颜色',
|
||||||
|
lineWidth: '自定义线宽',
|
||||||
|
},
|
||||||
|
'en-US': {
|
||||||
|
basic: 'basic',
|
||||||
|
penColor: 'penColor',
|
||||||
|
lineWidth: 'lineWidth',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const demoUrl = ref('');
|
||||||
|
|
||||||
|
const onSubmit = (data) => {
|
||||||
|
const { filePath, canvas } = data;
|
||||||
|
demoUrl.value = filePath;
|
||||||
|
|
||||||
|
console.log('submit', canvas, filePath);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onStart = () => console.log('start');
|
||||||
|
const onClear = () => console.log('clear');
|
||||||
|
const onEnd = () => console.log('end');
|
||||||
|
const onSigning = (e) => console.log('signing', e);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<demo-block :title="t('basic')">
|
||||||
|
<van-signature
|
||||||
|
@submit="onSubmit"
|
||||||
|
@clear="onClear"
|
||||||
|
@start="onStart"
|
||||||
|
@end="onEnd"
|
||||||
|
@signing="onSigning"
|
||||||
|
/>
|
||||||
|
</demo-block>
|
||||||
|
<van-image v-if="demoUrl" :src="demoUrl" />
|
||||||
|
<demo-block :title="t('penColor')">
|
||||||
|
<van-signature
|
||||||
|
pen-color="#ff0000"
|
||||||
|
@clear="onClear"
|
||||||
|
@submit="onSubmit"
|
||||||
|
@start="onStart"
|
||||||
|
@end="onEnd"
|
||||||
|
@signing="onSigning"
|
||||||
|
/>
|
||||||
|
</demo-block>
|
||||||
|
<demo-block :title="t('lineWidth')">
|
||||||
|
<van-signature
|
||||||
|
:line-width="6"
|
||||||
|
@clear="onClear"
|
||||||
|
@submit="onSubmit"
|
||||||
|
@start="onStart"
|
||||||
|
@end="onEnd"
|
||||||
|
@signing="onSigning"
|
||||||
|
/>
|
||||||
|
</demo-block>
|
||||||
|
</template>
|
27
packages/vant/src/signature/index.less
Normal file
27
packages/vant/src/signature/index.less
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
:root {
|
||||||
|
--van-signature-padding: var(--van-padding-xs);
|
||||||
|
--van-signature-content-height: 160px;
|
||||||
|
--van-signature-content-background: var(--van-background-2);
|
||||||
|
--van-signature-content-border: 1px dotted #dadada;
|
||||||
|
}
|
||||||
|
|
||||||
|
.van-signature {
|
||||||
|
padding: var(--van-signature-padding);
|
||||||
|
|
||||||
|
&__content {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
height: var(--van-signature-content-height);
|
||||||
|
background-color: var(--van-signature-content-background);
|
||||||
|
border: var(--van-signature-content-border);
|
||||||
|
}
|
||||||
|
&__footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
|
||||||
|
.van-button {
|
||||||
|
margin: 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
12
packages/vant/src/signature/index.ts
Normal file
12
packages/vant/src/signature/index.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import { withInstall } from '../utils';
|
||||||
|
import _Signature from './Signature';
|
||||||
|
|
||||||
|
export const Signature = withInstall(_Signature);
|
||||||
|
export default Signature;
|
||||||
|
export type { SignatureProps } from './Signature';
|
||||||
|
|
||||||
|
declare module 'vue' {
|
||||||
|
export interface GlobalComponents {
|
||||||
|
Signature: typeof Signature;
|
||||||
|
}
|
||||||
|
}
|
100
packages/vant/src/signature/test/__snapshots__/demo.spec.ts.snap
Normal file
100
packages/vant/src/signature/test/__snapshots__/demo.spec.ts.snap
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`should render demo and match snapshot 1`] = `
|
||||||
|
<div>
|
||||||
|
<div class="van-signature">
|
||||||
|
<div class="van-signature__content">
|
||||||
|
<canvas width="100"
|
||||||
|
height="100"
|
||||||
|
>
|
||||||
|
</canvas>
|
||||||
|
</div>
|
||||||
|
<div class="van-signature__footer">
|
||||||
|
<p>
|
||||||
|
</p>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--default van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Cancel
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--primary van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Confirm
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div class="van-signature">
|
||||||
|
<div class="van-signature__content">
|
||||||
|
<canvas width="100"
|
||||||
|
height="100"
|
||||||
|
>
|
||||||
|
</canvas>
|
||||||
|
</div>
|
||||||
|
<div class="van-signature__footer">
|
||||||
|
<p>
|
||||||
|
</p>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--default van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Cancel
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--primary van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Confirm
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div class="van-signature">
|
||||||
|
<div class="van-signature__content">
|
||||||
|
<canvas width="100"
|
||||||
|
height="100"
|
||||||
|
>
|
||||||
|
</canvas>
|
||||||
|
</div>
|
||||||
|
<div class="van-signature__footer">
|
||||||
|
<p>
|
||||||
|
</p>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--default van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Cancel
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
<button type="button"
|
||||||
|
class="van-button van-button--primary van-button--small"
|
||||||
|
>
|
||||||
|
<div class="van-button__content">
|
||||||
|
<span class="van-button__text">
|
||||||
|
Confirm
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
4
packages/vant/src/signature/test/demo.spec.ts
Normal file
4
packages/vant/src/signature/test/demo.spec.ts
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
import Demo from '../demo/index.vue';
|
||||||
|
import { snapshotDemo } from '../../../test/demo';
|
||||||
|
|
||||||
|
snapshotDemo(Demo);
|
58
packages/vant/src/signature/test/index.spec.ts
Normal file
58
packages/vant/src/signature/test/index.spec.ts
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import { Signature } from '..';
|
||||||
|
import { mount } from '../../../test';
|
||||||
|
|
||||||
|
test('renders a canvas element when canvas is supported', async () => {
|
||||||
|
const wrapper = mount(Signature);
|
||||||
|
expect(wrapper.find('canvas').exists()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should emit "start" event when touch starts', async () => {
|
||||||
|
const wrapper = mount(Signature);
|
||||||
|
const canvas = wrapper.find('canvas');
|
||||||
|
|
||||||
|
await canvas.trigger('touchstart');
|
||||||
|
|
||||||
|
expect(wrapper.emitted('start')).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should emit "signing" event when touch is moving', async () => {
|
||||||
|
const wrapper = mount(Signature);
|
||||||
|
const canvas = wrapper.find('canvas');
|
||||||
|
|
||||||
|
await canvas.trigger('touchstart');
|
||||||
|
await canvas.trigger('touchmove', {
|
||||||
|
changedTouches: [{ clientX: 10, clientY: 20 }],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(wrapper.emitted('signing')).toBeTruthy();
|
||||||
|
expect(wrapper.emitted('signing')![0][0]).toMatchObject({
|
||||||
|
clientX: 10,
|
||||||
|
clientY: 20,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should emit `end` event when touchend is triggered', async () => {
|
||||||
|
const wrapper = mount(Signature);
|
||||||
|
await wrapper.vm.$nextTick();
|
||||||
|
|
||||||
|
const canvas = wrapper.find('canvas');
|
||||||
|
await canvas.trigger('touchend');
|
||||||
|
expect(wrapper.emitted('end')).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('submit() should output a valid canvas', async () => {
|
||||||
|
const wrapper = mount(Signature);
|
||||||
|
|
||||||
|
await wrapper.vm.$nextTick();
|
||||||
|
|
||||||
|
wrapper.vm.$emit('submit', { canvas: null, filePath: '' });
|
||||||
|
|
||||||
|
const emitted = wrapper.emitted();
|
||||||
|
expect(emitted.submit).toBeTruthy();
|
||||||
|
const [data] = emitted.submit[0] as [
|
||||||
|
{ canvas: HTMLCanvasElement | null; filePath: string }
|
||||||
|
];
|
||||||
|
|
||||||
|
expect(data.canvas).toBeNull();
|
||||||
|
expect(data.filePath).toBe('');
|
||||||
|
});
|
@ -434,6 +434,10 @@ location.href = location.href.replace('youzan.github.io', 'vant-ui.github.io');
|
|||||||
path: 'submit-bar',
|
path: 'submit-bar',
|
||||||
title: 'SubmitBar 提交订单栏',
|
title: 'SubmitBar 提交订单栏',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'signature',
|
||||||
|
title: 'Signature 签名',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -878,6 +882,10 @@ location.href = location.href.replace('youzan.github.io', 'vant-ui.github.io');
|
|||||||
path: 'submit-bar',
|
path: 'submit-bar',
|
||||||
title: 'SubmitBar',
|
title: 'SubmitBar',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'signature',
|
||||||
|
title: 'Signature',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user