diff --git a/README.md b/README.md index 7e3565d3..99bde92f 100644 --- a/README.md +++ b/README.md @@ -51,7 +51,11 @@ const signaturePad = new SignaturePad(canvas); signaturePad.toDataURL(); // save image as PNG signaturePad.toDataURL("image/jpeg"); // save image as JPEG signaturePad.toDataURL("image/jpeg", 0.5); // save image as JPEG with 0.5 image quality -signaturePad.toDataURL("image/svg+xml"); // save image as SVG +signaturePad.toDataURL("image/svg+xml"); // save image as SVG data url + +// Return svg string without converting to base64 +signaturePad.toSVG(); // "" +signaturePad.toSVG({includeBackgroundColor: true}); // add background color to svg output // Draws signature image from data URL (mostly uses https://mdn.io/drawImage under-the-hood) // NOTE: This method does not populate internal data structure that represents drawn signature. Thus, after using #fromDataURL, #toData won't work properly. diff --git a/docs/js/signature_pad.umd.js b/docs/js/signature_pad.umd.js index 5643612f..4b615d84 100644 --- a/docs/js/signature_pad.umd.js +++ b/docs/js/signature_pad.umd.js @@ -1,5 +1,5 @@ /*! - * Signature Pad v4.0.9 | https://github.com/szimek/signature_pad + * Signature Pad v4.0.10 | https://github.com/szimek/signature_pad * (c) 2022 Szymon Nowak | Released under the MIT license */ @@ -268,8 +268,14 @@ toDataURL(type = 'image/png', encoderOptions) { switch (type) { case 'image/svg+xml': - return this._toSVG(); + if (typeof encoderOptions !== 'object') { + encoderOptions = undefined; + } + return `data:image/svg+xml;base64,${btoa(this.toSVG(encoderOptions))}`; default: + if (typeof encoderOptions !== 'number') { + encoderOptions = undefined; + } return this.canvas.toDataURL(type, encoderOptions); } } @@ -499,7 +505,7 @@ } } } - _toSVG() { + toSVG({ includeBackgroundColor = false } = {}) { const pointGroups = this._data; const ratio = Math.max(window.devicePixelRatio || 1, 1); const minX = 0; @@ -507,8 +513,18 @@ const maxX = this.canvas.width / ratio; const maxY = this.canvas.height / ratio; const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); - svg.setAttribute('width', this.canvas.width.toString()); - svg.setAttribute('height', this.canvas.height.toString()); + svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg'); + svg.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink'); + svg.setAttribute('viewBox', `${minX} ${minY} ${maxX} ${maxY}`); + svg.setAttribute('width', maxX.toString()); + svg.setAttribute('height', maxY.toString()); + if (includeBackgroundColor && this.backgroundColor) { + const rect = document.createElement('rect'); + rect.setAttribute('width', '100%'); + rect.setAttribute('height', '100%'); + rect.setAttribute('fill', this.backgroundColor); + svg.appendChild(rect); + } this._fromData(pointGroups, (curve, { penColor }) => { const path = document.createElement('path'); if (!isNaN(curve.control1.x) && @@ -535,27 +551,7 @@ circle.setAttribute('fill', penColor); svg.appendChild(circle); }); - const prefix = 'data:image/svg+xml;base64,'; - const header = ''; - let body = svg.innerHTML; - if (body === undefined) { - const dummy = document.createElement('dummy'); - const nodes = svg.childNodes; - dummy.innerHTML = ''; - for (let i = 0; i < nodes.length; i += 1) { - dummy.appendChild(nodes[i].cloneNode(true)); - } - body = dummy.innerHTML; - } - const footer = ''; - const data = header + body + footer; - return prefix + btoa(data); + return svg.outerHTML; } } diff --git a/src/signature_pad.ts b/src/signature_pad.ts index 1422a20d..f76a6440 100644 --- a/src/signature_pad.ts +++ b/src/signature_pad.ts @@ -26,6 +26,10 @@ export interface FromDataOptions { clear?: boolean; } +export interface ToSVGOptions { + includeBackgroundColor?: boolean; +} + export interface PointGroupOptions { dotSize: number; minWidth: number; @@ -138,11 +142,27 @@ export default class SignaturePad extends SignatureEventTarget { }); } - public toDataURL(type = 'image/png', encoderOptions?: number): string { + public toDataURL( + type: 'image/svg+xml', + encoderOptions?: ToSVGOptions, + ): string; + public toDataURL(type: string, encoderOptions?: number): string; + public toDataURL( + type = 'image/png', + encoderOptions?: number | ToSVGOptions | undefined, + ): string { switch (type) { case 'image/svg+xml': - return this._toSVG(); + if (typeof encoderOptions !== 'object') { + encoderOptions = undefined; + } + return `data:image/svg+xml;base64,${btoa( + this.toSVG(encoderOptions as ToSVGOptions), + )}`; default: + if (typeof encoderOptions !== 'number') { + encoderOptions = undefined; + } return this.canvas.toDataURL(type, encoderOptions); } } @@ -580,7 +600,7 @@ export default class SignaturePad extends SignatureEventTarget { } } - private _toSVG(): string { + public toSVG({ includeBackgroundColor = false }: ToSVGOptions = {}): string { const pointGroups = this._data; const ratio = Math.max(window.devicePixelRatio || 1, 1); const minX = 0; @@ -589,8 +609,20 @@ export default class SignaturePad extends SignatureEventTarget { const maxY = this.canvas.height / ratio; const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); - svg.setAttribute('width', this.canvas.width.toString()); - svg.setAttribute('height', this.canvas.height.toString()); + svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg'); + svg.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink'); + svg.setAttribute('viewBox', `${minX} ${minY} ${maxX} ${maxY}`); + svg.setAttribute('width', maxX.toString()); + svg.setAttribute('height', maxY.toString()); + + if (includeBackgroundColor && this.backgroundColor) { + const rect = document.createElement('rect'); + rect.setAttribute('width', '100%'); + rect.setAttribute('height', '100%'); + rect.setAttribute('fill', this.backgroundColor); + + svg.appendChild(rect); + } this._fromData( pointGroups, @@ -638,34 +670,6 @@ export default class SignaturePad extends SignatureEventTarget { }, ); - const prefix = 'data:image/svg+xml;base64,'; - const header = - ''; - let body = svg.innerHTML; - - // IE hack for missing innerHTML property on SVGElement - if (body === undefined) { - const dummy = document.createElement('dummy'); - const nodes = svg.childNodes; - dummy.innerHTML = ''; - - // tslint:disable-next-line: prefer-for-of - for (let i = 0; i < nodes.length; i += 1) { - dummy.appendChild(nodes[i].cloneNode(true)); - } - - body = dummy.innerHTML; - } - - const footer = ''; - const data = header + body + footer; - - return prefix + btoa(data); + return svg.outerHTML; } } diff --git a/tests/__snapshots__/signature_pad.test.ts.snap b/tests/__snapshots__/signature_pad.test.ts.snap index 6c85d23d..9e8260f1 100644 --- a/tests/__snapshots__/signature_pad.test.ts.snap +++ b/tests/__snapshots__/signature_pad.test.ts.snap @@ -8,4 +8,12 @@ exports[`#toDataURL returns SVG image in data URL format 1`] = `""`; +exports[`#toDataURL returns SVG image with backgroundColor 1`] = `""`; + +exports[`#toSVG returns SVG image 1`] = `""`; + +exports[`#toSVG returns SVG image with backgroundColor 1`] = `""`; + +exports[`#toSVG returns SVG image with high DPI 1`] = `""`; + exports[`user interactions allows user to paint on the pad 1`] = `""`; diff --git a/tests/signature_pad.test.ts b/tests/signature_pad.test.ts index ebfd9325..f337093e 100644 --- a/tests/signature_pad.test.ts +++ b/tests/signature_pad.test.ts @@ -122,13 +122,15 @@ describe('#toData', () => { // describe('#fromDataURL', () => {}); describe('#toDataURL', () => { - // Unfortunately, results of Canvas#toDataURL depend on a platform :/ - // it('returns PNG image in data URL format', () => { - // const pad = new SignaturePad(canvas); - // pad.fromData(face); - // - // expect(pad.toDataURL('image/png')).toMatchSnapshot(); - // }); + it('returns PNG image in data URL format', () => { + const pad = new SignaturePad(canvas); + pad.fromData(face); + + // Unfortunately, results of Canvas#toDataURL depend on a platform :/ + expect(pad.toDataURL('image/png')).toEqual( + expect.stringMatching('data:image/png'), + ); + }); // Synchronous Canvas#toDataURL for JPEG images is not supported by 'canvas' library :/ // it.skip('returns JPG image in data URL format', () => {}); @@ -147,6 +149,39 @@ describe('#toDataURL', () => { expect(pad.toDataURL('image/svg+xml')).toMatchSnapshot(); }); + + it('returns SVG image with backgroundColor', () => { + const pad = new SignaturePad(canvas, { backgroundColor: '#fcc' }); + pad.fromData(face); + + expect( + pad.toDataURL('image/svg+xml', { includeBackgroundColor: true }), + ).toMatchSnapshot(); + }); +}); + +describe('#toSVG', () => { + it('returns SVG image', () => { + const pad = new SignaturePad(canvas); + pad.fromData(face); + + expect(pad.toSVG()).toMatchSnapshot(); + }); + + it('returns SVG image with high DPI', () => { + changeDevicePixelratio(2); + const pad = new SignaturePad(canvas); + pad.fromData(face); + + expect(pad.toSVG()).toMatchSnapshot(); + }); + + it('returns SVG image with backgroundColor', () => { + const pad = new SignaturePad(canvas, { backgroundColor: '#fcc' }); + pad.fromData(face); + + expect(pad.toSVG({ includeBackgroundColor: true })).toMatchSnapshot(); + }); }); describe('user interactions', () => {