├── .npmrc ├── vitest.setup.ts ├── pnpm-workspace.yaml ├── playground ├── src │ ├── vite-env.d.ts │ ├── assets │ │ ├── logo.png │ │ └── style.css │ ├── main.ts │ ├── components │ │ └── Nav.vue │ └── App.vue ├── .gitignore ├── public │ └── favicon.ico ├── postcss.config.js ├── tailwind.config.js ├── tsconfig.node.json ├── vite.config.ts ├── index.html ├── package.json ├── tsconfig.json └── README.md ├── eslint.config.js ├── src ├── index.ts ├── utils │ ├── constants.ts │ ├── get-svg-path-from-stroke.ts │ └── h-demi.ts └── components │ └── VPerfectSignature.ts ├── tsup.config.ts ├── .gitignore ├── tsconfig.json ├── vitest.config.ts ├── LICENSE.md ├── package.json ├── test ├── VPerfectSignature.test.ts └── mock.ts └── README.md /.npmrc: -------------------------------------------------------------------------------- 1 | ignore-workspace-root-check=true 2 | -------------------------------------------------------------------------------- /vitest.setup.ts: -------------------------------------------------------------------------------- 1 | import 'vitest-canvas-mock' 2 | -------------------------------------------------------------------------------- /pnpm-workspace.yaml: -------------------------------------------------------------------------------- 1 | packages: 2 | - playground 3 | -------------------------------------------------------------------------------- /playground/src/vite-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | -------------------------------------------------------------------------------- /playground/.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .DS_Store 3 | dist 4 | dist-ssr 5 | *.local 6 | -------------------------------------------------------------------------------- /eslint.config.js: -------------------------------------------------------------------------------- 1 | import antfu from '@antfu/eslint-config' 2 | 3 | export default antfu() 4 | -------------------------------------------------------------------------------- /playground/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/wobsoriano/v-perfect-signature/HEAD/playground/public/favicon.ico -------------------------------------------------------------------------------- /playground/src/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/wobsoriano/v-perfect-signature/HEAD/playground/src/assets/logo.png -------------------------------------------------------------------------------- /playground/postcss.config.js: -------------------------------------------------------------------------------- 1 | export default { 2 | plugins: { 3 | tailwindcss: {}, 4 | autoprefixer: {}, 5 | }, 6 | } 7 | -------------------------------------------------------------------------------- /playground/src/assets/style.css: -------------------------------------------------------------------------------- 1 | @import 'tailwindcss/base'; 2 | @import 'tailwindcss/components'; 3 | @import 'tailwindcss/utilities'; -------------------------------------------------------------------------------- /playground/src/main.ts: -------------------------------------------------------------------------------- 1 | import { createApp } from 'vue' 2 | import './assets/style.css' 3 | 4 | import App from './App.vue' 5 | 6 | createApp(App).mount('#app') 7 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import VPerfectSignature from './components/VPerfectSignature' 2 | 3 | export { 4 | VPerfectSignature, 5 | } 6 | 7 | export default VPerfectSignature 8 | -------------------------------------------------------------------------------- /playground/tailwind.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | mode: 'jit', 3 | content: ['./index.html', './src/**/*.{js,ts,vue}'], 4 | darkMode: 'class', 5 | variants: {}, 6 | } 7 | -------------------------------------------------------------------------------- /src/utils/constants.ts: -------------------------------------------------------------------------------- 1 | export const DEFAULT_BACKGROUND_COLOR = 'rgba(0,0,0,0)' 2 | export const DEFAULT_PEN_COLOR = '#000' 3 | export const DEFAULT_WIDTH = '100%' 4 | export const DEFAULT_HEIGHT = '100%' 5 | export const IMAGE_TYPES = ['image/png', 'image/jpeg', 'image/svg+xml'] 6 | -------------------------------------------------------------------------------- /playground/tsconfig.node.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "composite": true, 4 | "module": "ESNext", 5 | "moduleResolution": "bundler", 6 | "allowSyntheticDefaultImports": true, 7 | "skipLibCheck": true 8 | }, 9 | "include": ["vite.config.ts"] 10 | } 11 | -------------------------------------------------------------------------------- /tsup.config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'tsup' 2 | 3 | export default defineConfig({ 4 | entry: ['src/index.ts'], 5 | splitting: false, 6 | sourcemap: true, 7 | clean: true, 8 | format: ['cjs', 'esm', 'iife'], 9 | globalName: 'VPerfectSignature', 10 | dts: true, 11 | }) 12 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | /dist 4 | 5 | 6 | # local env files 7 | .env.local 8 | .env.*.local 9 | 10 | # Log files 11 | npm-debug.log* 12 | yarn-debug.log* 13 | yarn-error.log* 14 | pnpm-debug.log* 15 | 16 | # Editor directories and files 17 | .idea 18 | .vscode 19 | *.suo 20 | *.ntvs* 21 | *.njsproj 22 | *.sln 23 | *.sw? -------------------------------------------------------------------------------- /playground/vite.config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'vite' 2 | import vue from '@vitejs/plugin-vue' 3 | 4 | // https://vitejs.dev/config/ 5 | export default defineConfig({ 6 | plugins: [vue()], 7 | // eslint-disable-next-line node/prefer-global/process 8 | base: process.env.NODE_ENV === 'production' ? '/v-perfect-signature/' : './', 9 | }) 10 | -------------------------------------------------------------------------------- /playground/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | v-perfect-signature 8 | 9 | 10 |
11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/utils/get-svg-path-from-stroke.ts: -------------------------------------------------------------------------------- 1 | export default function getSvgPathFromStroke(stroke: number[][]) { 2 | if (!stroke.length) 3 | return '' 4 | 5 | const d = stroke.reduce( 6 | (acc, [x0, y0], i, arr) => { 7 | const [x1, y1] = arr[(i + 1) % arr.length] 8 | acc.push(x0, y0, (x0 + x1) / 2, (y0 + y1) / 2) 9 | return acc 10 | }, 11 | ['M', ...stroke[0], 'Q'], 12 | ) 13 | 14 | d.push('Z') 15 | return d.join(' ') 16 | } 17 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es2020", 4 | "baseUrl": ".", 5 | "module": "esnext", 6 | "moduleResolution": "node", 7 | "resolveJsonModule": true, 8 | "types": ["vitest/globals"], 9 | "allowJs": true, 10 | "strict": true, 11 | "noImplicitAny": true, 12 | "noUnusedLocals": true, 13 | "noEmit": true, 14 | "outDir": "dist", 15 | "esModuleInterop": true, 16 | "skipLibCheck": true 17 | }, 18 | "include": [ 19 | "src/**/*.ts", 20 | "src/**/*.vue", 21 | "shims-vue.d.ts" 22 | ], 23 | "exclude": [ 24 | "node_modules" 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /playground/src/components/Nav.vue: -------------------------------------------------------------------------------- 1 | 16 | -------------------------------------------------------------------------------- /vitest.config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'vitest/config' 2 | 3 | export default defineConfig({ 4 | test: { 5 | setupFiles: ['./vitest.setup.ts'], 6 | environment: 'jsdom', 7 | deps: { 8 | optimizer: { 9 | web: { 10 | include: ['vitest-canvas-mock'], 11 | }, 12 | }, 13 | }, 14 | // For this config, check https://github.com/vitest-dev/vitest/issues/740 15 | poolOptions: { 16 | threads: { 17 | singleThread: true, 18 | }, 19 | }, 20 | environmentOptions: { 21 | jsdom: { 22 | resources: 'usable', 23 | }, 24 | }, 25 | }, 26 | }) 27 | -------------------------------------------------------------------------------- /playground/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "playground", 3 | "type": "module", 4 | "version": "0.0.0", 5 | "scripts": { 6 | "dev": "vite", 7 | "build": "vue-tsc --noEmit && vite build", 8 | "serve": "vite preview", 9 | "deploy": "gh-pages -d dist", 10 | "clean": "rm -rf node_modules && rm -rf dist" 11 | }, 12 | "dependencies": { 13 | "v-perfect-signature": "*", 14 | "vue": "^3.3.11" 15 | }, 16 | "devDependencies": { 17 | "@types/node": "^20.10.4", 18 | "@vitejs/plugin-vue": "^4.5.2", 19 | "autoprefixer": "^10.4.16", 20 | "gh-pages": "^6.1.0", 21 | "postcss": "^8.4.32", 22 | "tailwindcss": "^3.3.6", 23 | "typescript": "^5.3.3", 24 | "vite": "^5.0.8", 25 | "vue-tsc": "^1.8.25" 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /playground/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ES2020", 4 | "jsx": "preserve", 5 | "lib": ["ES2020", "DOM", "DOM.Iterable"], 6 | "useDefineForClassFields": true, 7 | "module": "ESNext", 8 | 9 | /* Bundler mode */ 10 | "moduleResolution": "bundler", 11 | "resolveJsonModule": true, 12 | "allowImportingTsExtensions": true, 13 | 14 | /* Linting */ 15 | "strict": true, 16 | "noFallthroughCasesInSwitch": true, 17 | "noUnusedLocals": true, 18 | "noUnusedParameters": true, 19 | "noEmit": true, 20 | "isolatedModules": true, 21 | "skipLibCheck": true 22 | }, 23 | "references": [{ "path": "./tsconfig.node.json" }], 24 | "include": ["src/**/*.ts", "src/**/*.tsx", "src/**/*.vue"] 25 | } 26 | -------------------------------------------------------------------------------- /src/utils/h-demi.ts: -------------------------------------------------------------------------------- 1 | import { h as hDemi, isVue2 } from 'vue-demi' 2 | 3 | interface Options { 4 | props?: Record 5 | domProps?: Record 6 | on?: Record 7 | } 8 | 9 | function adaptOnsV3(ons: Record) { 10 | if (!ons) 11 | return null 12 | return Object.entries(ons).reduce((ret, [key, handler]) => { 13 | key = key.charAt(0).toUpperCase() + key.slice(1) 14 | key = `on${key}` 15 | return { ...ret, [key]: handler } 16 | }, {}) 17 | } 18 | 19 | function h(type: string | Record, options: Options & any = {}, children?: any) { 20 | if (isVue2) 21 | return hDemi(type, options, children) 22 | 23 | const { props, domProps, on, ...extraOptions } = options 24 | 25 | const ons = adaptOnsV3(on) 26 | const params = { ...extraOptions, ...props, ...domProps, ...ons } 27 | return hDemi(type, params, children) 28 | } 29 | 30 | export default h 31 | -------------------------------------------------------------------------------- /LICENSE.md: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2022 Robert Soriano 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /playground/README.md: -------------------------------------------------------------------------------- 1 | # Vue 3 + Typescript + Vite 2 | 3 | This template should help get you started developing with Vue 3 and Typescript in Vite. 4 | 5 | ## Recommended IDE Setup 6 | 7 | [VSCode](https://code.visualstudio.com/) + [Vetur](https://marketplace.visualstudio.com/items?itemName=octref.vetur). Make sure to enable `vetur.experimental.templateInterpolationService` in settings! 8 | 9 | ### If Using ` 35 | 36 | 59 | 60 | 69 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "v-perfect-signature", 3 | "type": "module", 4 | "version": "1.4.0", 5 | "description": "Perfect pressure-sensitive signature drawing for Vue 2 and 3", 6 | "license": "MIT", 7 | "homepage": "https://github.com/wobsoriano/v-perfect-signature#readme", 8 | "repository": { 9 | "type": "git", 10 | "url": "git+https://github.com/wobsoriano/v-perfect-signature.git" 11 | }, 12 | "keywords": [ 13 | "vue", 14 | "signature", 15 | "pad", 16 | "freehand" 17 | ], 18 | "exports": { 19 | ".": { 20 | "types": { 21 | "import": "./dist/index.d.ts", 22 | "require": "./dist/index.d.cts" 23 | }, 24 | "import": "./dist/index.js", 25 | "require": "./dist/index.cjs" 26 | } 27 | }, 28 | "main": "./dist/index.cjs", 29 | "module": "./dist/index.js", 30 | "unpkg": "dist/index.global.js", 31 | "jsdelivr": "dist/index.global.js", 32 | "types": "./dist/index.d.ts", 33 | "typesVersions": { 34 | "*": { 35 | "*": [ 36 | "./dist/*" 37 | ] 38 | } 39 | }, 40 | "files": [ 41 | "dist", 42 | "nuxt.mjs" 43 | ], 44 | "scripts": { 45 | "dev": "tsup --watch --onSuccess \"pnpm --filter playground dev\"", 46 | "build": "tsup", 47 | "prepublishOnly": "pnpm run build", 48 | "test": "vitest run", 49 | "release": "bumpp && pnpm publish", 50 | "lint": "eslint .", 51 | "lint:fix": "eslint . --fix" 52 | }, 53 | "peerDependencies": { 54 | "@vue/composition-api": "^1.7.1", 55 | "vue": "^2.6.14 || ^3.2.0" 56 | }, 57 | "peerDependenciesMeta": { 58 | "@vue/composition-api": { 59 | "optional": true 60 | } 61 | }, 62 | "dependencies": { 63 | "perfect-freehand": "^1.2.0", 64 | "vue-demi": "^0.14.6" 65 | }, 66 | "devDependencies": { 67 | "@antfu/eslint-config": "^2.4.5", 68 | "@vue/test-utils": "2.4.1", 69 | "bumpp": "^9.2.1", 70 | "canvas": "^2.11.2", 71 | "eslint": "^8.55.0", 72 | "jsdom": "^23.0.1", 73 | "tslib": "^2.6.2", 74 | "tsup": "^8.0.1", 75 | "typescript": "^5.3.3", 76 | "vitest": "^1.0.4", 77 | "vitest-canvas-mock": "^0.3.3", 78 | "vue": "^3.3.11" 79 | } 80 | } 81 | -------------------------------------------------------------------------------- /test/VPerfectSignature.test.ts: -------------------------------------------------------------------------------- 1 | import { shallowMount } from '@vue/test-utils' 2 | 3 | import { describe, expect, it, vi } from 'vitest' 4 | import VPerfectSignature from '../src' 5 | import { inputPointsMockData, mockDataURL } from './mock' 6 | 7 | describe('#props', () => { 8 | it('should receive default props', () => { 9 | const wrapper = shallowMount(VPerfectSignature) 10 | 11 | const expectedWidth = '100%' 12 | const expectedHeight = '100%' 13 | const expectedPenColor = '#000' 14 | const expectedBackgroundColor = 'rgba(0,0,0,0)' 15 | const expectedStrokeOptions = {} 16 | 17 | expect(wrapper.props().width).toBe(expectedWidth) 18 | expect(wrapper.props().height).toBe(expectedHeight) 19 | expect(wrapper.props().penColor).toBe(expectedPenColor) 20 | expect(wrapper.props().backgroundColor).toBe(expectedBackgroundColor) 21 | expect(wrapper.props().strokeOptions).toEqual(expectedStrokeOptions) 22 | }) 23 | }) 24 | 25 | describe('#toDataURL', () => { 26 | it('throws error if invalid type', () => { 27 | const wrapper = shallowMount(VPerfectSignature) 28 | 29 | expect(() => wrapper.vm.toDataURL('text/html')).toThrow() 30 | }) 31 | 32 | it('returns undefined if pad is empty', () => { 33 | const wrapper = shallowMount(VPerfectSignature) 34 | 35 | expect(wrapper.vm.toDataURL()).toBeUndefined() 36 | }) 37 | 38 | // TODO: Returns incorrect data url. Bug? 39 | it.todo('should return data uri', async () => { 40 | const wrapper = shallowMount(VPerfectSignature) 41 | 42 | await wrapper.setData({ 43 | allInputPoints: inputPointsMockData, 44 | }) 45 | 46 | expect(wrapper.vm.toDataURL()).toBe(mockDataURL) 47 | }) 48 | }) 49 | 50 | describe('#fromDataURL', () => { 51 | it('should set signature from data uri', async () => { 52 | const wrapper = shallowMount(VPerfectSignature) 53 | await expect(wrapper.vm.fromDataURL(mockDataURL)).resolves.toBe(true) 54 | }) 55 | 56 | vi.spyOn(console, 'error').mockImplementation(() => {}) 57 | 58 | it('fails if data uri is incorrect', async () => { 59 | const wrapper = shallowMount(VPerfectSignature) 60 | 61 | await expect(wrapper.vm.fromDataURL('random string')).rejects.toThrow( 62 | 'Incorrect data uri provided', 63 | ) 64 | }) 65 | }) 66 | 67 | describe('#toData', () => { 68 | it('returns array of array input points', () => { 69 | const wrapper = shallowMount(VPerfectSignature) 70 | 71 | wrapper.setData({ 72 | allInputPoints: inputPointsMockData, 73 | }) 74 | 75 | expect(wrapper.vm.toData()).toEqual(inputPointsMockData) 76 | }) 77 | 78 | it('should set signature from array of array of input points', () => { 79 | const wrapper = shallowMount(VPerfectSignature) 80 | 81 | expect(wrapper.vm.fromData(inputPointsMockData)).toBeUndefined() 82 | expect(wrapper.vm.toData()).toEqual(inputPointsMockData) 83 | }) 84 | }) 85 | 86 | describe('#clear', () => { 87 | it('clears data structures and pad', () => { 88 | const wrapper = shallowMount(VPerfectSignature) 89 | 90 | wrapper.setData({ 91 | allInputPoints: inputPointsMockData, 92 | }) 93 | wrapper.vm.clear() 94 | 95 | expect(wrapper.vm.toData()).toEqual([]) 96 | }) 97 | }) 98 | 99 | describe('#isEmpty', () => { 100 | it('returns true if pad is empty', () => { 101 | const wrapper = shallowMount(VPerfectSignature) 102 | 103 | expect(wrapper.vm.isEmpty()).toBe(true) 104 | }) 105 | 106 | it('returns false if pad is not empty', () => { 107 | const wrapper = shallowMount(VPerfectSignature) 108 | 109 | wrapper.setData({ 110 | allInputPoints: inputPointsMockData, 111 | }) 112 | 113 | expect(wrapper.vm.isEmpty()).toBe(false) 114 | }) 115 | }) 116 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # v-perfect-signature 2 | 3 | Pressure-sensitive signature drawing for Vue 2 and 3 built on top of [perfect-freehand](https://github.com/steveruizok/perfect-freehand). 4 | 5 | Demo: https://wobsoriano.github.io/v-perfect-signature 6 | 7 | ## Install 8 | 9 | ```bash 10 | pnpm add v-perfect-signature 11 | ``` 12 | 13 | ## Usage 14 | 15 | ```vue 16 | 33 | 34 | 37 | ``` 38 | 39 | ## Props 40 | 41 | | Name | Type | Default | Description | 42 | | ----------------- | ------ | -------------------------------------------------------------------- | ------------------------ | 43 | | `width` | String | `100%` | Set canvas width | 44 | | `height` | String | `100%` | Set canvas height | 45 | | `backgroundColor` | String | `rgba(0,0,0,0)` | Canvas background color | 46 | | `penColor` | String | `#000` | Canvas pen color | 47 | | `strokeOptions` | Object | [Reference](https://github.com/steveruizok/perfect-freehand#options) | Perfect freehand options | 48 | 49 | ## Methods 50 | 51 | | Name | Argument Type | Description | 52 | | --------------------------- | ----------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------ | 53 | | `toDataURL(type)` | [String](https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toDataURL) | Returns signature image as data URL | 54 | | `fromDataURL(dataUri)` | [String](https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs) | Draws signature image from data URL | 55 | | `toData` | - | Returns signature image as an array of array of input points | 56 | | `fromData(data)` | [Array](https://github.com/wobsoriano/v-perfect-signature/blob/master/packages/lib/src/components/__tests__/mock.ts#L1) | Draws signature image from array of array of input points | 57 | | `clear()` | - | Clears the canvas | 58 | | `isEmpty()` | - | Returns true if canvas is empty | 59 | | `resizeCanvas(shouldClear)` | `Boolean` | Resizes and recalculate dimensions | 60 | 61 | Note: Like [signature_pad](https://github.com/szimek/signature_pad), `fromDataURL` does not populate internal data structure. Thus, after using `fromDataURL`, `toData` won't work properly. 62 | 63 | ## Events 64 | 65 | | Name | Type | Default | Description | 66 | | --------- | -------- | ------- | ----------------------- | 67 | | `onBegin` | Function | - | Fired when stroke begin | 68 | | `onEnd` | Function | - | Fired when stroke end | 69 | 70 | ## Nuxt 71 | 72 | ```js 73 | export default defineNuxtConfig({ 74 | build: { 75 | transpile: ['v-perfect-signature'] 76 | } 77 | }) 78 | ``` 79 | 80 | ## License 81 | 82 | MIT 83 | -------------------------------------------------------------------------------- /src/components/VPerfectSignature.ts: -------------------------------------------------------------------------------- 1 | import type { PropType } from 'vue-demi' 2 | import { defineComponent } from 'vue-demi' 3 | import type { StrokeOptions } from 'perfect-freehand' 4 | import * as PerfectFreehand from 'perfect-freehand' 5 | 6 | import h from '../utils/h-demi' 7 | import getSvgPathFromStroke from '../utils/get-svg-path-from-stroke' 8 | import { 9 | DEFAULT_BACKGROUND_COLOR, 10 | DEFAULT_HEIGHT, 11 | DEFAULT_PEN_COLOR, 12 | DEFAULT_WIDTH, 13 | IMAGE_TYPES, 14 | } from '../utils/constants' 15 | 16 | const { getStroke } = PerfectFreehand 17 | 18 | type InputPoints = number[] 19 | 20 | export default defineComponent({ 21 | props: { 22 | width: { 23 | type: String, 24 | required: false, 25 | default: DEFAULT_WIDTH, 26 | }, 27 | height: { 28 | type: String, 29 | required: false, 30 | default: DEFAULT_HEIGHT, 31 | }, 32 | backgroundColor: { 33 | type: String, 34 | required: false, 35 | default: DEFAULT_BACKGROUND_COLOR, 36 | }, 37 | penColor: { 38 | type: String, 39 | required: false, 40 | default: DEFAULT_PEN_COLOR, 41 | }, 42 | strokeOptions: { 43 | type: Object as PropType, 44 | required: false, 45 | default: () => ({}), 46 | }, 47 | }, 48 | emits: ['onBegin', 'onEnd'], 49 | data: () => ({ 50 | allInputPoints: [] as InputPoints[][], 51 | currentInputPoints: null as InputPoints[] | null, 52 | isDrawing: false, 53 | isLocked: false, 54 | cachedImages: [] as HTMLImageElement[], 55 | ctx: null as null | CanvasRenderingContext2D, 56 | }), 57 | watch: { 58 | backgroundColor() { 59 | this.setBackgroundAndPenColor() 60 | }, 61 | penColor(color: string) { 62 | const ctx = this.getCanvasContext() 63 | if (ctx) 64 | ctx.fillStyle = color 65 | }, 66 | allInputPoints: { 67 | deep: true, 68 | handler() { 69 | this.inputPointsHandler() 70 | }, 71 | }, 72 | currentInputPoints: { 73 | deep: true, 74 | handler() { 75 | this.inputPointsHandler() 76 | }, 77 | }, 78 | }, 79 | mounted() { 80 | this.resizeCanvas() 81 | }, 82 | methods: { 83 | _drawImage(image: HTMLImageElement) { 84 | const canvas = this.getCanvasElement() 85 | const ctx = this.getCanvasContext() 86 | const dpr = window.devicePixelRatio || 1 87 | 88 | ctx?.scale(1 / dpr, 1 / dpr) // To allow proper scaling of the image on HiDPI, we need to reset the scaling before calling `drawImage` 89 | ctx?.drawImage(image, 0, 0, canvas.width, canvas.height) 90 | ctx?.scale(dpr, dpr) // Set scaling back to original value 91 | }, 92 | handlePointerDown(e: PointerEvent) { 93 | if (this.isLocked) 94 | return 95 | 96 | e.preventDefault() 97 | 98 | const canvas = e.composedPath()[0] as HTMLCanvasElement 99 | const rect = canvas.getBoundingClientRect() 100 | const x = e.clientX - rect.left 101 | const y = e.clientY - rect.top 102 | 103 | this.currentInputPoints = [[x, y, e.pressure]] 104 | this.isDrawing = true 105 | this.$emit('onBegin', e) 106 | }, 107 | handlePointerMove(e: PointerEvent) { 108 | if (this.isLocked) 109 | return 110 | if (!this.isDrawing) 111 | return 112 | 113 | if (e.buttons === 1) { 114 | e.preventDefault() 115 | 116 | const canvas = e.composedPath()[0] as HTMLCanvasElement 117 | const rect = canvas.getBoundingClientRect() 118 | const x = e.clientX - rect.left 119 | const y = e.clientY - rect.top 120 | 121 | this.currentInputPoints = [ 122 | ...(this.currentInputPoints ?? []), 123 | [x, y, e.pressure], 124 | ] 125 | } 126 | }, 127 | handlePointerUp(e: PointerEvent) { 128 | if (this.isLocked) 129 | return 130 | 131 | e.preventDefault() 132 | this.isDrawing = false 133 | 134 | if (!this.currentInputPoints) 135 | return 136 | 137 | this.allInputPoints = [...this.allInputPoints, this.currentInputPoints] 138 | this.currentInputPoints = null 139 | 140 | this.$emit('onEnd', e) 141 | }, 142 | handlePointerEnter(e: PointerEvent) { 143 | if (this.isLocked) 144 | return 145 | 146 | if (e.buttons === 1) 147 | this.handlePointerDown(e) 148 | }, 149 | handlePointerLeave(e: PointerEvent) { 150 | if (this.isLocked) 151 | return 152 | if (!this.isDrawing) 153 | return 154 | this.handlePointerUp(e) 155 | }, 156 | isEmpty() { 157 | return !this.allInputPoints.length && !this.cachedImages.length 158 | }, 159 | clear() { 160 | this.cachedImages = [] 161 | this.allInputPoints = [] 162 | this.currentInputPoints = null 163 | }, 164 | fromData(data: InputPoints[][]) { 165 | this.allInputPoints = [...this.allInputPoints, ...data] 166 | }, 167 | lock() { 168 | this.isLocked = true 169 | }, 170 | unlock() { 171 | this.isLocked = false 172 | }, 173 | toData() { 174 | return this.allInputPoints 175 | }, 176 | fromDataURL(data: string) { 177 | return new Promise((resolve, reject) => { 178 | const image = new Image() 179 | 180 | image.onload = () => { 181 | this._drawImage(image) 182 | this.cachedImages.push(image) 183 | resolve(true) 184 | } 185 | 186 | image.onerror = () => { 187 | reject(new Error('Incorrect data uri provided')) 188 | } 189 | 190 | image.crossOrigin = 'anonymous' 191 | image.src = data 192 | }) 193 | }, 194 | toDataURL(type?: string) { 195 | if (type && !IMAGE_TYPES.includes(type)) { 196 | throw new Error( 197 | `Incorrect image type. Must be one of ${IMAGE_TYPES.join(', ')}.`, 198 | ) 199 | } 200 | 201 | if (this.isEmpty()) 202 | return 203 | 204 | const canvas = this.getCanvasElement() 205 | return canvas.toDataURL(type ?? 'image/png') 206 | }, 207 | getCanvasElement() { 208 | return this.$refs.signaturePad as HTMLCanvasElement 209 | }, 210 | getCanvasContext() { 211 | if (!this.ctx) { 212 | const canvas = this.getCanvasElement() 213 | this.ctx = canvas.getContext('2d') 214 | } 215 | return this.ctx 216 | }, 217 | setBackgroundAndPenColor() { 218 | const canvas = this.getCanvasElement() 219 | const ctx = canvas.getContext('2d') 220 | ctx!.fillStyle = this.backgroundColor 221 | ctx?.fillRect(0, 0, canvas.width, canvas.height) 222 | ctx!.fillStyle = this.penColor 223 | }, 224 | resizeCanvas(clearCanvas = true) { 225 | const canvas = this.getCanvasElement() 226 | const rect = canvas.getBoundingClientRect() 227 | const dpr = window.devicePixelRatio || 1 228 | 229 | canvas.width = rect.width * dpr 230 | canvas.height = rect.height * dpr 231 | const ctx = this.getCanvasContext() 232 | ctx?.scale(dpr, dpr) 233 | 234 | canvas.style.width = `${rect.width}px` 235 | canvas.style.height = `${rect.height}px` 236 | 237 | if (clearCanvas) { 238 | ctx?.clearRect(0, 0, canvas.width, canvas.height) 239 | this.clear() 240 | } 241 | 242 | this.setBackgroundAndPenColor() 243 | }, 244 | inputPointsHandler() { 245 | const canvas = this.getCanvasElement() 246 | const ctx = this.getCanvasContext() 247 | 248 | // Makes smooth lines 249 | ctx?.clearRect(0, 0, canvas.width, canvas.height) 250 | 251 | this.cachedImages.forEach(image => this._drawImage(image)) 252 | this.setBackgroundAndPenColor() 253 | 254 | this.allInputPoints.forEach((point: InputPoints[]) => { 255 | const pathData = getSvgPathFromStroke( 256 | getStroke(point, this.strokeOptions), 257 | ) 258 | const myPath = new Path2D(pathData) 259 | ctx?.fill(myPath) 260 | }) 261 | 262 | if (!this.currentInputPoints) 263 | return 264 | const pathData = getSvgPathFromStroke( 265 | getStroke(this.currentInputPoints!, this.strokeOptions), 266 | ) 267 | const myPath = new Path2D(pathData) 268 | ctx?.fill(myPath) 269 | }, 270 | }, 271 | render() { 272 | const { width, height } = this 273 | 274 | const handlers = { 275 | pointerdown: this.handlePointerDown, 276 | pointerup: this.handlePointerUp, 277 | pointermove: this.handlePointerMove, 278 | pointerenter: this.handlePointerEnter, 279 | pointerleave: this.handlePointerLeave, 280 | } 281 | 282 | return h('canvas', { 283 | ref: 'signaturePad', 284 | style: { 285 | height, 286 | width, 287 | touchAction: 'none', 288 | }, 289 | on: handlers, 290 | }) 291 | }, 292 | }) 293 | -------------------------------------------------------------------------------- /test/mock.ts: -------------------------------------------------------------------------------- 1 | export const inputPointsMockData = [ 2 | [ 3 | [213.0234375, 415.578125, 0.5], 4 | [213.25, 415.578125, 0.5], 5 | [213.25, 415.34765625, 0.5], 6 | [213.4765625, 415.34765625, 0.5], 7 | [215.1015625, 415.34765625, 0.5], 8 | [218.99609375, 414.234375, 0.5], 9 | [228.234375, 410.93359375, 0.5], 10 | [235.30078125, 407.07421875, 0.5], 11 | [251.10546875, 397.015625, 0.5], 12 | [272.53125, 381.5859375, 0.5], 13 | [293.80078125, 363.859375, 0.5], 14 | [299.58203125, 358.07421875, 0.5], 15 | [315.0078125, 336.64453125, 0.5], 16 | [323.46875, 320.41796875, 0.5], 17 | [329.80859375, 299.01171875, 0.5], 18 | [331.8984375, 283.6640625, 0.5], 19 | [332.59765625, 266.7890625, 0.5], 20 | [330.44140625, 248.109375, 0.5], 21 | [326.25390625, 232.76171875, 0.5], 22 | [322.25, 223.42578125, 0.5], 23 | [318.53515625, 217.85546875, 0.5], 24 | [316.4453125, 215.2421875, 0.5], 25 | [314.0625, 213.8125, 0.5], 26 | [313.1875, 213.8125, 0.5], 27 | [312.95703125, 213.8125, 0.5], 28 | [312.7265625, 214.0390625, 0.5], 29 | [311.91015625, 215.6640625, 0.5], 30 | [306.484375, 225.8359375, 0.5], 31 | [302.38671875, 238.125, 0.5], 32 | [297.62890625, 260.3203125, 0.5], 33 | [294.66796875, 294.828125, 0.5], 34 | [294.66796875, 310.93359375, 0.5], 35 | [296.31640625, 334.88671875, 0.5], 36 | [300.625, 353.5625, 0.5], 37 | [305.50390625, 368.20703125, 0.5], 38 | [308.59375, 373.7734375, 0.5], 39 | [313.15625, 381.59375, 0.5], 40 | [315.9375, 384.9296875, 0.5], 41 | [317.15625, 386.1484375, 0.5], 42 | [317.15625, 386.375, 0.5], 43 | [317.15625, 386.14453125, 0.5], 44 | [317.15625, 385.9140625, 0.5], 45 | [316.74609375, 384.28515625, 0.5], 46 | [316.74609375, 381.15234375, 0.5], 47 | [316.74609375, 374.203125, 0.5], 48 | [317.38671875, 366.48828125, 0.5], 49 | [319.95703125, 358.7734375, 0.5], 50 | [325.171875, 351.6015625, 0.5], 51 | [326.390625, 350.37890625, 0.5], 52 | [335.72265625, 345.7109375, 0.5], 53 | [342.66796875, 345.078125, 0.5], 54 | [346.5625, 346.1875, 0.5], 55 | [351.5078125, 349.8984375, 0.5], 56 | [353.41015625, 351.80078125, 0.5], 57 | [357.625, 356.015625, 0.5], 58 | [364.69140625, 359.87109375, 0.5], 59 | [377.89453125, 362.23828125, 0.5], 60 | [383.3125, 362.23828125, 0.5], 61 | [387.20703125, 361.125, 0.5], 62 | [395.02734375, 357.2109375, 0.5], 63 | [403.02734375, 351.20703125, 0.5], 64 | [408.80859375, 345.421875, 0.5], 65 | [417.73828125, 335.11328125, 0.5], 66 | [424.60546875, 323.4296875, 0.5], 67 | [427.26953125, 313.42578125, 0.5], 68 | [427.87109375, 308.00390625, 0.5], 69 | [427.2109375, 298.765625, 0.5], 70 | [425.78125, 296.859375, 0.5], 71 | [425.55078125, 296.62890625, 0.5], 72 | [425.08984375, 296.62890625, 0.5], 73 | [424.859375, 296.85546875, 0.5], 74 | [422.4765625, 298.28125, 0.5], 75 | [418.76171875, 303.2265625, 0.5], 76 | [415.046875, 308.79296875, 0.5], 77 | [411.1328125, 316.61328125, 0.5], 78 | [408.65625, 322.796875, 0.5], 79 | [405.96484375, 333.55859375, 0.5], 80 | [405.28515625, 345.0859375, 0.5], 81 | [406.5859375, 353.55859375, 0.5], 82 | [412.046875, 365.1640625, 0.5], 83 | [415.7578125, 370.109375, 0.5], 84 | [423.828125, 377.5078125, 0.5], 85 | [433.16015625, 381.5078125, 0.5], 86 | [450.03125, 383.61328125, 0.5], 87 | [456.21484375, 383.61328125, 0.5], 88 | [469.26953125, 379.48828125, 0.5], 89 | [481.765625, 371.84765625, 0.5], 90 | [491.484375, 360.734375, 0.5], 91 | [498.45703125, 346.78125, 0.5], 92 | [499.69140625, 340.59375, 0.5], 93 | [500.37890625, 326.7734375, 0.5], 94 | [497.57421875, 310.6640625, 0.5], 95 | [492.6328125, 293.02734375, 0.5], 96 | [487.1328125, 280.65625, 0.5], 97 | [485.32421875, 275.234375, 0.5], 98 | [480.54296875, 263.625, 0.5], 99 | [477.17578125, 252.859375, 0.5], 100 | [475.31640625, 246.671875, 0.5], 101 | [474.79296875, 243.5390625, 0.5], 102 | [474.79296875, 243.30859375, 0.5], 103 | [474.5625, 243.765625, 0.5], 104 | [474.5625, 244.63671875, 0.5], 105 | [474.5625, 253.875, 0.5], 106 | [474.5625, 261.5859375, 0.5], 107 | [475.265625, 279.21875, 0.5], 108 | [478.0859375, 296.8515625, 0.5], 109 | [484.42578125, 318.25390625, 0.5], 110 | [488.21484375, 324.5703125, 0.5], 111 | [496.50390625, 336.3125, 0.5], 112 | [506.12109375, 345.9296875, 0.5], 113 | [513.37890625, 351.8671875, 0.5], 114 | [522.7109375, 356.53125, 0.5], 115 | [525.83984375, 357.05078125, 0.5], 116 | [532.0234375, 357.05078125, 0.5], 117 | [539.84375, 353.13671875, 0.5], 118 | [547.9140625, 345.0625, 0.5], 119 | [555.51171875, 333.31640625, 0.5], 120 | [560.17578125, 323.98046875, 0.5], 121 | [567.30859375, 302.57421875, 0.5], 122 | [570.1171875, 285.69921875, 0.5], 123 | [571.55078125, 267.01953125, 0.5], 124 | [571.55078125, 246.578125, 0.5], 125 | [570.8671875, 234.28515625, 0.5], 126 | [569.5, 221.9921875, 0.5], 127 | [566.765625, 209.69921875, 0.5], 128 | [564.95703125, 204.27734375, 0.5], 129 | [564.546875, 202.6484375, 0.5], 130 | [564.31640625, 202.6484375, 0.5], 131 | [564.31640625, 202.875, 0.5], 132 | [563.1484375, 207.53125, 0.5], 133 | [561.7890625, 219.05859375, 0.5], 134 | [561.7890625, 235.9296875, 0.5], 135 | [562.48046875, 250.5078125, 0.5], 136 | [566.0703125, 269.18359375, 0.5], 137 | [571.09765625, 287.859375, 0.5], 138 | [576.62109375, 300.984375, 0.5], 139 | [582.1171875, 313.3515625, 0.5], 140 | [585.2734375, 319.66796875, 0.5], 141 | [590.48828125, 326.8359375, 0.5], 142 | [594.703125, 331.05078125, 0.5], 143 | [597.83203125, 332.61328125, 0.5], 144 | [598.703125, 332.90234375, 0.5], 145 | [599.57421875, 332.90234375, 0.5], 146 | [602.703125, 332.37890625, 0.5], 147 | [606.91796875, 328.76171875, 0.5], 148 | [612.05859375, 322.9765625, 0.5], 149 | [618.88671875, 312.73046875, 0.5], 150 | [625.828125, 299.53515625, 0.5], 151 | [630.74609375, 283.36328125, 0.5], 152 | [633.5234375, 268.78125, 0.5], 153 | [633.5234375, 258.015625, 0.5], 154 | [633.5234375, 250.30078125, 0.5], 155 | [632.35546875, 245.640625, 0.5], 156 | [632.125, 245.1796875, 0.5], 157 | [632.125, 245.40625, 0.5], 158 | [632.125, 245.86328125, 0.5], 159 | [631.6015625, 248.9921875, 0.5], 160 | [630.921875, 260.51953125, 0.5], 161 | [630.921875, 274.3359375, 0.5], 162 | [630.921875, 281.28125, 0.5], 163 | [634.35546875, 294.3359375, 0.5], 164 | [638.97265625, 302.9140625, 0.5], 165 | [645.69921875, 311.66015625, 0.5], 166 | [649.19140625, 315.15234375, 0.5], 167 | [659.36328125, 320.57421875, 0.5], 168 | [671.65234375, 323.3046875, 0.5], 169 | [680.890625, 323.9609375, 0.5], 170 | [692.41796875, 323.28125, 0.5], 171 | [697.07421875, 322.11328125, 0.5], 172 | [701.890625, 319.09765625, 0.5], 173 | [704.671875, 315.7578125, 0.5], 174 | [705.2890625, 309.5703125, 0.5], 175 | [704.61328125, 298.8046875, 0.5], 176 | [701.3515625, 290.98046875, 0.5], 177 | [694.51953125, 280.05078125, 0.5], 178 | [687.18359375, 272.71484375, 0.5], 179 | [676.12890625, 264.421875, 0.5], 180 | [666.79296875, 260.41796875, 0.5], 181 | [664.41015625, 259.46484375, 0.5], 182 | [659.75, 258.87890625, 0.5], 183 | [654.328125, 258.87890625, 0.5], 184 | [648.140625, 260.11328125, 0.5], 185 | [646.234375, 261.5390625, 0.5], 186 | [645.7734375, 261.99609375, 0.5], 187 | [645.48046875, 262.8671875, 0.5], 188 | ], 189 | ] 190 | 191 | export const mockDataURL 192 | = '' 193 | --------------------------------------------------------------------------------