├── .nvmrc ├── .npmrc ├── dist ├── freshRequire.d.ts ├── index.d.ts ├── index.js.map ├── index.js ├── freshRequire.js.map └── freshRequire.js ├── codealike.json ├── resources ├── logo.png ├── logo.xcf ├── example.gif ├── example.png ├── memory-usage-same-instance.png ├── memory-usage-new-instance-parallel.png ├── memory-usage-new-instance-sequential.png ├── chartjs-logo.svg └── nodejs-logo.svg ├── testData ├── Anthrope.ttf ├── VTKS UNAMOUR.ttf ├── linux │ ├── font.png │ ├── render-to-buffer-diff.png │ ├── render-to-stream-diff.png │ ├── render-to-buffer-actual.png │ ├── render-to-data-URL-diff.txt │ ├── render-to-stream-actual.png │ ├── chartjs-plugin-annotation.png │ ├── chartjs-plugin-datalabels.png │ ├── no-background-color-actual.png │ ├── no-background-color-diff.png │ ├── chartjs-plugin-annotation-diff.png │ ├── chartjs-plugin-datalabels-diff.png │ ├── chartjs-plugin-annotation-actual.png │ ├── chartjs-plugin-datalabels-actual.png │ ├── render-to-data-URL-actual.txt │ └── render-to-data-URL-compare.html ├── linuxw │ ├── font.png │ ├── background-color.png │ ├── render-to-buffer.png │ ├── render-to-stream.png │ ├── no-background-color.png │ ├── chartjs-plugin-annotation.png │ ├── chartjs-plugin-datalabels.png │ └── render-to-data-URL.txt └── win32 │ ├── font.png │ ├── background-color.png │ ├── render-to-buffer.png │ ├── render-to-stream.png │ ├── no-background-color.png │ ├── chartjs-plugin-annotation.png │ ├── chartjs-plugin-datalabels.png │ └── render-to-data-URL.txt ├── src ├── index.ts ├── global.d.ts ├── freshRequire.ts ├── backgroundColourPlugin.ts ├── freshImport.ts ├── example.ts ├── animatedChartJSNodeCanvas.ts ├── chartJSNodeCanvas.ts ├── chartJSNodeCanvasBase.ts ├── index.spec.ts └── index.e2e.spec.ts ├── tests └── require │ └── wtfnode.js ├── docker ├── production │ └── Dockerfile ├── build │ └── Dockerfile └── base │ └── Dockerfile ├── .c8rc ├── .editorconfig ├── .npmignore ├── scripts ├── publish.sh ├── clean-dest.js └── package-size.js ├── docker-compose.override.yml ├── .dockerignore ├── .mocharc.js ├── CHANGELOG.md ├── .vscode ├── launch.json ├── settings.json └── tasks.json ├── LICENSE ├── docker-compose.yml ├── .github ├── ISSUE_TEMPLATE │ └── bug_report.md └── workflows │ └── nodejs.yml ├── tsconfig.json ├── .gitignore ├── package.json ├── tslint.json ├── API.md └── README.md /.nvmrc: -------------------------------------------------------------------------------- 1 | 21.7.3 -------------------------------------------------------------------------------- /.npmrc: -------------------------------------------------------------------------------- 1 | msvs_version = 2022 -------------------------------------------------------------------------------- /dist/freshRequire.d.ts: -------------------------------------------------------------------------------- 1 | export declare const freshRequire: (id: string) => any; 2 | -------------------------------------------------------------------------------- /codealike.json: -------------------------------------------------------------------------------- 1 | {"projectId":"edd7e570-506c-11e9-af99-077b651e3121","projectName":"chartjs-node-canvas"} -------------------------------------------------------------------------------- /resources/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/logo.png -------------------------------------------------------------------------------- /resources/logo.xcf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/logo.xcf -------------------------------------------------------------------------------- /resources/example.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/example.gif -------------------------------------------------------------------------------- /resources/example.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/example.png -------------------------------------------------------------------------------- /testData/Anthrope.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/Anthrope.ttf -------------------------------------------------------------------------------- /testData/VTKS UNAMOUR.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/VTKS UNAMOUR.ttf -------------------------------------------------------------------------------- /testData/linux/font.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/font.png -------------------------------------------------------------------------------- /testData/linuxw/font.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/font.png -------------------------------------------------------------------------------- /testData/win32/font.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/font.png -------------------------------------------------------------------------------- /testData/win32/background-color.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/background-color.png -------------------------------------------------------------------------------- /testData/win32/render-to-buffer.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/render-to-buffer.png -------------------------------------------------------------------------------- /testData/win32/render-to-stream.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/render-to-stream.png -------------------------------------------------------------------------------- /testData/linuxw/background-color.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/background-color.png -------------------------------------------------------------------------------- /testData/linuxw/render-to-buffer.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/render-to-buffer.png -------------------------------------------------------------------------------- /testData/linuxw/render-to-stream.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/render-to-stream.png -------------------------------------------------------------------------------- /resources/memory-usage-same-instance.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/memory-usage-same-instance.png -------------------------------------------------------------------------------- /testData/linux/render-to-buffer-diff.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/render-to-buffer-diff.png -------------------------------------------------------------------------------- /testData/linux/render-to-stream-diff.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/render-to-stream-diff.png -------------------------------------------------------------------------------- /testData/linuxw/no-background-color.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/no-background-color.png -------------------------------------------------------------------------------- /testData/win32/no-background-color.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/no-background-color.png -------------------------------------------------------------------------------- /testData/linux/render-to-buffer-actual.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/render-to-buffer-actual.png -------------------------------------------------------------------------------- /testData/linux/render-to-data-URL-diff.txt: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/render-to-data-URL-diff.txt -------------------------------------------------------------------------------- /testData/linux/render-to-stream-actual.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/render-to-stream-actual.png -------------------------------------------------------------------------------- /dist/index.d.ts: -------------------------------------------------------------------------------- 1 | export * from './animatedChartJSNodeCanvas'; 2 | export * from './chartJSNodeCanvas'; 3 | export * from './chartJSNodeCanvasBase'; 4 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | export * from './animatedChartJSNodeCanvas'; 2 | export * from './chartJSNodeCanvas'; 3 | export * from './chartJSNodeCanvasBase'; 4 | -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-annotation.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-annotation.png -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-datalabels.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-datalabels.png -------------------------------------------------------------------------------- /testData/linux/no-background-color-actual.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/no-background-color-actual.png -------------------------------------------------------------------------------- /testData/linux/no-background-color-diff.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/no-background-color-diff.png -------------------------------------------------------------------------------- /testData/linuxw/chartjs-plugin-annotation.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/chartjs-plugin-annotation.png -------------------------------------------------------------------------------- /testData/linuxw/chartjs-plugin-datalabels.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linuxw/chartjs-plugin-datalabels.png -------------------------------------------------------------------------------- /testData/win32/chartjs-plugin-annotation.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/chartjs-plugin-annotation.png -------------------------------------------------------------------------------- /testData/win32/chartjs-plugin-datalabels.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/win32/chartjs-plugin-datalabels.png -------------------------------------------------------------------------------- /dist/index.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,sEAA4C;AAC5C,8DAAoC;AACpC,kEAAwC"} -------------------------------------------------------------------------------- /resources/memory-usage-new-instance-parallel.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/memory-usage-new-instance-parallel.png -------------------------------------------------------------------------------- /resources/memory-usage-new-instance-sequential.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/resources/memory-usage-new-instance-sequential.png -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-annotation-diff.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-annotation-diff.png -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-datalabels-diff.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-datalabels-diff.png -------------------------------------------------------------------------------- /tests/require/wtfnode.js: -------------------------------------------------------------------------------- 1 | const wtfnode = require('wtfnode'); 2 | const { after } = require('mocha'); 3 | 4 | after(() => { 5 | wtfnode.dump(); 6 | }); 7 | -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-annotation-actual.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-annotation-actual.png -------------------------------------------------------------------------------- /testData/linux/chartjs-plugin-datalabels-actual.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/SeanSobey/ChartjsNodeCanvas/HEAD/testData/linux/chartjs-plugin-datalabels-actual.png -------------------------------------------------------------------------------- /src/global.d.ts: -------------------------------------------------------------------------------- 1 | // syntax: 2 | // declare module ''; 3 | 4 | declare module 'resemblejs'; // Has a 'declare global' that causes typescript to add a '/// ' 5 | -------------------------------------------------------------------------------- /docker/production/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM chartjs-node-canvas-base 2 | 3 | LABEL maintainer = "sean.m.sobey@gmail.com" 4 | 5 | WORKDIR /usr/server 6 | 7 | COPY --from=chartjs-node-canvas-build /usr/server/dist/ ./dist/ 8 | 9 | CMD ["/bin/bash", "-c", "echo production image complete"] -------------------------------------------------------------------------------- /docker/build/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM chartjs-node-canvas-base 2 | 3 | LABEL maintainer = "sean.m.sobey@gmail.com" 4 | 5 | WORKDIR /usr/server 6 | 7 | RUN npm ci --no-color 8 | 9 | COPY . . 10 | 11 | RUN npm run build 12 | 13 | CMD ["/bin/bash", "-c", "echo base image complete"] -------------------------------------------------------------------------------- /.c8rc: -------------------------------------------------------------------------------- 1 | { 2 | "reporter": [ 3 | "lcov", 4 | "cobertura", 5 | "text-summary", 6 | "json" 7 | ], 8 | "exclude": [ 9 | "dist/**/*.spec.js", 10 | "dist/example.js" 11 | ], 12 | "include": [ 13 | "dist/**/*.js" 14 | ], 15 | "temp-directory": "./coverage", 16 | "reports-dir": "./reports/test-coverage" 17 | } -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | [*.{ts,js,css,html}] 2 | indent_style = tab 3 | tab_width = 4 4 | charset = utf-8 5 | trim_trailing_whitespace = true 6 | insert_final_newline = true 7 | 8 | [*.{ts,js}] 9 | quote_type = single 10 | 11 | [*.{json,yaml,yml}] 12 | indent_style = space 13 | tab_width = 2 14 | 15 | [*.md] 16 | indent_style = space 17 | tab_width = 4 -------------------------------------------------------------------------------- /dist/index.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | Object.defineProperty(exports, "__esModule", { value: true }); 3 | const tslib_1 = require("tslib"); 4 | tslib_1.__exportStar(require("./animatedChartJSNodeCanvas"), exports); 5 | tslib_1.__exportStar(require("./chartJSNodeCanvas"), exports); 6 | tslib_1.__exportStar(require("./chartJSNodeCanvasBase"), exports); 7 | //# sourceMappingURL=index.js.map -------------------------------------------------------------------------------- /src/freshRequire.ts: -------------------------------------------------------------------------------- 1 | // https://github.com/hughsk/fresh-require 2 | 3 | export const freshRequire: /*NodeJS.Require*/ (id: string) => any = (file) => { 4 | 5 | const resolvedFile = require.resolve(file); 6 | const temp = require.cache[resolvedFile]; 7 | delete require.cache[resolvedFile]; 8 | const modified = require(resolvedFile); 9 | require.cache[resolvedFile] = temp; 10 | return modified; 11 | }; 12 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | .vscode/**/* 2 | docker/ 3 | .circleci/ 4 | .github/ 5 | reports/ 6 | scripts/ 7 | tests/ 8 | testData/ 9 | coverage/ 10 | resources/ 11 | 12 | .dockerignore 13 | docker-compose.override.yml 14 | docker-compose.yml 15 | .editorconfig 16 | codealike.json 17 | .mocharc.json 18 | tsconfig.json 19 | tslint.json 20 | .c8rc 21 | .nvmrc 22 | .tsbuildinfo 23 | .mocharc.js 24 | 25 | src/**/*.spec.* 26 | !dist/** 27 | dist/**/*.spec.* -------------------------------------------------------------------------------- /scripts/publish.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | packageName=$(node -p "require('./package.json').name") 4 | packageVersion=$(node -p "require('./package.json').version") 5 | publishVersion=$(npm view "$packageName" version) 6 | if [ "$packageVersion" != "$publishVersion" ] 7 | then 8 | npm config set //registry.npmjs.org/:_authToken=$NPM_AUTH_TOKEN 9 | npm publish 10 | else 11 | echo "Existing version $packageVersion for $packageName is already published...skipping" 12 | fi -------------------------------------------------------------------------------- /docker-compose.override.yml: -------------------------------------------------------------------------------- 1 | version: "3" 2 | 3 | # Override default compose file here. 4 | # https://docs.docker.com/compose/extends/#example-use-case 5 | 6 | 7 | # ci can skip copying the override file or delete it. 8 | # or ci can run command 9 | # docker-compose -f docker-compose.yml -f docker-compose.prod.yml up -d 10 | 11 | # services: 12 | # test: 13 | # command: /bin/bash -c echo skipping tests 14 | # production: 15 | # depends_on: 16 | # - build -------------------------------------------------------------------------------- /dist/freshRequire.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"file":"freshRequire.js","sourceRoot":"","sources":["../src/freshRequire.ts"],"names":[],"mappings":";AAAA,0CAA0C;;;AAEnC,MAAM,YAAY,GAA2C,CAAC,IAAI,EAAE,EAAE;IAE5E,MAAM,YAAY,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IACzC,OAAO,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IACnC,MAAM,QAAQ,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC;IACvC,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC;IACnC,OAAO,QAAQ,CAAC;AACjB,CAAC,CAAC;AARW,QAAA,YAAY,gBAQvB"} -------------------------------------------------------------------------------- /scripts/clean-dest.js: -------------------------------------------------------------------------------- 1 | //@ts-check 2 | 'use-strict'; 3 | 4 | const path = require('path'); 5 | 6 | module.exports = exports = { 7 | '.ts': (destFilePath) => { 8 | const dirname = path.dirname(destFilePath); 9 | const basename = path.basename(destFilePath, path.extname(destFilePath)); 10 | return [ 11 | path.posix.join(dirname, basename + '.d.ts'), 12 | path.posix.join(dirname, basename + '.js'), 13 | path.posix.join(dirname, basename + '.js.map'), 14 | ] 15 | } 16 | }; 17 | -------------------------------------------------------------------------------- /docker/base/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:21.7.3-slim 2 | 3 | LABEL maintainer = "sean.m.sobey@gmail.com" 4 | 5 | RUN apt-get update && apt-get install -y --no-install-recommends \ 6 | # - For node-gyp 7 | # python2 make g++ \ 8 | # - For canvas 9 | fontconfig build-essential libcairo2-dev libpango1.0-dev libjpeg-dev libgif-dev librsvg2-dev 10 | 11 | WORKDIR /usr/server 12 | 13 | COPY package.json package-lock.json ./ 14 | 15 | RUN npm ci --production --no-color 16 | 17 | CMD ["/bin/bash", "-c", "echo base image complete"] -------------------------------------------------------------------------------- /.dockerignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | 6 | # Dependency directories 7 | **/node_modules/** 8 | 9 | # Git 10 | .git/ 11 | .gitignore 12 | .gitattributes 13 | 14 | # VSCode 15 | .vscode/ 16 | 17 | #dist 18 | dist/ 19 | 20 | # Coverage directory used by tools like istanbul 21 | coverage 22 | 23 | # nyc test coverage 24 | .nyc_output 25 | 26 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 27 | .grunt 28 | 29 | #docker 30 | docker/ 31 | docker-compose.yml 32 | docker-compose.override.yml -------------------------------------------------------------------------------- /dist/freshRequire.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | // https://github.com/hughsk/fresh-require 3 | Object.defineProperty(exports, "__esModule", { value: true }); 4 | exports.freshRequire = void 0; 5 | const freshRequire = (file) => { 6 | const resolvedFile = require.resolve(file); 7 | const temp = require.cache[resolvedFile]; 8 | delete require.cache[resolvedFile]; 9 | const modified = require(resolvedFile); 10 | require.cache[resolvedFile] = temp; 11 | return modified; 12 | }; 13 | exports.freshRequire = freshRequire; 14 | //# sourceMappingURL=freshRequire.js.map -------------------------------------------------------------------------------- /.mocharc.js: -------------------------------------------------------------------------------- 1 | //@ts-check 2 | 'use strict'; 3 | 4 | // https://mochajs.org/#command-line-usage 5 | 6 | const NODE_ENV = process.env.NODE_ENV; // 'production' | 'ci' | 'test' | 'debug' 7 | 8 | /**@type {import('mocha').MochaOptions}*/ 9 | module.exports = { 10 | 'ignore-leaks': false, 11 | 'allow-uncaught': true, 12 | 'globals': [ 13 | 'window' 14 | ], 15 | 'require': [ 16 | 'source-map-support/register', 17 | ], 18 | 'timeout': 50000, 19 | 'file': NODE_ENV === 'debug' 20 | ? [] 21 | : [ 22 | './tests/require/wtfnode.js' 23 | ], 24 | }; 25 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | ## 5.0.0 4 | 5 | Migrated to support chart.js v4.x.x, dropped support for 3.x.x 6 | Upgraded to canvas v3.x.x 7 | 8 | WIP support for animated charts, exporting to GIF. 9 | 10 | ## 4.0.0 11 | 12 | Migrated to support chart.js v3.x.x, dropped support for 2.x.x 13 | Removed the legacy API. 14 | 15 | ## 3.2.0 16 | 17 | Another deploy for minor versions. 18 | 19 | ## 3.1.1 20 | 21 | Added back legacy API to address a wrong version being deployed. See [#60](https://github.com/SeanSobey/ChartjsNodeCanvas/issues/60). 22 | 23 | ## 3.0.0 24 | 25 | Rewrote `ChartJSNodeCanvas` API, the constructor and plugins in particular. 26 | -------------------------------------------------------------------------------- /src/backgroundColourPlugin.ts: -------------------------------------------------------------------------------- 1 | // @ts-expect-error moduleResolution:nodenext issue 54523 2 | import { Chart as ChartJS, Plugin as ChartJSPlugin } from 'chart.js/auto'; 3 | 4 | export class BackgroundColourPlugin implements ChartJSPlugin { 5 | public readonly id: string = 'chartjs-plugin-chartjs-node-canvas-background-colour'; 6 | 7 | public constructor( 8 | private readonly _width: number, 9 | private readonly _height: number, 10 | private readonly _fillStyle: string 11 | ) { } 12 | 13 | public beforeDraw(chart: ChartJS): boolean | void { 14 | 15 | const ctx = chart.ctx; 16 | ctx.save(); 17 | ctx.globalCompositeOperation = 'destination-over'; 18 | ctx.fillStyle = this._fillStyle; 19 | ctx.fillRect(0, 0, this._width, this._height); 20 | ctx.restore(); 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /src/freshImport.ts: -------------------------------------------------------------------------------- 1 | // ES Module version of fresh-require 2 | // export const freshImport = async (modulePath: string): Promise => { 3 | // // For ES modules, we need to construct the full URL 4 | // const moduleUrl = new URL(modulePath, import.meta.url).href; 5 | 6 | // // Clear module from import cache if possible 7 | // // Note: ES modules caching works differently than CommonJS 8 | // try { 9 | // // Force a new module instance by appending a cache-busting query parameter 10 | // const timestamp = Date.now(); 11 | // const urlWithCacheBuster = `${moduleUrl}?cache=${timestamp}`; 12 | 13 | // // Dynamic import with cache buster 14 | // const module = await import(/* @vite-ignore */ urlWithCacheBuster); 15 | // return module; 16 | // } catch (error) { 17 | // console.error(`Error importing module ${modulePath}:`, error); 18 | // throw error; 19 | // } 20 | // }; 21 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "name": "Launch Tests", 9 | "type": "node", 10 | "request": "launch", 11 | "cwd": "${workspaceRoot}", 12 | "runtimeExecutable": "npm", 13 | "windows": { 14 | "runtimeExecutable": "npm.cmd" 15 | }, 16 | "runtimeArgs": [ 17 | "run-script", 18 | "debug-test" 19 | ], 20 | "preLaunchTask": "nvm", 21 | "timeout": 50000, 22 | "port": 33295, 23 | "sourceMaps": true, 24 | "outFiles": [ 25 | "${workspaceRoot}/dist/**/*.js" 26 | ], 27 | "skipFiles": [ 28 | "/**" 29 | ], 30 | "env": { 31 | "NODE_ENV": "debug" 32 | } 33 | } 34 | ] 35 | } -------------------------------------------------------------------------------- /.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "typescript.preferences.importModuleSpecifier": "relative", 3 | "typescript.preferences.quoteStyle": "single", 4 | "files.associations": { 5 | ".c8rc": "json", 6 | ".dockerignore": "gitignore", 7 | ".env.*": "dotenv", 8 | ".nycrc": "json", 9 | "Dockerfile.*": "dockerfile" 10 | }, 11 | "eslint.validate": [ 12 | "javascript", 13 | "javascriptreact", 14 | "typescript", 15 | "typescriptreact", 16 | ], 17 | "mochaExplorer.debuggerConfig": "Launch Tests", 18 | "mochaExplorer.files": "dist/**/*.js", 19 | "mochaExplorer.require": "source-map-support/register", 20 | "mocha.enabled": true, 21 | "mocha.options": { 22 | "checkLeaks": true, 23 | "throwDeprecation": true, 24 | "traceDeprecation": true, 25 | "traceWarnings": true, 26 | "allowUncaught": true 27 | }, 28 | "mocha.sourceDir": "src/", 29 | "mocha.outputDir": "dist/", 30 | "mocha.glob": "**/*.spec.js", 31 | "mocha.compilerScript": "build", 32 | "mocha.requires": [ 33 | "source-map-support/register" 34 | ], 35 | "deepcode.review.results.hideInformationIssues": false 36 | } -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 Sean Sobey 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 | -------------------------------------------------------------------------------- /docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: "3" 2 | 3 | services: 4 | base: 5 | image: chartjs-node-canvas-base 6 | build: 7 | context: . 8 | dockerfile: docker/base/Dockerfile 9 | container_name: chartjs-node-canvas-base 10 | build: 11 | image: chartjs-node-canvas-build 12 | depends_on: 13 | - base 14 | build: 15 | context: . 16 | dockerfile: docker/build/Dockerfile 17 | container_name: chartjs-node-canvas-build 18 | test: 19 | image: chartjs-node-canvas-build 20 | container_name: chartjs-node-canvas-test 21 | depends_on: 22 | - build 23 | environment: 24 | - NODE_ENV=test 25 | - FONTCONFIG_PATH=/etc/fonts 26 | command: npm run test 27 | production: 28 | image: chartjs-node-canvas 29 | depends_on: 30 | - test 31 | build: 32 | context: . 33 | dockerfile: docker/production/Dockerfile 34 | container_name: chartjs-node-canvas 35 | environment: 36 | - NODE_ENV=production 37 | 38 | #<< Docker Commands >> 39 | 40 | # Build Images 41 | #docker-compose build 42 | 43 | # Run Containers 44 | #docker-compose up 45 | 46 | # Run Tests specifically 47 | #docker-compose up test 48 | -------------------------------------------------------------------------------- /resources/chartjs-logo.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 7 | 9 | 12 | 14 | 15 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/bug_report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Bug report 3 | about: Create a report to help us improve 4 | title: '' 5 | labels: '' 6 | assignees: '' 7 | 8 | --- 9 | 10 | **Troubleshooting Checklist** 11 | * You have a supported [Node JS version](https://github.com/SeanSobey/ChartjsNodeCanvas#node-js-version). 12 | * You have a supported [Chart.JS version](https://github.com/SeanSobey/ChartjsNodeCanvas#chartsjs-version). 13 | * If this is plugin related: 14 | - You have read the [plugin docs](https://github.com/SeanSobey/ChartjsNodeCanvas#loading-plugins). 15 | - You have read the chart.js [plugin docs](https://www.chartjs.org/docs/latest/developers/plugins.html). 16 | - You have searched for [similar plugin issues](https://github.com/SeanSobey/ChartjsNodeCanvas/issues?q=is%3Aissue+label%3Aplugin). 17 | * If this is deployment/environment related: 18 | - You have read the [canvas docs](https://github.com/Automattic/node-canvas) and followed any environment specific steps. 19 | - You have searched for [similar deployment issues](https://github.com/SeanSobey/ChartjsNodeCanvas/issues?q=is%3Aissue+label%3Adeployment). 20 | 21 | 22 | **Describe the bug** 23 | A clear and concise description of what the bug is. 24 | 25 | **Versions** 26 | * NodeJS version: x.x.x 27 | * Chart.JS version: x.x.x 28 | * Typescript version (if applicable): x.x.x 29 | 30 | **Additional context** 31 | Add any other context about the problem here. 32 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | /* Basic Options */ 4 | "target": "es2017", 5 | "module": "Node16", 6 | "declaration": true, 7 | "sourceMap": true, 8 | "outDir": "./dist/", 9 | "importHelpers": true, 10 | "incremental": true, 11 | "tsBuildInfoFile": "./.tsbuildinfo", 12 | "noEmitOnError": false, 13 | 14 | /* Strict Type-Checking Options */ 15 | "strict": true, 16 | "noImplicitAny": true, 17 | "strictNullChecks": true, 18 | "strictFunctionTypes": true, 19 | "strictPropertyInitialization": true, 20 | "noImplicitThis": true, 21 | "alwaysStrict": true, 22 | 23 | /* Additional Checks */ 24 | "noUnusedLocals": false, 25 | "noUnusedParameters": true, 26 | "noImplicitReturns": true, 27 | "noFallthroughCasesInSwitch": true, 28 | "skipLibCheck": true, 29 | 30 | /* Module Resolution Options */ 31 | "moduleResolution": "node16", 32 | "baseUrl": "./", 33 | "allowSyntheticDefaultImports": true, 34 | "esModuleInterop": true, 35 | "resolveJsonModule": true, 36 | "typeRoots": [ 37 | "./node_modules/@types", 38 | "./node_modules/chart.js/auto" 39 | ], 40 | 41 | /* Source Map Options */ 42 | "inlineSourceMap": false, 43 | "inlineSources": false, 44 | 45 | /* Experimental Options */ 46 | "experimentalDecorators": true, 47 | "emitDecoratorMetadata": true, 48 | "pretty": true 49 | }, 50 | "include": [ 51 | "./src/**/*" 52 | ], 53 | "buildOptions": { 54 | "verbose": true 55 | } 56 | } -------------------------------------------------------------------------------- /.vscode/tasks.json: -------------------------------------------------------------------------------- 1 | { 2 | // See https://go.microsoft.com/fwlink/?LinkId=733558 3 | // for the documentation about the tasks.json format 4 | "version": "2.0.0", 5 | "tasks": [ 6 | { 7 | "type": "shell", 8 | "windows": { 9 | "command": "nvm use $(Get-Content .nvmrc)", 10 | }, 11 | "linux":{ 12 | "command": "cat .nvmrc | nvm use" 13 | }, 14 | "osx":{ 15 | "command": "cat .nvmrc | nvm use" 16 | }, 17 | "label": "nvm", 18 | }, 19 | { 20 | "type": "npm", 21 | "script": "build", 22 | "dependsOn": "nvm", 23 | "group": { 24 | "kind": "build", 25 | "isDefault": true 26 | }, 27 | "problemMatcher": "$tsc" 28 | }, 29 | { 30 | "type": "npm", 31 | "script": "test", 32 | "dependsOn": "nvm", 33 | "group": { 34 | "kind": "test", 35 | "isDefault": true 36 | } 37 | }, 38 | { 39 | "type": "npm", 40 | "script": "clean", 41 | "dependsOn": "nvm" 42 | }, 43 | { 44 | "type": "npm", 45 | "script": "lint", 46 | "dependsOn": "nvm", 47 | "problemMatcher": "$eslint-stylish" 48 | }, 49 | { 50 | "type": "npm", 51 | "script": "docs" 52 | }, 53 | { 54 | "type": "npm", 55 | "script": "watch-build", 56 | "dependsOn": "nvm", 57 | "group": "build", 58 | "isBackground": true, 59 | "problemMatcher": "$tsc" 60 | }, 61 | { 62 | "type": "npm", 63 | "script": "watch-clean", 64 | "isBackground": true 65 | }, 66 | { 67 | "type": "npm", 68 | "script": "watch-test", 69 | "dependsOn": "nvm", 70 | "isBackground": true, 71 | "group": "test" 72 | }, 73 | { 74 | "type": "npm", 75 | "script": "watch-lint", 76 | "dependsOn": "nvm", 77 | "isBackground": true, 78 | "problemMatcher": "$eslint-stylish" 79 | } 80 | ] 81 | } -------------------------------------------------------------------------------- /scripts/package-size.js: -------------------------------------------------------------------------------- 1 | //@ts-check 2 | 'use-strict'; 3 | 4 | // const yargs = require('yargs'); 5 | const { exec } = require('child_process'); 6 | const { EOL } = require('os'); 7 | const { promisify } = require('util'); 8 | 9 | const execAsync = promisify(exec); 10 | 11 | // const argv = yargs.options({}) 12 | // .strict() 13 | // .config() 14 | // .argv; 15 | 16 | async function main() { 17 | 18 | const maxIncreaseDelta = 0.05; 19 | const packageInfo = require('../package.json'); 20 | const package = await dryRunPack(); 21 | const publishedPackage = await dryRunPack(packageInfo.name); 22 | const increaseDelta = (package.size - publishedPackage.size) / publishedPackage.size; 23 | if (increaseDelta > maxIncreaseDelta) { 24 | const packageContents = package.files.map(file => file.path); 25 | throw new Error(`Package increased in size by ${Math.round(increaseDelta * 100)}%, (${package.size - publishedPackage.size} bytes)${EOL}== Tarball Contents ===${EOL}${packageContents.join(EOL)}`); 26 | } 27 | console.log('Package size is within delta threshold', { publishedPackageSize: publishedPackage.size, packageSize: package.size, increaseDelta }); 28 | } 29 | 30 | /** 31 | * @param {string} name 32 | * @param {string} version 33 | * @returns {Promise} 34 | */ 35 | async function dryRunPack(name = null, version = null) { 36 | 37 | // https://docs.npmjs.com/cli-commands/pack.html 38 | const package = name 39 | ? `${name}@${version || 'latest'}` 40 | : ''; 41 | const { stderr, stdout } = await execAsync(`npm pack ${package} --dry-run --json`); 42 | try { 43 | const data = JSON.parse(stdout); 44 | if (!data || !Array.isArray(data)) { 45 | throw new Error(`Bad json data: ${EOL}${data}${EOL}json: ${EOL}${stdout}`); 46 | } 47 | if (data.length !== 1) { 48 | throw new Error(`Bad array data: ${EOL}${data}${EOL}`); 49 | } 50 | return data[0]; 51 | } catch (error) { 52 | if (error instanceof SyntaxError) { 53 | throw new Error(`Failed to parse json with error: ${EOL}${error}${EOL}json: ${EOL}${stdout}`) 54 | } 55 | throw error; 56 | } 57 | } 58 | 59 | main() 60 | .catch((error) => console.error(error)); 61 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | 9 | # Diagnostic reports (https://nodejs.org/api/report.html) 10 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 11 | 12 | # Runtime data 13 | pids 14 | *.pid 15 | *.seed 16 | *.pid.lock 17 | 18 | # Directory for instrumented libs generated by jscoverage/JSCover 19 | lib-cov 20 | 21 | # Coverage directory used by tools like istanbul 22 | coverage 23 | *.lcov 24 | 25 | # nyc test coverage 26 | .nyc_output 27 | 28 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 29 | .grunt 30 | 31 | # Bower dependency directory (https://bower.io/) 32 | bower_components 33 | 34 | # node-waf configuration 35 | .lock-wscript 36 | 37 | # Compiled binary addons (http://nodejs.org/api/addons.html) 38 | build/Release 39 | 40 | # Dependency directories 41 | node_modules/ 42 | jspm_packages/ 43 | 44 | # TypeScript v1 declaration files 45 | typings/ 46 | 47 | # TypeScript cache 48 | *.tsbuildinfo 49 | 50 | # Optional npm cache directory 51 | .npm 52 | 53 | # Optional eslint cache 54 | .eslintcache 55 | 56 | # Microbundle cache 57 | .rpt2_cache/ 58 | .rts2_cache_cjs/ 59 | .rts2_cache_es/ 60 | .rts2_cache_umd/ 61 | 62 | # Optional REPL history 63 | .node_repl_history 64 | 65 | # Output of 'npm pack' 66 | *.tgz 67 | 68 | # Yarn Integrity file 69 | .yarn-integrity 70 | 71 | # dotenv environment variables file 72 | .env 73 | .env.test 74 | 75 | # parcel-bundler cache (https://parceljs.org/) 76 | .cache 77 | 78 | # Next.js build output 79 | .next 80 | 81 | # Nuxt.js build / generate output 82 | .nuxt 83 | 84 | # Gatsby files 85 | .cache/ 86 | # Comment in the public line in if your project uses Gatsby and *not* Next.js 87 | # https://nextjs.org/blog/next-9-1#public-directory-support 88 | # public 89 | 90 | # vuepress build output 91 | .vuepress/dist 92 | 93 | # Serverless directories 94 | .serverless/ 95 | 96 | # FuseBox cache 97 | .fusebox/ 98 | 99 | # DynamoDB Local files 100 | .dynamodb/ 101 | 102 | # TernJS port file 103 | .tern-port 104 | 105 | # reports 106 | /reports/** 107 | !/reports/index.html 108 | 109 | #dist 110 | dist/** 111 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "chartjs-node-canvas", 3 | "version": "5.0.0", 4 | "description": "A node renderer for Chart.js using canvas.", 5 | "main": "./dist/index", 6 | "type": "commonjs", 7 | "types": "./dist/index.d.ts", 8 | "scripts": { 9 | "nvm": "cat .nvmrc | nvm use", 10 | "build": "tsc", 11 | "clean": "clean-dest -s ./src -d ./dist --file-map ./scripts/clean-dest --permanent --verbose && echo TODO: Delete .tsbuildinfo?", 12 | "lint": "echo TODO: Add linting", 13 | "test": "c8 --all mocha dist/**/*.spec.js", 14 | "test-unit": "mocha --exclude dist/**/*.e2e.spec.js dist/**/*.spec.js", 15 | "test-e2e": "mocha dist/**/*.e2e.spec.js", 16 | "package-size": "node ./scripts/package-size", 17 | "debug-test": "node --inspect-brk=33295 --nolazy node_modules/mocha/bin/_mocha dist/**/*.spec.js", 18 | "watch-build": "tsc --watch", 19 | "watch-clean": "nodemon --watch ./src -e ts --exec npm run-script clean", 20 | "watch-test": "nodemon --watch ./dist -e js --exec npm run-script test", 21 | "docs": "jsdoc2md dist/index.js > API.md" 22 | }, 23 | "bugs": { 24 | "url": "https://github.com/SeanSobey/ChartjsNodeCanvas/issues", 25 | "email": "sean.m.sobey@gmail.com" 26 | }, 27 | "author": { 28 | "name": "Sean Sobey", 29 | "email": "sean.m.sobey@gmail.com" 30 | }, 31 | "homepage": "https://github.com/SeanSobey/ChartjsNodeCanvas", 32 | "repository": { 33 | "type": "git", 34 | "url": "https://github.com/SeanSobey/ChartjsNodeCanvas.git" 35 | }, 36 | "readme": "README.md", 37 | "license": "MIT", 38 | "dependencies": { 39 | "canvas": "^3.1.0", 40 | "tslib": "^2.8.1" 41 | }, 42 | "peerDependencies": { 43 | "chart.js": "^4.4.8" 44 | }, 45 | "devDependencies": { 46 | "@types/mocha": "^7.0.2", 47 | "@types/node": "^16.10.4", 48 | "@types/offscreencanvas": "^2019.7.3", 49 | "c8": "^7.10.0", 50 | "chart.js": "^4.4.8", 51 | "chartjs-plugin-annotation": "^3.1.0", 52 | "chartjs-plugin-crosshair": "^2.0.0", 53 | "chartjs-plugin-datalabels": "^2.2.0", 54 | "clean-dest": "^1.3.3", 55 | "gifencoder": "^2.0.1", 56 | "jsdoc-to-markdown": "^5.0.3", 57 | "mocha": "^7.2.0", 58 | "nodemon": "^2.0.13", 59 | "release-it": "^14.11.6", 60 | "resemblejs": "^4.0.0", 61 | "source-map-support": "^0.5.20", 62 | "ts-std-lib": "^1.2.2", 63 | "typescript": "^5.7.3", 64 | "wtfnode": "^0.9.1" 65 | } 66 | } 67 | -------------------------------------------------------------------------------- /src/example.ts: -------------------------------------------------------------------------------- 1 | import { ChartJSNodeCanvas, ChartCallback } from './'; 2 | // @ts-expect-error moduleResolution:nodenext issue 54523 3 | import { ChartConfiguration } from 'chart.js/auto'; 4 | import path from 'path'; 5 | import { promises as fs } from 'fs'; 6 | 7 | async function main(): Promise { 8 | 9 | const width = 400; 10 | const height = 400; 11 | const configuration: ChartConfiguration = { 12 | type: 'bar', 13 | data: { 14 | labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], 15 | datasets: [{ 16 | label: '# of Votes', 17 | data: [12, 19, 3, 5, 2, 3], 18 | backgroundColor: [ 19 | 'rgba(255, 99, 132, 0.2)', 20 | 'rgba(54, 162, 235, 0.2)', 21 | 'rgba(255, 206, 86, 0.2)', 22 | 'rgba(75, 192, 192, 0.2)', 23 | 'rgba(153, 102, 255, 0.2)', 24 | 'rgba(255, 159, 64, 0.2)' 25 | ], 26 | borderColor: [ 27 | 'rgba(255,99,132,1)', 28 | 'rgba(54, 162, 235, 1)', 29 | 'rgba(255, 206, 86, 1)', 30 | 'rgba(75, 192, 192, 1)', 31 | 'rgba(153, 102, 255, 1)', 32 | 'rgba(255, 159, 64, 1)' 33 | ], 34 | borderWidth: 1 35 | }] 36 | }, 37 | options: { 38 | }, 39 | plugins: [{ 40 | id: 'background-colour', 41 | beforeDraw: (chart) => { 42 | const ctx = chart.ctx; 43 | ctx.save(); 44 | ctx.fillStyle = 'white'; 45 | ctx.fillRect(0, 0, width, height); 46 | ctx.restore(); 47 | } 48 | }] 49 | }; 50 | const chartCallback: ChartCallback = (ChartJS) => { 51 | ChartJS.defaults.responsive = true; 52 | ChartJS.defaults.maintainAspectRatio = false; 53 | }; 54 | console.log('here'); 55 | 56 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback }); 57 | const buffer = await chartJSNodeCanvas.renderToBuffer(configuration); 58 | await fs.writeFile('./resources/example.png', buffer, 'base64'); 59 | 60 | const k = Object.keys(require.cache).find(key => key.includes(path.join('node_modules','chart.js'))); 61 | console.log('keys', k); 62 | 63 | // const animatedChartJSNodeCanvas = new AnimatedChartJSNodeCanvas({ width, height, chartCallback }); 64 | // const buffers = await animatedChartJSNodeCanvas.renderToBuffer(configuration); 65 | // const { Gif } = await import('make-a-gif'); 66 | // const gif = new Gif(width, height, 1); 67 | // const totalDuration = 1000; 68 | // const duration = totalDuration / buffers.length; 69 | // await gif.setFrames(buffers.map(buffer => ({ src: new Uint8Array(buffer), duration }))); 70 | // // const data = await chartJSNodeCanvas.renderToDataURL(configuration); 71 | // // console.log(data.length); 72 | // const image = await gif.encode(); 73 | // await fs.writeFile('./resources/example.gif', image); 74 | } 75 | main(); 76 | -------------------------------------------------------------------------------- /.github/workflows/nodejs.yml: -------------------------------------------------------------------------------- 1 | name: Node CI 2 | 3 | on: 4 | push: 5 | branches: 6 | - master 7 | pull_request: 8 | branches: 9 | - master 10 | 11 | jobs: 12 | build-and-test: 13 | runs-on: windows-latest 14 | strategy: 15 | matrix: 16 | node-version: ['20.x', '21.x'] 17 | steps: 18 | - uses: actions/checkout@v2 19 | - name: Use Node.js ${{ matrix.node-version }} 20 | uses: actions/setup-node@v2 21 | with: 22 | node-version: ${{ matrix.node-version }} 23 | - name: Install Cairo dependencies 24 | run: | 25 | Invoke-WebRequest "http://ftp.gnome.org/pub/GNOME/binaries/win64/gtk+/2.22/gtk+-bundle_2.22.1-20101229_win64.zip" -OutFile "gtk.zip" 26 | Expand-Archive gtk.zip -DestinationPath "C:\GTK" 27 | Invoke-WebRequest "https://downloads.sourceforge.net/project/libjpeg-turbo/2.0.4/libjpeg-turbo-2.0.4-vc64.exe" -OutFile "libjpeg.exe" -UserAgent NativeHost 28 | .\libjpeg.exe /S 29 | - name: Install Anthrope font 30 | run: | 31 | Copy-Item -Path "${{ github.workspace }}\\testData\\Anthrope.ttf" -Destination "$Env:WINDIR\Fonts" 32 | reg add "HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Fonts" /v "Anthrope (TrueType)" /t REG_SZ /d "Anthrope.ttf" /f 33 | - name: Add msbuild to PATH 34 | uses: microsoft/setup-msbuild@v1.1 35 | with: 36 | vs-version: '15.0' 37 | - name: NPM install 38 | run: npm ci 39 | - name: Build 40 | run: npm run-script build 41 | - name: Lint 42 | run: npm run-script lint 43 | - name: Test 44 | run: npm run-script test 45 | - uses: codecov/codecov-action@v1 46 | with: 47 | name: Node v${{ matrix.node-version }} 48 | token: ${{ secrets.CODECOV_TOKEN }} 49 | file: ./reports/test-coverage/cobertura-coverage.xml 50 | fail_ci_if_error: true 51 | # Does not yet support wildcards! Needs v2: https://github.com/actions/upload-artifact/pull/54 52 | # When ready see https://help.github.com/en/actions/configuring-and-managing-workflows/persisting-workflow-data-using-artifacts#uploading-build-and-test-artifacts 53 | # - name: Package Artifact 54 | # run: npm pack 55 | # - name: Upload Artifact 56 | # uses: actions/upload-artifact@v1 57 | # with: 58 | # name: Node v${{ matrix.node-version }} 59 | # path: '*.tgz' 60 | publish: 61 | if: github.ref == 'refs/heads/master' 62 | needs: build-and-test 63 | runs-on: windows-latest 64 | strategy: 65 | matrix: 66 | node-version: ['21.x'] 67 | steps: 68 | - uses: actions/checkout@v2 69 | - name: Use Node.js ${{ matrix.node-version }} 70 | uses: actions/setup-node@v2 71 | with: 72 | node-version: ${{ matrix.node-version }} 73 | - name: NPM install 74 | run: npm ci 75 | - name: Build 76 | run: npm run-script build 77 | - name: Check Size 78 | run: npm run-script package-size 79 | - name: Publish 80 | shell: bash 81 | run: ./scripts/publish.sh 82 | env: 83 | CI: true 84 | NPM_AUTH_TOKEN: ${{ secrets.NPM_AUTH_TOKEN }} -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": [ 3 | "tslint:recommended", 4 | "tslint-eslint-rules", 5 | "tslint-immutable", 6 | "tslint-divid" 7 | ], 8 | "rules": { 9 | "arrow-parens": false, 10 | "array-type": [ 11 | true, 12 | "generic" 13 | ], 14 | "ban": [ 15 | true, 16 | [ 17 | "_", 18 | "extend" 19 | ], 20 | [ 21 | "_", 22 | "isNull" 23 | ], 24 | [ 25 | "_", 26 | "isDefined" 27 | ] 28 | ], 29 | "class-name": true, 30 | "comment-format": [ 31 | false 32 | ], 33 | "curly": true, 34 | "eofline": false, 35 | "forin": true, 36 | "indent": [ 37 | true, 38 | "tabs", 39 | 4 40 | ], 41 | "interface-name": [ 42 | false 43 | ], 44 | "interface-over-type-literal": false, 45 | "jsdoc-format": true, 46 | "label-position": true, 47 | "max-line-length": [ 48 | false 49 | ], 50 | "max-classes-per-file": [ 51 | false 52 | ], 53 | "no-arg": true, 54 | "no-bitwise": true, 55 | "no-console": [ 56 | true, 57 | "debug", 58 | "info", 59 | "time", 60 | "timeEnd", 61 | "trace" 62 | ], 63 | "no-construct": true, 64 | "no-debugger": true, 65 | "no-duplicate-variable": true, 66 | "no-empty": true, 67 | "no-empty-interface": false, 68 | "no-eval": true, 69 | "no-inferrable-types": [ 70 | true, 71 | "ignore-params" 72 | ], 73 | "no-string-literal": true, 74 | "no-trailing-whitespace": true, 75 | "no-unused-expression": true, 76 | "no-unused-variable": true, 77 | "no-use-before-declare": false, 78 | "no-var-requires": false, 79 | "object-literal-sort-keys": false, 80 | "one-line": [ 81 | true, 82 | "check-open-brace", 83 | "check-catch", 84 | "check-else", 85 | "check-whitespace" 86 | ], 87 | "ordered-imports": [ 88 | false 89 | ], 90 | "quotemark": [ 91 | true, 92 | "single" 93 | ], 94 | "radix": true, 95 | "semicolon": [ 96 | true, 97 | "always" 98 | ], 99 | "trailing-comma": [ 100 | false 101 | ], 102 | "triple-equals": [ 103 | true, 104 | "allow-null-check" 105 | ], 106 | "typedef": [ 107 | true, 108 | "call-signature", 109 | "parameter", 110 | "property-declaration" 111 | ], 112 | "typedef-whitespace": [ 113 | true, 114 | { 115 | "call-signature": "nospace", 116 | "index-signature": "nospace", 117 | "parameter": "nospace", 118 | "property-declaration": "nospace", 119 | "variable-declaration": "nospace" 120 | }, 121 | { 122 | "call-signature": "onespace", 123 | "index-signature": "onespace", 124 | "parameter": "onespace", 125 | "property-declaration": "onespace", 126 | "variable-declaration": "onespace" 127 | } 128 | ], 129 | "variable-name": false, 130 | "whitespace": [ 131 | false, 132 | "check-branch", 133 | "check-decl", 134 | "check-operator", 135 | "check-separator", 136 | "check-type" 137 | ], 138 | //https://github.com/jonaskello/tslint-immutable 139 | // :: Immutability rules :: 140 | "readonly-keyword": [ 141 | true, 142 | "ignore-local" 143 | ], 144 | "readonly-array": [ 145 | true, 146 | "ignore-local" 147 | ], 148 | "no-let": true, 149 | // :: Functional style rules :: 150 | //"no-this": true, 151 | //"no-class": true, 152 | //"no-mixed-interface": true, 153 | //"no-expression-statement": true, 154 | //https://github.com/jonaskello/tslint-divid 155 | // :: Other rules :: 156 | "no-arguments": true, 157 | "no-label": true, 158 | //"no-semicolon-interface": true, 159 | "import-containment": [ 160 | true, 161 | { 162 | "containmentPath": "path/to/libs", 163 | "allowedExternalFileNames": [ 164 | "index" 165 | ], 166 | "disallowedInternalFileNames": [ 167 | "index" 168 | ] 169 | } 170 | ] 171 | }, 172 | "defaultSeverity": "warning" 173 | } -------------------------------------------------------------------------------- /src/animatedChartJSNodeCanvas.ts: -------------------------------------------------------------------------------- 1 | // @ts-expect-error moduleResolution:nodenext issue 54523 2 | import { Chart as ChartJS, ChartConfiguration } from 'chart.js/auto'; 3 | import { ChartJSNodeCanvasBase, MimeType, Canvas } from './chartJSNodeCanvasBase'; 4 | 5 | const animationFrameProvider: AnimationFrameProvider = { 6 | cancelAnimationFrame: (handle) => clearImmediate(handle as any), 7 | requestAnimationFrame: (callback) => setImmediate(() => callback(Date.now())) as any, 8 | }; 9 | 10 | type OnProgress = (chart: ChartJS, progress: number, initial: boolean) => void; 11 | type OnComplete = (chart: ChartJS, initial: boolean) => void; 12 | 13 | export class AnimatedChartJSNodeCanvas extends ChartJSNodeCanvasBase { 14 | 15 | /** 16 | * Render to a data url array. 17 | * @see https://github.com/Automattic/node-canvas#canvastodataurl 18 | * 19 | * @param configuration The Chart JS configuration for the chart to render. 20 | * @param mimeType A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support), `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom), `application/pdf` (for PDF canvases) and image/svg+xml (for SVG canvases). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. 21 | */ 22 | public renderToDataURL(configuration: ChartConfiguration, mimeType: MimeType = 'image/png'): Promise> { 23 | 24 | const frames: Array = []; 25 | return new Promise((resolve, _reject) => { 26 | this.renderChart(configuration, (chart) => { 27 | 28 | const canvas = chart.canvas as Canvas; 29 | if (!canvas) { 30 | throw new Error('Canvas is null'); 31 | } 32 | const dataUrl = canvas.toDataURL(mimeType); 33 | frames.push(dataUrl); 34 | }, (chart) => { 35 | 36 | resolve(frames); 37 | chart.destroy(); 38 | }); 39 | }); 40 | } 41 | 42 | /** 43 | * Render to a buffer. 44 | * @see https://github.com/Automattic/node-canvas#canvastobuffer 45 | * 46 | * @param configuration The Chart JS configuration for the chart to render. 47 | * @param mimeType A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support) or `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. 48 | */ 49 | public renderToBuffer(configuration: ChartConfiguration, mimeType: MimeType = 'image/png'): Promise> { 50 | 51 | return new Promise((resolve, _reject) => { 52 | 53 | const frames: Array = []; 54 | this.renderChart(configuration, (chart) => { 55 | 56 | const canvas = chart.canvas as Canvas; 57 | if (!canvas) { 58 | throw new Error('Canvas is null'); 59 | } 60 | const buffer = canvas.toBuffer(mimeType); 61 | frames.push(buffer); 62 | }, (chart) => { 63 | 64 | resolve(frames); 65 | chart.destroy(); 66 | }); 67 | }); 68 | } 69 | 70 | private renderChart(configuration: ChartConfiguration, onProgress: OnProgress, onComplete: OnComplete): ChartJS { 71 | 72 | const canvas = this._createCanvas(this._width, this._height, this._type); 73 | (canvas as any).style = (canvas as any).style || {}; 74 | const options = Object.assign({}, configuration.options); 75 | options.responsive = false; 76 | const animation = options.animation || {}; 77 | if (!animation.duration) { 78 | animation.duration = 1000; 79 | } 80 | const baseOnProgress = animation.onProgress; 81 | animation.onProgress = (event) => { 82 | const currentStep: number = (event as any).currentStep; // type docs wrong? 83 | const initial = !!(event as any).initial ? (event as any).initial : false; // added around 3.2.x 84 | const progress = currentStep / event.numSteps; 85 | if (baseOnProgress) { 86 | //baseOnComplete(event.chart); 87 | baseOnProgress.call(animation as any, event); 88 | } 89 | onProgress(event.chart, progress, initial); 90 | }; 91 | const baseOnComplete = animation.onProgress; 92 | animation.onComplete = (event) => { 93 | const initial = !!(event as any).initial ? (event as any).initial : false; // added around 3.2.x 94 | if (baseOnComplete) { 95 | //baseOnComplete(event.chart); 96 | baseOnComplete.call(animation as any, event); 97 | } 98 | onComplete(event.chart, initial); 99 | }; 100 | const plugins = configuration.plugins || []; 101 | const configuredChartConfig = { ...configuration, options, plugins }; 102 | global.window = global.window || {}; 103 | global.window.requestAnimationFrame = animationFrameProvider.requestAnimationFrame; 104 | global.window.cancelAnimationFrame = animationFrameProvider.cancelAnimationFrame; 105 | const context = canvas.getContext('2d'); 106 | (global as any).Image = this._image; // Some plugins use this API 107 | const chart = new this._chartJs((context as any), configuredChartConfig); 108 | delete (global as any).Image; 109 | return chart; 110 | } 111 | } 112 | -------------------------------------------------------------------------------- /API.md: -------------------------------------------------------------------------------- 1 | 2 | 3 | ## ChartJSNodeCanvas 4 | **Kind**: global class 5 | 6 | * [ChartJSNodeCanvas](#ChartJSNodeCanvas) 7 | * [new ChartJSNodeCanvas(options)](#new_ChartJSNodeCanvas_new) 8 | * [.renderToDataURL(configuration, mimeType)](#ChartJSNodeCanvas+renderToDataURL) 9 | * [.renderToDataURLSync(configuration, mimeType)](#ChartJSNodeCanvas+renderToDataURLSync) 10 | * [.renderToBuffer(configuration, mimeType)](#ChartJSNodeCanvas+renderToBuffer) 11 | * [.renderToBufferSync(configuration, mimeType)](#ChartJSNodeCanvas+renderToBufferSync) 12 | * [.renderToStream(configuration, mimeType)](#ChartJSNodeCanvas+renderToStream) 13 | * [.registerFont(path, options)](#ChartJSNodeCanvas+registerFont) 14 | 15 | 16 | 17 | ### new ChartJSNodeCanvas(options) 18 | Create a new instance of CanvasRenderService. 19 | 20 | 21 | | Param | Description | 22 | | --- | --- | 23 | | options | Configuration for this instance | 24 | 25 | 26 | 27 | ### chartJSNodeCanvas.renderToDataURL(configuration, mimeType) 28 | Render to a data url. 29 | 30 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 31 | **See**: https://github.com/Automattic/node-canvas#canvastodataurl 32 | 33 | | Param | Default | Description | 34 | | --- | --- | --- | 35 | | configuration | | The Chart JS configuration for the chart to render. | 36 | | mimeType | image/png | The image format, `image/png` or `image/jpeg`. | 37 | 38 | 39 | 40 | ### chartJSNodeCanvas.renderToDataURLSync(configuration, mimeType) 41 | Render to a data url synchronously. 42 | 43 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 44 | **See**: https://github.com/Automattic/node-canvas#canvastodataurl 45 | 46 | | Param | Default | Description | 47 | | --- | --- | --- | 48 | | configuration | | The Chart JS configuration for the chart to render. | 49 | | mimeType | image/png | The image format, `image/png` or `image/jpeg`. | 50 | 51 | 52 | 53 | ### chartJSNodeCanvas.renderToBuffer(configuration, mimeType) 54 | Render to a buffer. 55 | 56 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 57 | **See**: https://github.com/Automattic/node-canvas#canvastobuffer 58 | 59 | | Param | Default | Description | 60 | | --- | --- | --- | 61 | | configuration | | The Chart JS configuration for the chart to render. | 62 | | mimeType | image/png | A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support) or `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. | 63 | 64 | 65 | 66 | ### chartJSNodeCanvas.renderToBufferSync(configuration, mimeType) 67 | Render to a buffer synchronously. 68 | 69 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 70 | **See**: https://github.com/Automattic/node-canvas#canvastobuffer 71 | 72 | | Param | Default | Description | 73 | | --- | --- | --- | 74 | | configuration | | The Chart JS configuration for the chart to render. | 75 | | mimeType | image/png | A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support), `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom), `application/pdf` (for PDF canvases) and image/svg+xml (for SVG canvases). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. | 76 | 77 | 78 | 79 | ### chartJSNodeCanvas.renderToStream(configuration, mimeType) 80 | Render to a stream. 81 | 82 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 83 | **See**: https://github.com/Automattic/node-canvas#canvascreatepngstream 84 | 85 | | Param | Default | Description | 86 | | --- | --- | --- | 87 | | configuration | | The Chart JS configuration for the chart to render. | 88 | | mimeType | image/png | A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support), `application/pdf` (for PDF canvases) and image/svg+xml (for SVG canvases). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. | 89 | 90 | 91 | 92 | ### chartJSNodeCanvas.registerFont(path, options) 93 | Use to register the font with Canvas to use a font file that is not installed as a system font, this must be done before the Canvas is created. 94 | 95 | **Kind**: instance method of [ChartJSNodeCanvas](#ChartJSNodeCanvas) 96 | 97 | | Param | Description | 98 | | --- | --- | 99 | | path | The path to the font file. | 100 | | options | The font options. | 101 | 102 | **Example** 103 | ```js 104 | registerFont('comicsans.ttf', { family: 'Comic Sans' }); 105 | ``` 106 | -------------------------------------------------------------------------------- /resources/nodejs-logo.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /src/chartJSNodeCanvas.ts: -------------------------------------------------------------------------------- 1 | import { Readable } from 'stream'; 2 | // @ts-expect-error moduleResolution:nodenext issue 54523 3 | import { Chart as ChartJS, ChartConfiguration, ChartComponentLike } from 'chart.js/auto'; 4 | import { ChartJSNodeCanvasBase, MimeType, Canvas } from './chartJSNodeCanvasBase'; 5 | 6 | export class ChartJSNodeCanvas extends ChartJSNodeCanvasBase { 7 | 8 | /** 9 | * Render to a data url. 10 | * @see https://github.com/Automattic/node-canvas#canvastodataurl 11 | * 12 | * @param configuration The Chart JS configuration for the chart to render. 13 | * @param mimeType The image format, `image/png` or `image/jpeg`. 14 | */ 15 | public renderToDataURL(configuration: ChartConfiguration, mimeType: MimeType = 'image/png'): Promise { 16 | 17 | const chart = this.renderChart(configuration); 18 | return new Promise((resolve, reject) => { 19 | if (!chart.canvas) { 20 | return reject(new Error('Canvas is null')); 21 | } 22 | const canvas = chart.canvas as Canvas; 23 | canvas.toDataURL(mimeType, (error: Error | null, png: string) => { 24 | chart.destroy(); 25 | if (error) { 26 | return reject(error); 27 | } 28 | return resolve(png); 29 | }); 30 | }); 31 | } 32 | 33 | /** 34 | * Render to a data url synchronously. 35 | * @see https://github.com/Automattic/node-canvas#canvastodataurl 36 | * 37 | * @param configuration The Chart JS configuration for the chart to render. 38 | * @param mimeType The image format, `image/png` or `image/jpeg`. 39 | */ 40 | public renderToDataURLSync(configuration: ChartConfiguration, mimeType: MimeType = 'image/png'): string { 41 | 42 | const chart = this.renderChart(configuration); 43 | if (!chart.canvas) { 44 | throw new Error('Canvas is null'); 45 | } 46 | const canvas = chart.canvas as Canvas; 47 | const dataUrl = canvas.toDataURL(mimeType); 48 | // Use this to destroy any chart instances that are created. 49 | // This will clean up any references stored to the chart object within Chart.js, along with any associated event listeners attached by Chart.js. 50 | // This must be called before the canvas is reused for a new chart. 51 | chart.destroy(); 52 | return dataUrl; 53 | } 54 | 55 | /** 56 | * Render to a buffer. 57 | * @see https://github.com/Automattic/node-canvas#canvastobuffer 58 | * 59 | * @param configuration The Chart JS configuration for the chart to render. 60 | * @param mimeType A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support) or `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. 61 | */ 62 | public renderToBuffer(configuration: ChartConfiguration, mimeType: MimeType = 'image/png'): Promise { 63 | 64 | const chart = this.renderChart(configuration); 65 | return new Promise((resolve, reject) => { 66 | if (!chart.canvas) { 67 | throw new Error('Canvas is null'); 68 | } 69 | const canvas = chart.canvas as Canvas; 70 | canvas.toBuffer((error: Error | null, buffer: Buffer) => { 71 | chart.destroy(); 72 | if (error) { 73 | return reject(error); 74 | } 75 | return resolve(buffer); 76 | }, mimeType); 77 | }); 78 | } 79 | 80 | /** 81 | * Render to a buffer synchronously. 82 | * @see https://github.com/Automattic/node-canvas#canvastobuffer 83 | * 84 | * @param configuration The Chart JS configuration for the chart to render. 85 | * @param mimeType A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support), `raw` (unencoded ARGB32 data in native-endian byte order, top-to-bottom), `application/pdf` (for PDF canvases) and image/svg+xml (for SVG canvases). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. 86 | */ 87 | public renderToBufferSync(configuration: ChartConfiguration, mimeType: MimeType | 'application/pdf' | 'image/svg+xml' = 'image/png'): Buffer { 88 | 89 | const chart = this.renderChart(configuration); 90 | if (!chart.canvas) { 91 | throw new Error('Canvas is null'); 92 | } 93 | const canvas = chart.canvas as Canvas; 94 | const buffer = canvas.toBuffer(mimeType); 95 | chart.destroy(); 96 | return buffer; 97 | } 98 | 99 | /** 100 | * Render to a stream. 101 | * @see https://github.com/Automattic/node-canvas#canvascreatepngstream 102 | * 103 | * @param configuration The Chart JS configuration for the chart to render. 104 | * @param mimeType A string indicating the image format. Valid options are `image/png`, `image/jpeg` (if node-canvas was built with JPEG support), `application/pdf` (for PDF canvases) and image/svg+xml (for SVG canvases). Defaults to `image/png` for image canvases, or the corresponding type for PDF or SVG canvas. 105 | */ 106 | public renderToStream(configuration: ChartConfiguration, mimeType: MimeType | 'application/pdf' = 'image/png'): Readable { 107 | 108 | const chart = this.renderChart(configuration); 109 | if (!chart.canvas) { 110 | throw new Error('Canvas is null'); 111 | } 112 | const canvas = chart.canvas as Canvas; 113 | setImmediate(() => chart.destroy()); 114 | switch (mimeType) { 115 | case 'image/png': 116 | return canvas.createPNGStream(); 117 | case 'image/jpeg': 118 | return canvas.createJPEGStream(); 119 | case 'application/pdf': 120 | return canvas.createPDFStream(); 121 | default: 122 | throw new Error(`Un-handled mimeType: ${mimeType}`); 123 | } 124 | } 125 | 126 | private renderChart(configuration: ChartConfiguration): ChartJS { 127 | 128 | const canvas = this._createCanvas(this._width, this._height, this._type); 129 | (canvas as any).style = (canvas as any).style || {}; 130 | configuration.options = configuration.options || {}; 131 | configuration.options.responsive = false; 132 | // Disable animation (otherwise charts will throw exceptions) 133 | configuration.options.animation = false; 134 | const context = canvas.getContext('2d'); 135 | (global as any).Image = this._image; // Some plugins use this API 136 | const chart = new this._chartJs((context as any), configuration); 137 | delete (global as any).Image; 138 | return chart; 139 | } 140 | } 141 | -------------------------------------------------------------------------------- /src/chartJSNodeCanvasBase.ts: -------------------------------------------------------------------------------- 1 | import { Readable } from 'stream'; 2 | // @ts-expect-error moduleResolution:nodenext issue 54523 3 | import { Chart as ChartJS, ChartComponentLike } from 'chart.js/auto'; 4 | import { createCanvas, registerFont, Image } from 'canvas'; 5 | import { join as pathJoin } from 'path'; 6 | import { freshRequire } from './freshRequire'; 7 | import { BackgroundColourPlugin } from './backgroundColourPlugin'; 8 | 9 | export type ChartJSNodeCanvasPlugins = { 10 | /** 11 | * Global plugins, see https://www.chartjs.org/docs/latest/developers/plugins.html. 12 | */ 13 | readonly modern?: ReadonlyArray; 14 | /** 15 | * This will work for plugins that `require` ChartJS themselves. 16 | */ 17 | readonly requireChartJSLegacy?: ReadonlyArray; 18 | /** 19 | * This should work for any plugin that expects a global Chart variable. 20 | */ 21 | readonly globalVariableLegacy?: ReadonlyArray; 22 | /** 23 | * This will work with plugins that just return a plugin object and do no specific loading themselves. 24 | */ 25 | readonly requireLegacy?: ReadonlyArray; 26 | }; 27 | export type ChartCallback = (chartJS: typeof ChartJS) => void | Promise; 28 | export type CanvasType = 'pdf' | 'svg'; 29 | export type MimeType = 'image/png' | 'image/jpeg'; 30 | 31 | // https://github.com/Automattic/node-canvas#non-standard-apis 32 | export type Canvas = HTMLCanvasElement & { 33 | toBuffer(callback: (err: Error | null, result: Buffer) => void, mimeType?: string, config?: any): void; 34 | toBuffer(mimeType?: string, config?: any): Buffer; 35 | createPNGStream(config?: any): Readable; 36 | createJPEGStream(config?: any): Readable; 37 | createPDFStream(config?: any): Readable; 38 | }; 39 | 40 | export interface ChartJSNodeCanvasOptions { 41 | /** 42 | * The width of the charts to render, in pixels. 43 | */ 44 | readonly width: number; 45 | /** 46 | * The height of the charts to render, in pixels. 47 | */ 48 | readonly height: number; 49 | /** 50 | * Optional callback which is called once with a new ChartJS global reference as the only parameter. 51 | */ 52 | readonly chartCallback?: ChartCallback; 53 | /** 54 | * Optional canvas type ('PDF' or 'SVG'), see the [canvas pdf doc](https://github.com/Automattic/node-canvas#pdf-output-support). 55 | */ 56 | readonly type?: CanvasType; 57 | /** 58 | * Optional plugins to register. 59 | */ 60 | readonly plugins?: ChartJSNodeCanvasPlugins; 61 | /** 62 | * Optional background color for the chart, otherwise it will be transparent. Note, this will apply to all charts. See the [fillStyle](https://www.w3schools.com/tags/canvas_fillstyle.asp) canvas API used for possible values. 63 | */ 64 | readonly backgroundColour?: string; 65 | } 66 | 67 | export abstract class ChartJSNodeCanvasBase { 68 | 69 | protected readonly _width: number; 70 | protected readonly _height: number; 71 | protected readonly _chartJs: typeof ChartJS; 72 | protected readonly _createCanvas: typeof createCanvas; 73 | protected readonly _registerFont: typeof registerFont; 74 | protected readonly _image: typeof Image; 75 | protected readonly _type?: CanvasType; 76 | 77 | /** 78 | * Create a new instance of CanvasRenderService. 79 | * 80 | * @param options Configuration for this instance 81 | */ 82 | constructor(options: ChartJSNodeCanvasOptions) { 83 | 84 | if (options === null || typeof (options) !== 'object') { 85 | throw new Error('An options parameter object is required'); 86 | } 87 | if (!options.width || typeof (options.width) !== 'number') { 88 | throw new Error('A width option is required'); 89 | } 90 | if (!options.height || typeof (options.height) !== 'number') { 91 | throw new Error('A height option is required'); 92 | } 93 | 94 | this._width = options.width; 95 | this._height = options.height; 96 | const canvas = freshRequire('canvas'); 97 | this._createCanvas = canvas.createCanvas; 98 | this._registerFont = canvas.registerFont; 99 | this._image = canvas.Image; 100 | this._type = options.type && options.type.toLowerCase() as CanvasType; 101 | this._chartJs = this.initialize(options); 102 | } 103 | 104 | /** 105 | * Use to register the font with Canvas to use a font file that is not installed as a system font, this must be done before the Canvas is created. 106 | * 107 | * @param path The path to the font file. 108 | * @param options The font options. 109 | * @example 110 | * registerFont('comicsans.ttf', { family: 'Comic Sans' }); 111 | */ 112 | public registerFont(path: string, options: { readonly family: string, readonly weight?: string, readonly style?: string }): void { 113 | 114 | this._registerFont(path, options); 115 | } 116 | 117 | protected initialize(options: ChartJSNodeCanvasOptions): typeof ChartJS { 118 | 119 | const chartJs: typeof ChartJS = require('chart.js/auto'); 120 | 121 | if (options.plugins?.requireChartJSLegacy) { 122 | for (const plugin of options.plugins.requireChartJSLegacy) { 123 | require(plugin); 124 | delete require.cache[require.resolve(plugin)]; 125 | } 126 | } 127 | 128 | if (options.plugins?.globalVariableLegacy) { 129 | (global as any).Chart = chartJs; 130 | for (const plugin of options.plugins.globalVariableLegacy) { 131 | freshRequire(plugin); 132 | } 133 | delete (global as any).Chart; 134 | } 135 | 136 | if (options.plugins?.modern) { 137 | for (const plugin of options.plugins.modern) { 138 | if (typeof plugin === 'string') { 139 | chartJs.register(freshRequire(plugin)); 140 | } else { 141 | chartJs.register(plugin); 142 | } 143 | } 144 | } 145 | 146 | if (options.plugins?.requireLegacy) { 147 | for (const plugin of options.plugins.requireLegacy) { 148 | chartJs.register(freshRequire(plugin)); 149 | } 150 | } 151 | 152 | if (options.chartCallback) { 153 | options.chartCallback(chartJs); 154 | } 155 | 156 | if (options.backgroundColour) { 157 | chartJs.register(new BackgroundColourPlugin(options.width, options.height, options.backgroundColour)); 158 | } 159 | const chartJsPath = pathJoin('node_modules','chart.js'); 160 | 161 | for (const key of Object.keys(require.cache)) { 162 | if (key.includes(chartJsPath)) { 163 | delete require.cache[key]; 164 | } 165 | } 166 | 167 | return chartJs; 168 | } 169 | } 170 | -------------------------------------------------------------------------------- /src/index.spec.ts: -------------------------------------------------------------------------------- 1 | import { Assert } from 'ts-std-lib'; 2 | import { describe, it } from 'mocha'; 3 | // @ts-expect-error moduleResolution:nodenext issue 54523 4 | import { ChartConfiguration } from 'chart.js/auto'; 5 | 6 | import { ChartJSNodeCanvas, ChartCallback, CanvasType, MimeType, ChartJSNodeCanvasPlugins } from './'; 7 | 8 | const assert = new Assert(); 9 | 10 | describe(ChartJSNodeCanvas.name, () => { 11 | 12 | // const chartColors = { 13 | // red: 'rgb(255, 99, 132)', 14 | // orange: 'rgb(255, 159, 64)', 15 | // yellow: 'rgb(255, 205, 86)', 16 | // green: 'rgb(75, 192, 192)', 17 | // blue: 'rgb(54, 162, 235)', 18 | // purple: 'rgb(153, 102, 255)', 19 | // grey: 'rgb(201, 203, 207)' 20 | // }; 21 | const width = 400; 22 | const height = 400; 23 | const configuration: ChartConfiguration = { 24 | type: 'bar', 25 | data: { 26 | labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], 27 | datasets: [{ 28 | label: '# of Votes', 29 | data: [12, 19, 3, 5, 2, 3], 30 | backgroundColor: [ 31 | 'rgba(255, 99, 132, 0.2)', 32 | 'rgba(54, 162, 235, 0.2)', 33 | 'rgba(255, 206, 86, 0.2)', 34 | 'rgba(75, 192, 192, 0.2)', 35 | 'rgba(153, 102, 255, 0.2)', 36 | 'rgba(255, 159, 64, 0.2)' 37 | ], 38 | borderColor: [ 39 | 'rgba(255,99,132,1)', 40 | 'rgba(54, 162, 235, 1)', 41 | 'rgba(255, 206, 86, 1)', 42 | 'rgba(75, 192, 192, 1)', 43 | 'rgba(153, 102, 255, 1)', 44 | 'rgba(255, 159, 64, 1)' 45 | ], 46 | borderWidth: 1 47 | }] 48 | }, 49 | options: { 50 | scales: { 51 | y: { 52 | ticks: { 53 | beginAtZero: true, 54 | callback: (value: number) => '$' + value 55 | } as any 56 | } 57 | } 58 | }, 59 | plugins: { 60 | annotation: { 61 | } 62 | } as any 63 | }; 64 | 65 | function createSUT(type?: CanvasType, plugins?: ChartJSNodeCanvasPlugins): ChartJSNodeCanvas { 66 | 67 | const chartCallback: ChartCallback = (ChartJS) => { 68 | 69 | ChartJS.defaults.responsive = true; 70 | ChartJS.defaults.maintainAspectRatio = false; 71 | }; 72 | return new ChartJSNodeCanvas({ width, height, chartCallback, type, plugins }); 73 | } 74 | 75 | const mimeTypes: ReadonlyArray = ['image/png', 'image/jpeg']; 76 | 77 | describe(ChartJSNodeCanvas.prototype.renderToDataURL.name, () => { 78 | 79 | describe(`given canvasType 'undefined'`, () => { 80 | 81 | const canvasType = undefined; 82 | 83 | mimeTypes.forEach((mimeType) => { 84 | 85 | describe(`given mimeType '${mimeType}'`, () => { 86 | 87 | it('renders data url', async () => { 88 | const chartJSNodeCanvas = createSUT(canvasType); 89 | const dataUrl = await chartJSNodeCanvas.renderToDataURL(configuration, mimeType); 90 | assert.equal(dataUrl.startsWith(`data:${mimeType};base64,`), true); 91 | }); 92 | 93 | it('renders data url in parallel', async () => { 94 | const chartJSNodeCanvas = createSUT(canvasType); 95 | const promises = Array(3).fill(undefined).map(() => chartJSNodeCanvas.renderToDataURL(configuration, mimeType)); 96 | const dataUrls = await Promise.all(promises); 97 | dataUrls.forEach((dataUrl) => assert.equal(dataUrl.startsWith(`data:${mimeType};base64,`), true)); 98 | }); 99 | }); 100 | }); 101 | }); 102 | }); 103 | 104 | describe(ChartJSNodeCanvas.prototype.renderToDataURLSync.name, () => { 105 | 106 | describe(`given canvasType 'undefined'`, () => { 107 | 108 | const canvasType = undefined; 109 | 110 | mimeTypes.forEach((mimeType) => { 111 | 112 | describe(`given mimeType '${mimeType}'`, () => { 113 | 114 | it('renders data url', () => { 115 | const chartJSNodeCanvas = createSUT(canvasType); 116 | const dataUrl = chartJSNodeCanvas.renderToDataURLSync(configuration, mimeType); 117 | assert.equal(dataUrl.startsWith(`data:${mimeType};base64,`), true); 118 | }); 119 | 120 | it('renders data url in parallel', () => { 121 | const chartJSNodeCanvas = createSUT(canvasType); 122 | const dataUrls = Array(3).fill(undefined).map(() => chartJSNodeCanvas.renderToDataURLSync(configuration, mimeType)); 123 | dataUrls.forEach((dataUrl) => assert.equal(dataUrl.startsWith(`data:${mimeType};base64,`), true)); 124 | }); 125 | }); 126 | }); 127 | }); 128 | }); 129 | 130 | describe(ChartJSNodeCanvas.prototype.renderToBuffer.name, () => { 131 | 132 | describe(`given canvasType 'undefined'`, () => { 133 | 134 | const canvasType = undefined; 135 | 136 | mimeTypes.forEach((mimeType) => { 137 | 138 | describe(`given extended mimeType '${mimeType}'`, () => { 139 | 140 | it('renders chart', async () => { 141 | const chartJSNodeCanvas = createSUT(canvasType); 142 | const image = await chartJSNodeCanvas.renderToBuffer(configuration, mimeType); 143 | assert.equal(image instanceof Buffer, true); 144 | }); 145 | }); 146 | }); 147 | }); 148 | }); 149 | 150 | describe(ChartJSNodeCanvas.prototype.renderToBufferSync.name, () => { 151 | 152 | ([ 153 | [undefined, mimeTypes], 154 | ['svg', ['image/svg+xml']], 155 | ['pdf', ['application/pdf']] 156 | ] as ReadonlyArray<[CanvasType, ReadonlyArray]>).forEach(([canvasType, extendedMimeTypes]) => { 157 | 158 | describe(`given canvasType '${canvasType}'`, () => { 159 | 160 | extendedMimeTypes.forEach((mimeType) => { 161 | 162 | describe(`given mimeType '${mimeType}'`, () => { 163 | 164 | it('renders chart', async () => { 165 | const chartJSNodeCanvas = createSUT(canvasType); 166 | const image = chartJSNodeCanvas.renderToBufferSync(configuration, mimeType); 167 | assert.equal(image instanceof Buffer, true); 168 | }); 169 | }); 170 | }); 171 | }); 172 | }); 173 | }); 174 | 175 | describe(ChartJSNodeCanvas.prototype.renderToStream.name, () => { 176 | 177 | ([ 178 | [undefined, mimeTypes], 179 | ['pdf', ['application/pdf']] 180 | ] as ReadonlyArray<[CanvasType | undefined, ReadonlyArray]>).forEach(([canvasType, extendedMimeTypes]) => { 181 | 182 | describe(`given canvasType '${canvasType}'`, () => { 183 | 184 | extendedMimeTypes.forEach((mimeType) => { 185 | 186 | describe(`given extended mimeType '${mimeType}'`, () => { 187 | 188 | it('renders stream', (done) => { 189 | const chartJSNodeCanvas = createSUT(canvasType); 190 | const stream = chartJSNodeCanvas.renderToStream(configuration, mimeType); 191 | const data: Array = []; 192 | stream.on('data', (chunk: Buffer) => { 193 | data.push(chunk); 194 | }); 195 | stream.on('end', () => { 196 | assert.equal(Buffer.concat(data).length > 0, true); 197 | done(); 198 | }); 199 | stream.on('finish', () => { 200 | assert.equal(Buffer.concat(data).length > 0, true); 201 | done(); 202 | }); 203 | stream.on('error', (error) => { 204 | done(error); 205 | }); 206 | }); 207 | }); 208 | }); 209 | }); 210 | }); 211 | }); 212 | }); 213 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | # chartjs-node-canvas 7 | 8 | 9 | [![GitHub](https://github.com/SeanSobey/ChartjsNodeCanvas/workflows/Node%20CI/badge.svg)](https://github.com/SeanSobey/ChartjsNodeCanvas/actions) 10 | [![codecov](https://codecov.io/gh/SeanSobey/ChartjsNodeCanvas/branch/master/graph/badge.svg)](https://codecov.io/gh/SeanSobey/ChartjsNodeCanvas) 11 | [![NPM](https://img.shields.io/npm/v/chartjs-node-canvas.svg)](https://www.npmjs.com/package/chartjs-node-canvas) 12 | [![packagephobia publish](https://badgen.net/packagephobia/publish/chartjs-node-canvas@latest)](https://bundlephobia.com/result?p=chartjs-node-canvas) 13 | 14 | 15 | 16 | A Node JS renderer for [Chart.js](http://www.chartjs.org) using [canvas](https://github.com/Automattic/node-canvas). 17 | 18 | Provides and alternative to [chartjs-node](https://www.npmjs.com/package/chartjs-node) that does not require jsdom (or the global variables that this requires) and allows chartJS as a peer dependency, so you can manage its version yourself. 19 | 20 | ## Contents 21 | 22 | 1. [Installation](#installation) 23 | 2. [Node JS version](#node-js-version) 24 | 3. [Features](#features) 25 | 4. [Limitations](#limitations) 26 | 5. [API](#api) 27 | 6. [Usage](#usage) 28 | 7. [Known Issues](#known-issues) 29 | 8. [Sponsors](#sponsors) 30 | 31 | ## Installation 32 | 33 | ``` 34 | npm i chartjs-node-canvas chart.js 35 | ``` 36 | 37 | ### Node JS version 38 | 39 | This is limited by the upstream dependency [canvas](https://github.com/Automattic/node-canvas). 40 | 41 | See the GitHub Actions [yml](.github/workflows/nodejs.yml) section for the current supported Node version(s). You will need to do a `npm rebuild` to rebuild the canvas binaries. 42 | 43 | ### Charts.JS version 44 | 45 | Currently supports 4.x.x. You are given the ability to maintain the version yourself via peer dependency, but be aware that going above the specified [version](./package.json) might result in errors. 46 | 47 | ## Features 48 | 49 | * Supports all Chart JS features and charts. 50 | * No heavy DOM virtualization libraries, thanks to a [pull request](https://github.com/chartjs/Chart.js/pull/5324) to chart.js allowing it to run natively on node, requiring only a Canvas API. 51 | * Chart JS is a peer dependency, so you can bump and manage it yourself. 52 | * Provides a callback with the global ChartJS variable, so you can use the [Global Configuration](https://www.chartjs.org/docs/latest/configuration/#global-configuration). 53 | * Uses (similar to) [fresh-require](https://www.npmjs.com/package/fresh-require) for each instance of `ChartJSNodeCanvas`, so you can mutate the ChartJS global variable separately within each instance. 54 | * Support for custom fonts. 55 | 56 | ## Limitations 57 | 58 | ### Node Modules 59 | 60 | I hope to convert this package to a node module in the future, but since it uses the CommonJS API to manage memory for ChartJS this is not a simple task. It it a top priority for the next major release. 61 | 62 | ### Animations 63 | 64 | Chart animation (and responsive resize) is disabled by this library. This is necessary since the animation API's required are not available in Node JS/canvas-node (this is not a browser environment after all). 65 | 66 | This is the same as: 67 | 68 | ```js 69 | Chart.defaults.animation = false; 70 | Chart.defaults.responsive = false; 71 | ``` 72 | 73 | *Note this is WIP, see the [change log](CHANGELOG.md) for most recent development. 74 | 75 | ### SVG and PDF 76 | 77 | For some unknown reason canvas requires use of the [sync](https://github.com/Automattic/node-canvas#canvastobuffer) API's to use SVG's or PDF's. This libraries which support these are: 78 | 79 | * [renderToBufferSync](./API.md#ChartJSNodeCanvas+renderToBufferSync) ('application/pdf' | 'image/svg+xml') 80 | * [renderToStream](./API.md#ChartJSNodeCanvas+renderToStream) ('application/pdf') 81 | 82 | You also need to set the canvas type when you initialize the `ChartJSNodeCanvas` instance like the following: 83 | 84 | ```js 85 | const { ChartJSNodeCanvas } = require('chartjs-node-canvas'); 86 | 87 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ type: 'svg', width: 800, height: 600 }); 88 | ``` 89 | 90 | ## API 91 | 92 | See the [API docs](https://github.com/SeanSobey/ChartjsNodeCanvas/blob/master/API.md). 93 | 94 | ## Usage 95 | 96 | ```js 97 | const { ChartJSNodeCanvas } = require('chartjs-node-canvas'); 98 | 99 | const width = 400; //px 100 | const height = 400; //px 101 | const backgroundColour = 'white'; // Uses https://www.w3schools.com/tags/canvas_fillstyle.asp 102 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, backgroundColour}); 103 | 104 | (async () => { 105 | const configuration = { 106 | ... // See https://www.chartjs.org/docs/latest/configuration 107 | }; 108 | const image = await chartJSNodeCanvas.renderToBuffer(configuration); 109 | const dataUrl = await chartJSNodeCanvas.renderToDataURL(configuration); 110 | const stream = chartJSNodeCanvas.renderToStream(configuration); 111 | })(); 112 | ``` 113 | 114 | Also see the [example](./src/example.ts) and the generated [image](./example.png). 115 | 116 | ### Memory Management 117 | 118 | Every instance of `ChartJSNodeCanvas` creates its own [canvas](https://github.com/Automattic/node-canvas). To ensure efficient memory and GC use make sure your implementation creates as few instances as possible and reuses them: 119 | 120 | ```js 121 | // Re-use one service, or as many as you need for different canvas size requirements 122 | const smallChartJSNodeCanvas = new ChartJSNodeCanvas({ width: 400, height: 400 }); 123 | const bigCChartJSNodeCanvas = new ChartJSNodeCanvas({ width: 2000, height: 2000 }); 124 | 125 | // Expose just the 'render' methods to downstream code so they don't have to worry about life-cycle management. 126 | exports = { 127 | renderSmallChart: (configuration) => smallChartJSNodeCanvas.renderToBuffer(configuration), 128 | renderBigChart: (configuration) => bigCChartJSNodeCanvas.renderToBuffer(configuration) 129 | }; 130 | ``` 131 | 132 | ### Custom Charts 133 | 134 | Just use the ChartJS reference in the callback: 135 | 136 | ```js 137 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback: (ChartJS) => { 138 | // New chart type example: https://www.chartjs.org/docs/latest/developers/charts.html 139 | class MyType extends Chart.DatasetController { 140 | 141 | } 142 | 143 | Chart.register(MyType); 144 | } 145 | }); 146 | ``` 147 | 148 | ### Global Config 149 | 150 | Just use the ChartJS reference in the callback: 151 | 152 | ```js 153 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback: (ChartJS) => { 154 | // Global config example: https://www.chartjs.org/docs/latest/configuration/ 155 | ChartJS.defaults.elements.line.borderWidth = 2; 156 | } }); 157 | ``` 158 | 159 | ### Custom Fonts 160 | 161 | Just use the `registerFont` method: 162 | 163 | ```js 164 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback: (ChartJS) => { 165 | // Just example usage 166 | ChartJS.global.defaultFontFamily = 'VTKS UNAMOUR'; 167 | } }); 168 | // Register before rendering any charts 169 | chartJSNodeCanvas.registerFont('./testData/VTKS UNAMOUR.ttf', { family: 'VTKS UNAMOUR' }); 170 | ``` 171 | 172 | See the node-canvas [docs](https://github.com/Automattic/node-canvas#registerfont) and the chart js [docs](https://www.chartjs.org/docs/latest/general/fonts.html). 173 | 174 | #### Windows 175 | 176 | On windows you need to install the font first, before running your app. Otherwise you will get an error something like: 177 | `Pango-WARNING **: 11:13:09.211: couldn't load font "vtks unamour Not-Rotated 12px", falling back to "Sans Not-Rotated 12px", expect ugly output.` 178 | 179 | See [here](https://github.com/Automattic/node-canvas/issues/1643). 180 | 181 | ### Background color 182 | 183 | Due to the many issues and question this includes a [convenience plugin](./src/backgroundColourPlugin.ts) to fill the otherwise transparent background. It uses the [fillStyle](https://www.w3schools.com/tags/canvas_fillstyle.asp) canvas API; 184 | 185 | ```js 186 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, backgroundColour: 'purple' }); 187 | ``` 188 | 189 | ### Loading plugins 190 | 191 | This library is designed to make loading plugins as simple as possible. For legacy plugins, you should just be able to add the module name to the appropriate array option and the library handles the rest. 192 | 193 | The Chart.JS [plugin API](https://www.chartjs.org/docs/latest/developers/plugins.html) has changed over time and this requires compatibility options for the different ways plugins have been historically loaded. ChartJS Node Canvas has a `plugin` option with specifiers for the different ways supported plugin loading methods are handled. If you are not sure about your plugin, just try the different ones until your plugin loads: 194 | 195 | #### Newer plugins 196 | 197 | Let `ChartJSNodeCanvas` manage the lifecycle of the plugin itself, each instance will have a separate instance of the plugin: 198 | 199 | ```js 200 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 201 | modern: ['chartjs-plugin-annotation'] 202 | } }); 203 | ``` 204 | 205 | You want to share the plugin instance, this may cause unwanted issues, use at own risk: 206 | 207 | ```js 208 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 209 | modern: [require('chartjs-plugin-annotation')] 210 | } }); 211 | ``` 212 | 213 | #### Older plugins 214 | 215 | --- 216 | 217 | 1. Plugin that expects a global Chart variable. 218 | 219 | ```js 220 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 221 | requireChartJSLegacy: [''] 222 | }}); 223 | ``` 224 | 225 | 2. Plugins that `require` ChartJS themselves. 226 | 227 | ```js 228 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 229 | globalVariableLegacy: ['chartjs-plugin-crosshair'] 230 | } }); 231 | ``` 232 | 233 | 3. Register plugin directly with ChartJS: 234 | 235 | ```js 236 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 237 | requireLegacy: ['chartjs-plugin-datalabels'] 238 | } }); 239 | ``` 240 | 241 | --- 242 | 243 | These approaches can be combined also: 244 | 245 | ```js 246 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, plugins: { 247 | modern: ['chartjs-plugin-annotation'], 248 | requireLegacy: ['chartjs-plugin-datalabels'] 249 | } }); 250 | ``` 251 | 252 | See the [tests](src/index.e2e.spec.ts#106) for some examples. 253 | 254 | ## Known Issues 255 | 256 | There is a problem with persisting config objects between render calls, see this [issue](https://github.com/SeanSobey/ChartjsNodeCanvas/issues/9) for details and workarounds. 257 | 258 | ## Sponsors 259 | 260 | @athombv at https://homey.app 261 | -------------------------------------------------------------------------------- /testData/win32/render-to-data-URL.txt: -------------------------------------------------------------------------------- 1 |  -------------------------------------------------------------------------------- /src/index.e2e.spec.ts: -------------------------------------------------------------------------------- 1 | import { AssertionError } from 'assert'; 2 | import { promises as fs } from 'fs'; 3 | import { platform, EOL } from 'os'; 4 | import { join } from 'path'; 5 | import { Readable } from 'stream'; 6 | import { describe, it } from 'mocha'; 7 | import { Stream } from 'stream'; 8 | // @ts-expect-error moduleResolution:nodenext issue 54523 9 | import { ChartConfiguration } from 'chart.js/auto'; 10 | import resemble /*, { ResembleSingleCallbackComparisonOptions, ResembleSingleCallbackComparisonResult }*/ from 'resemblejs'; 11 | 12 | import { ChartJSNodeCanvas, ChartCallback } from './'; 13 | 14 | describe(ChartJSNodeCanvas.name, () => { 15 | 16 | const chartColors = { 17 | red: 'rgb(255, 99, 132)', 18 | orange: 'rgb(255, 159, 64)', 19 | yellow: 'rgb(255, 205, 86)', 20 | green: 'rgb(75, 192, 192)', 21 | blue: 'rgb(54, 162, 235)', 22 | purple: 'rgb(153, 102, 255)', 23 | grey: 'rgb(201, 203, 207)' 24 | }; 25 | const width = 400; 26 | const height = 400; 27 | const configuration: ChartConfiguration = { 28 | type: 'bar', 29 | data: { 30 | labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], 31 | datasets: [{ 32 | label: '# of Votes', 33 | data: [12, 19, 3, 5, 2, 3], 34 | backgroundColor: [ 35 | 'rgba(255, 99, 132, 0.2)', 36 | 'rgba(54, 162, 235, 0.2)', 37 | 'rgba(255, 206, 86, 0.2)', 38 | 'rgba(75, 192, 192, 0.2)', 39 | 'rgba(153, 102, 255, 0.2)', 40 | 'rgba(255, 159, 64, 0.2)' 41 | ], 42 | borderColor: [ 43 | 'rgba(255,99,132,1)', 44 | 'rgba(54, 162, 235, 1)', 45 | 'rgba(255, 206, 86, 1)', 46 | 'rgba(75, 192, 192, 1)', 47 | 'rgba(153, 102, 255, 1)', 48 | 'rgba(255, 159, 64, 1)' 49 | ], 50 | borderWidth: 1 51 | }] 52 | }, 53 | options: { 54 | scales: { 55 | y: { 56 | beginAtZero: true, 57 | ticks: { 58 | callback: (tickValue, _index, _ticks) => '$' + tickValue 59 | } 60 | } 61 | }, 62 | plugins: { 63 | annotation: { 64 | } 65 | } as any 66 | } 67 | }; 68 | const chartCallback: ChartCallback = (ChartJS) => { 69 | ChartJS.defaults.responsive = true; 70 | ChartJS.defaults.maintainAspectRatio = false; 71 | }; 72 | 73 | it('works with render to buffer', async () => { 74 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback, backgroundColour: 'white' }); 75 | const actual = await chartJSNodeCanvas.renderToBuffer(configuration); 76 | await assertImage(actual, 'render-to-buffer'); 77 | }); 78 | 79 | it('works with render to data url', async () => { 80 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback, backgroundColour: 'white' }); 81 | const actual = await chartJSNodeCanvas.renderToDataURL(configuration); 82 | const extension = '.txt'; 83 | const fileName = 'render-to-data-URL'; 84 | const fileNameWithExtension = fileName + extension; 85 | const expectedDataPath = join(process.cwd(), 'testData', platform(), fileName + extension); 86 | const expected = await fs.readFile(expectedDataPath, 'utf8'); 87 | // const result = actual === expected; 88 | const compareData = await compareImages(actual, expected, { output: { useCrossOrigin: false } }); 89 | const misMatchPercentage = Number(compareData.misMatchPercentage); 90 | const result = misMatchPercentage > 0; 91 | if (result) { 92 | const actualDataPath = expectedDataPath.replace(fileNameWithExtension, fileName + '-actual' + extension); 93 | await fs.writeFile(actualDataPath, actual); 94 | const compare = `
Actual:
${EOL}${EOL}
Expected:
`; 95 | const compareDataPath = expectedDataPath.replace(fileNameWithExtension, fileName + '-compare.html'); 96 | await fs.writeFile(compareDataPath, compare); 97 | const diffPng = compareData.getBuffer(); 98 | await writeDiff(expectedDataPath.replace(fileNameWithExtension, fileName + '-diff' + extension), diffPng); 99 | throw new AssertionError({ 100 | message: `Expected data urls to match, mismatch was ${misMatchPercentage}%${EOL}See '${compareDataPath}'`, 101 | actual: actualDataPath, 102 | expected: expectedDataPath, 103 | }); 104 | } 105 | }); 106 | 107 | it('works with render to stream', async () => { 108 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback, backgroundColour: 'white' }); 109 | const stream = chartJSNodeCanvas.renderToStream(configuration); 110 | const actual = await streamToBuffer(stream); 111 | await assertImage(actual, 'render-to-stream'); 112 | }); 113 | 114 | it('works with registering plugin', async () => { 115 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ 116 | width, height, backgroundColour: 'white', plugins: { 117 | modern: ['chartjs-plugin-annotation'] 118 | } 119 | }); 120 | const actual = await chartJSNodeCanvas.renderToBuffer({ 121 | type: 'bar', 122 | data: { 123 | labels: ['January', 'February', 'March', 'April', 'May', 'June', 'July'], 124 | datasets: [ 125 | { 126 | type: 'line', 127 | label: 'Dataset 1', 128 | borderColor: chartColors.blue, 129 | borderWidth: 2, 130 | fill: false, 131 | data: [-39, 44, -22, -45, -27, 12, 18] 132 | }, 133 | { 134 | type: 'bar', 135 | label: 'Dataset 2', 136 | backgroundColor: chartColors.red, 137 | data: [-18, -43, 36, -37, 1, -1, 26], 138 | borderColor: 'white', 139 | borderWidth: 2 140 | }, 141 | { 142 | type: 'bar', 143 | label: 'Dataset 3', 144 | backgroundColor: chartColors.green, 145 | data: [-7, 21, 1, 7, 34, -29, -36] 146 | } 147 | ] 148 | }, 149 | options: { 150 | responsive: true, 151 | plugins: { 152 | annotation: { 153 | annotations: { 154 | label1: { 155 | type: 'label', 156 | xValue: 3, 157 | yValue: 20, 158 | backgroundColor: 'rgba(245,245,245)', 159 | content: ['This is my text', 'This is my text, second line'], 160 | font: { 161 | size: 18 162 | } 163 | } 164 | } 165 | } 166 | } as any 167 | } 168 | }); 169 | await assertImage(actual, 'chartjs-plugin-annotation'); 170 | }); 171 | 172 | it('works with self registering plugin', async () => { 173 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ 174 | width, height, backgroundColour: 'white', plugins: { 175 | requireLegacy: [ 176 | 'chartjs-plugin-datalabels' 177 | ] 178 | } 179 | }); 180 | const actual = await chartJSNodeCanvas.renderToBuffer({ 181 | type: 'bar', 182 | data: { 183 | labels: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] as any, 184 | datasets: [{ 185 | label: 'Red', 186 | backgroundColor: chartColors.red, 187 | data: [12, 19, 3, 5, 2, 3], 188 | datalabels: { 189 | align: 'end', 190 | anchor: 'start' 191 | } 192 | }, { 193 | label: 'Blue', 194 | backgroundColor: chartColors.blue, 195 | data: [3, 5, 2, 3, 30, 15, 19, 2], 196 | datalabels: { 197 | align: 'center', 198 | anchor: 'center' 199 | } 200 | }, { 201 | label: 'Green', 202 | backgroundColor: chartColors.green, 203 | data: [12, 19, 3, 5, 2, 3], 204 | datalabels: { 205 | anchor: 'end', 206 | align: 'start', 207 | } 208 | }] as any 209 | }, 210 | options: { 211 | plugins: { 212 | datalabels: { 213 | color: 'white', 214 | display(context: any): boolean { 215 | return context.dataset.data[context.dataIndex] > 15; 216 | }, 217 | font: { 218 | weight: 'bold' 219 | }, 220 | formatter: Math.round 221 | } 222 | } as any, // TODO: resolve type 223 | scales: { 224 | x: { 225 | stacked: true 226 | }, 227 | y: { 228 | stacked: true 229 | } 230 | } 231 | } 232 | }); 233 | await assertImage(actual, 'chartjs-plugin-datalabels'); 234 | }); 235 | 236 | // it.skip('works with global variable plugin', async () => { 237 | // const chartJSNodeCanvas = new ChartJSNodeCanvas({ 238 | // width, height, backgroundColour: 'white', plugins: { 239 | // globalVariableLegacy: [ 240 | // 'chartjs-plugin-crosshair' 241 | // ] 242 | // } 243 | // }); 244 | // const actual = await chartJSNodeCanvas.renderToBuffer({ 245 | // }); 246 | // await assertImage(actual, 'chartjs-plugin-funnel'); 247 | // }); 248 | 249 | it('works with custom font', async () => { 250 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ 251 | width, height, backgroundColour: 'white', chartCallback: (ChartJS) => { 252 | ChartJS.defaults.font.family = 'Anthrope'; 253 | } 254 | }); 255 | chartJSNodeCanvas.registerFont('./testData/Anthrope.ttf', { family: 'Anthrope' }); 256 | const actual = await chartJSNodeCanvas.renderToBuffer({ 257 | type: 'bar', 258 | data: { 259 | labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], 260 | datasets: [{ 261 | label: '# of Votes', 262 | data: [12, 19, 3, 5, 2, 3], 263 | backgroundColor: [ 264 | 'rgba(255, 99, 132, 0.2)', 265 | 'rgba(54, 162, 235, 0.2)', 266 | 'rgba(255, 206, 86, 0.2)', 267 | 'rgba(75, 192, 192, 0.2)', 268 | 'rgba(153, 102, 255, 0.2)', 269 | 'rgba(255, 159, 64, 0.2)' 270 | ], 271 | borderColor: [ 272 | 'rgba(255,99,132,1)', 273 | 'rgba(54, 162, 235, 1)', 274 | 'rgba(255, 206, 86, 1)', 275 | 'rgba(75, 192, 192, 1)', 276 | 'rgba(153, 102, 255, 1)', 277 | 'rgba(255, 159, 64, 1)' 278 | ], 279 | borderWidth: 1, 280 | }] 281 | }, 282 | options: { 283 | scales: { 284 | y: { 285 | ticks: { 286 | beginAtZero: true, 287 | callback: (value: number) => '$' + value 288 | } as any 289 | } 290 | } 291 | }, 292 | plugins: { 293 | annotation: { 294 | } 295 | } as any 296 | }); 297 | await assertImage(actual, 'font'); 298 | }); 299 | 300 | it('works without background color', async () => { 301 | 302 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback }); 303 | const actual = await chartJSNodeCanvas.renderToBuffer(configuration); 304 | await assertImage(actual, 'no-background-color'); 305 | }); 306 | 307 | /* 308 | function hashCode(string: string): number { 309 | 310 | let hash = 0; 311 | if (string.length === 0) { 312 | return hash; 313 | } 314 | for (let i = 0; i < string.length; i++) { 315 | const chr = string.charCodeAt(i); 316 | hash = ((hash << 5) - hash) + chr; 317 | hash |= 0; // Convert to 32bit integer 318 | } 319 | return hash; 320 | } 321 | */ 322 | 323 | async function assertImage(actual: Buffer, fileName: string): Promise { 324 | const extension = '.png'; 325 | const fileNameWithExtension = fileName + extension; 326 | const testDataPath = join(process.cwd(), 'testData', platform(), fileNameWithExtension); 327 | const exists = await pathExists(testDataPath); 328 | if (!exists) { 329 | console.error(`Warning: expected image path does not exist!, creating '${testDataPath}'`); 330 | await fs.writeFile(testDataPath, actual, 'base64'); 331 | return; 332 | } 333 | const expected = await fs.readFile(testDataPath); 334 | const compareData = await compareImages(actual, expected); 335 | // const compareData = await new Promise((resolve) => { 336 | // const diff = resemble(actual) 337 | // .compareTo(expected) 338 | // .onComplete((data) => { 339 | // resolve(data); 340 | // }); 341 | // }); 342 | const misMatchPercentage = Number(compareData.misMatchPercentage); 343 | // const result = actual.equals(expected); 344 | const result = misMatchPercentage > 0; 345 | // const actual = hashCode(image.toString('base64')); 346 | // const expected = -1377895140; 347 | // assert.equal(actual, expected); 348 | if (result) { 349 | await fs.writeFile(testDataPath.replace(fileNameWithExtension, fileName + '-actual' + extension), actual); 350 | const diffPng = compareData.getBuffer(); 351 | await writeDiff(testDataPath.replace(fileNameWithExtension, fileName + '-diff' + extension), diffPng); 352 | throw new AssertionError({ 353 | message: `Expected image to match '${testDataPath}', mismatch was ${misMatchPercentage}%'`, 354 | // actual: JSON.stringify(actual), 355 | // expected: JSON.stringify(expected), 356 | operator: 'to equal', 357 | stackStartFn: assertImage, 358 | }); 359 | } 360 | } 361 | 362 | // resemblejs/compareImages 363 | //function compareImages(image1: string | Buffer, image2: string | Buffer, options?: ResembleSingleCallbackComparisonOptions): Promise { 364 | function compareImages(image1: string | Buffer, image2: string | Buffer, options?: any): Promise { 365 | return new Promise((resolve, reject) => { 366 | //resemble.compare(image1, image2, options || {}, (err, data) => { 367 | resemble.compare(image1, image2, options || {}, (err: any, data: any) => { 368 | if (err) { 369 | reject(err); 370 | } else { 371 | resolve(data); 372 | } 373 | }); 374 | }); 375 | } 376 | 377 | function streamToBuffer(stream: Readable): Promise { 378 | const data: Array = []; 379 | return new Promise((resolve, reject) => { 380 | stream.on('data', (chunk: Buffer) => { 381 | data.push(chunk); 382 | }); 383 | stream.on('end', () => { 384 | const buffer = Buffer.concat(data); 385 | resolve(buffer); 386 | }); 387 | stream.on('error', (error) => { 388 | reject(error); 389 | }); 390 | }); 391 | } 392 | 393 | function writeDiff(filepath: string, png: Stream | Buffer): Promise { 394 | return new Promise((resolve, reject) => { 395 | if (Buffer.isBuffer(png)) { 396 | fs.writeFile(filepath, png.toString('base64'), 'base64') 397 | .then(() => resolve()) 398 | .catch(reject); 399 | return; 400 | } 401 | const chunks: Array = []; 402 | png.on('data', (chunk: Uint8Array) => { 403 | chunks.push(chunk); 404 | }); 405 | png.on('end', () => { 406 | const buffer = Buffer.concat(chunks); 407 | fs.writeFile(filepath, buffer.toString('base64'), 'base64') 408 | .then(() => resolve()) 409 | .catch(reject); 410 | }); 411 | png.on('error', (err) => reject(err)); 412 | }); 413 | } 414 | 415 | function pathExists(path: string): Promise { 416 | return fs.access(path).then(() => true).catch(() => false); 417 | } 418 | 419 | describe('Memory tests', () => { 420 | 421 | const count = 20; 422 | // TODO: Replace node-memwatch with a new lib! 423 | 424 | it('does not leak with new instance parallel', async () => { 425 | 426 | const memoryUsages = new Array(count + 2); 427 | memoryUsages[0] = process.memoryUsage(); 428 | const diffs = await Promise.all([...Array(count).keys()].map((iteration) => { 429 | console.log('generated heap for iteration ' + (iteration + 1)); 430 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback }); 431 | return chartJSNodeCanvas.renderToBuffer(configuration, 'image/png') 432 | .then(() => { 433 | memoryUsages[iteration + 1] = process.memoryUsage(); 434 | return 1; 435 | }); 436 | })); 437 | memoryUsages[count + 1] = process.memoryUsage(); 438 | 439 | const config = generateMemoryUsageChartConfig(memoryUsages, 'New Instance Test'); 440 | const buffer = await new ChartJSNodeCanvas({ width: 800, height: 600 }).renderToBuffer(config); 441 | await fs.writeFile('./resources/memory-usage-new-instance-parallel.png', buffer); 442 | }); 443 | 444 | it('does not leak with new instance sequential', async () => { 445 | 446 | const memoryUsages = new Array(count + 2); 447 | memoryUsages[0] = process.memoryUsage(); 448 | for (let index = 0; index < count; index++) { 449 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback }); 450 | await chartJSNodeCanvas.renderToBuffer(configuration, 'image/png'); 451 | memoryUsages[index + 1] = process.memoryUsage(); 452 | 453 | } 454 | memoryUsages[count + 1] = process.memoryUsage(); 455 | 456 | const config = generateMemoryUsageChartConfig(memoryUsages, 'New Instance Test'); 457 | const buffer = await new ChartJSNodeCanvas({ width: 800, height: 600 }).renderToBuffer(config); 458 | await fs.writeFile('./resources/memory-usage-new-instance-sequential.png', buffer); 459 | }); 460 | 461 | it('does not leak with same instance', async () => { 462 | 463 | const memoryUsages = new Array(count + 2); 464 | memoryUsages[0] = process.memoryUsage(); 465 | const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height, chartCallback }); 466 | const diffs = await Promise.all([...Array(count).keys()].map((iteration) => { 467 | return chartJSNodeCanvas.renderToBuffer(configuration, 'image/png') 468 | .then(() => { 469 | memoryUsages[iteration + 1] = process.memoryUsage(); 470 | return 1; 471 | }); 472 | })); 473 | memoryUsages[count + 1] = process.memoryUsage(); 474 | const config = generateMemoryUsageChartConfig(memoryUsages, 'Same Instance Test'); 475 | const buffer = await new ChartJSNodeCanvas({ width: 800, height: 600 }).renderToBuffer(config); 476 | await fs.writeFile('./resources/memory-usage-same-instance.png', buffer); 477 | }); 478 | 479 | function generateMemoryUsageChartConfig(memoryUsages: NodeJS.MemoryUsage[], _testName: string): ChartConfiguration { 480 | const labels = memoryUsages.map((_, index) => `Iteration ${index}`); 481 | const data = { 482 | labels, 483 | datasets: [ 484 | { 485 | label: 'RSS', 486 | data: memoryUsages.map(usage => usage.rss / 1000000), // Convert to MB 487 | backgroundColor: 'rgba(255, 99, 132, 0.2)', 488 | borderColor: 'rgba(255, 99, 132, 1)', 489 | borderWidth: 1 490 | }, 491 | { 492 | label: 'Heap Total', 493 | data: memoryUsages.map(usage => usage.heapTotal / 1000000), // Convert to MB 494 | backgroundColor: 'rgba(54, 162, 235, 0.2)', 495 | borderColor: 'rgba(54, 162, 235, 1)', 496 | borderWidth: 1 497 | }, 498 | { 499 | label: 'Heap Used', 500 | data: memoryUsages.map(usage => usage.heapUsed / 1000000), // Convert to MB 501 | backgroundColor: 'rgba(75, 192, 192, 0.2)', 502 | borderColor: 'rgba(75, 192, 192, 1)', 503 | borderWidth: 1 504 | }, 505 | { 506 | label: 'External', 507 | data: memoryUsages.map(usage => usage.external / 1000000), // Convert to MB 508 | backgroundColor: 'rgba(153, 102, 255, 0.2)', 509 | borderColor: 'rgba(153, 102, 255, 1)', 510 | borderWidth: 1 511 | } 512 | ] 513 | }; 514 | 515 | return { 516 | type: 'line', 517 | data, 518 | options: { 519 | responsive: true, 520 | maintainAspectRatio: false, 521 | scales: { 522 | y: { 523 | beginAtZero: true, 524 | ticks: { 525 | callback: (value: number) => `${value} MB` 526 | } as any 527 | } 528 | } 529 | } 530 | }; 531 | } 532 | }); 533 | }); 534 | -------------------------------------------------------------------------------- /testData/linuxw/render-to-data-URL.txt: -------------------------------------------------------------------------------- 1 |  -------------------------------------------------------------------------------- /testData/linux/render-to-data-URL-actual.txt: -------------------------------------------------------------------------------- 1 |  -------------------------------------------------------------------------------- /testData/linux/render-to-data-URL-compare.html: -------------------------------------------------------------------------------- 1 |
Actual:
2 | 3 |
Expected:
--------------------------------------------------------------------------------