├── .editorconfig ├── .eslintignore ├── .eslintrc ├── .github ├── PULL_REQUEST_TEMPLATE.md └── workflows │ └── npm-publish.yml ├── .gitignore ├── .prettierignore ├── .prettierrc ├── .vscode └── settings.json ├── LICENSE ├── README.md ├── examples ├── package.json ├── src │ ├── basic.ts │ ├── network.ts │ ├── printBarCode.ts │ ├── printImage.ts │ ├── printTable.ts │ └── templates.ts ├── tsconfig.json └── yarn.lock ├── package.json ├── resources └── ESCPOS_Command_Manual.pdf ├── src ├── buffer-builder.ts ├── command.ts ├── escpos.ts ├── image.ts ├── index.ts ├── node-factory.ts ├── nodes │ ├── align-node.ts │ ├── barcode-node.ts │ ├── bold-node.ts │ ├── break-line-node.ts │ ├── document-node.ts │ ├── image-node.ts │ ├── line-feed-node.ts │ ├── open-cash-drawer-node.ts │ ├── paper-cut-node.ts │ ├── print-mode.ts │ ├── qrcode-node.ts │ ├── small-node.ts │ ├── text-line-node.ts │ ├── text-node.ts │ ├── underline-node.ts │ └── white-mode-node.ts ├── template-parser.ts ├── xml-node.ts └── xml-parser.ts ├── tsconfig.build.json ├── tsconfig.json ├── tslint.json └── yarn.lock /.editorconfig: -------------------------------------------------------------------------------- 1 | # EditorConfig helps developers define and maintain consistent coding styles between different editors and IDEs 2 | # editorconfig.org 3 | 4 | root = true 5 | 6 | [*] 7 | indent_style = space 8 | indent_size = 2 9 | 10 | # We recommend you to keep these unchanged 11 | end_of_line = lf 12 | charset = utf-8 13 | trim_trailing_whitespace = true 14 | insert_final_newline = true 15 | 16 | [*.md] 17 | trim_trailing_whitespace = false -------------------------------------------------------------------------------- /.eslintignore: -------------------------------------------------------------------------------- 1 | **/node_modules 2 | **/lib 3 | eslint_report.json 4 | .nyc_output 5 | coverage 6 | .vscode 7 | jest.config.js 8 | *.d.ts -------------------------------------------------------------------------------- /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "parser": "@typescript-eslint/parser", 3 | "extends": [ 4 | "plugin:@typescript-eslint/recommended", 5 | "prettier", 6 | "prettier/@typescript-eslint" 7 | ], 8 | "plugins": ["@typescript-eslint"], 9 | "rules": { 10 | "no-trailing-spaces": "warn", 11 | "comma-dangle": "off", 12 | "quotes": [ 13 | "error", 14 | "single", 15 | { 16 | "avoidEscape": true 17 | } 18 | ], 19 | "semi": "warn", 20 | "@typescript-eslint/no-var-requires": "off", 21 | "@typescript-eslint/ban-types": "off", 22 | "@typescript-eslint/explicit-module-boundary-types": "off", 23 | "@typescript-eslint/ban-ts-ignore": "off", 24 | "@typescript-eslint/camelcase": "off" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /.github/PULL_REQUEST_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | ## Description 2 | 3 | 4 | ## Related Issues 5 | 6 | 7 | 8 | 9 | Issue: #XXX 10 | 11 | ## How Has This Been Tested 12 | 13 | 14 | 15 | 16 | ## Types of changes 17 | 18 | 19 | - [ ] Bug fix (non-breaking change which fixes an issue) 20 | - [ ] New feature (non-breaking change which adds functionality) 21 | - [ ] Breaking change (fix or feature that would cause existing functionality to change) 22 | 23 | ## Checklist 24 | 25 | 26 | 27 | - [ ] My code follows the code style of this project. 28 | - [ ] My change requires a change to the documentation. 29 | - [ ] I have updated the documentation accordingly. 30 | - [ ] I have added tests to cover my changes. 31 | - [ ] All new and existing tests passed. 32 | 33 | ## Screenshots (if appropriate) 34 | -------------------------------------------------------------------------------- /.github/workflows/npm-publish.yml: -------------------------------------------------------------------------------- 1 | name: npm-publish 2 | on: 3 | push: 4 | tags: 5 | - 'v*.*.*' 6 | 7 | jobs: 8 | npm-publish: 9 | name: npm-publish 10 | runs-on: ubuntu-latest 11 | steps: 12 | - name: Checkout repository 13 | uses: actions/checkout@v2 14 | - uses: actions/setup-node@v1 15 | with: 16 | node-version: 12 17 | registry-url: https://registry.npmjs.org/ 18 | - name: Install Node Dependencies 19 | run: yarn install; cd examples; yarn install; cd .. 20 | - name: Build packages 21 | run: yarn build 22 | env: 23 | CI: "TRUE" 24 | - run: npm publish --access public 25 | env: 26 | NODE_AUTH_TOKEN: ${{secrets.NPM_AUTH_TOKEN}} 27 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | **/lib 2 | **/node_modules 3 | coverage/* 4 | **/dist 5 | -------------------------------------------------------------------------------- /.prettierignore: -------------------------------------------------------------------------------- 1 | **/node_modules 2 | **/build 3 | **/dist 4 | **/.github 5 | eslint_report.json 6 | .nyc_output 7 | coverage 8 | .vscode 9 | **/.jest 10 | .jest -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "trailingComma": "all", 3 | "singleQuote": true, 4 | "semi": true, 5 | "tabWidth": 2, 6 | "useTabs": false, 7 | "arrowParens": "avoid", 8 | "endOfLine": "auto", 9 | "bracketSpacing": true 10 | } -------------------------------------------------------------------------------- /.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "files.exclude": { 3 | "**/.git": true, 4 | "**/.github": true, 5 | "**/.hg": true, 6 | "**/CVS": true, 7 | "**/.DS_Store": true, 8 | "**/node_modules": true, 9 | "**/coverage": true, 10 | "**/lib": true, 11 | "**/yarn.lock": true, 12 | "**/dist": true 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, 10 | and distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by 13 | the copyright owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all 16 | other entities that control, are controlled by, or are under common 17 | control with that entity. For the purposes of this definition, 18 | "control" means (i) the power, direct or indirect, to cause the 19 | direction or management of such entity, whether by contract or 20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 21 | outstanding shares, or (iii) beneficial ownership of such entity. 22 | 23 | "You" (or "Your") shall mean an individual or Legal Entity 24 | exercising permissions granted by this License. 25 | 26 | "Source" form shall mean the preferred form for making modifications, 27 | including but not limited to software source code, documentation 28 | source, and configuration files. 29 | 30 | "Object" form shall mean any form resulting from mechanical 31 | transformation or translation of a Source form, including but 32 | not limited to compiled object code, generated documentation, 33 | and conversions to other media types. 34 | 35 | "Work" shall mean the work of authorship, whether in Source or 36 | Object form, made available under the License, as indicated by a 37 | copyright notice that is included in or attached to the work 38 | (an example is provided in the Appendix below). 39 | 40 | "Derivative Works" shall mean any work, whether in Source or Object 41 | form, that is based on (or derived from) the Work and for which the 42 | editorial revisions, annotations, elaborations, or other modifications 43 | represent, as a whole, an original work of authorship. For the purposes 44 | of this License, Derivative Works shall not include works that remain 45 | separable from, or merely link (or bind by name) to the interfaces of, 46 | the Work and Derivative Works thereof. 47 | 48 | "Contribution" shall mean any work of authorship, including 49 | the original version of the Work and any modifications or additions 50 | to that Work or Derivative Works thereof, that is intentionally 51 | submitted to Licensor for inclusion in the Work by the copyright owner 52 | or by an individual or Legal Entity authorized to submit on behalf of 53 | the copyright owner. For the purposes of this definition, "submitted" 54 | means any form of electronic, verbal, or written communication sent 55 | to the Licensor or its representatives, including but not limited to 56 | communication on electronic mailing lists, source code control systems, 57 | and issue tracking systems that are managed by, or on behalf of, the 58 | Licensor for the purpose of discussing and improving the Work, but 59 | excluding communication that is conspicuously marked or otherwise 60 | designated in writing by the copyright owner as "Not a Contribution." 61 | 62 | "Contributor" shall mean Licensor and any individual or Legal Entity 63 | on behalf of whom a Contribution has been received by Licensor and 64 | subsequently incorporated within the Work. 65 | 66 | 2. Grant of Copyright License. Subject to the terms and conditions of 67 | this License, each Contributor hereby grants to You a perpetual, 68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 69 | copyright license to reproduce, prepare Derivative Works of, 70 | publicly display, publicly perform, sublicense, and distribute the 71 | Work and such Derivative Works in Source or Object form. 72 | 73 | 3. Grant of Patent License. Subject to the terms and conditions of 74 | this License, each Contributor hereby grants to You a perpetual, 75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 76 | (except as stated in this section) patent license to make, have made, 77 | use, offer to sell, sell, import, and otherwise transfer the Work, 78 | where such license applies only to those patent claims licensable 79 | by such Contributor that are necessarily infringed by their 80 | Contribution(s) alone or by combination of their Contribution(s) 81 | with the Work to which such Contribution(s) was submitted. If You 82 | institute patent litigation against any entity (including a 83 | cross-claim or counterclaim in a lawsuit) alleging that the Work 84 | or a Contribution incorporated within the Work constitutes direct 85 | or contributory patent infringement, then any patent licenses 86 | granted to You under this License for that Work shall terminate 87 | as of the date such litigation is filed. 88 | 89 | 4. Redistribution. You may reproduce and distribute copies of the 90 | Work or Derivative Works thereof in any medium, with or without 91 | modifications, and in Source or Object form, provided that You 92 | meet the following conditions: 93 | 94 | (a) You must give any other recipients of the Work or 95 | Derivative Works a copy of this License; and 96 | 97 | (b) You must cause any modified files to carry prominent notices 98 | stating that You changed the files; and 99 | 100 | (c) You must retain, in the Source form of any Derivative Works 101 | that You distribute, all copyright, patent, trademark, and 102 | attribution notices from the Source form of the Work, 103 | excluding those notices that do not pertain to any part of 104 | the Derivative Works; and 105 | 106 | (d) If the Work includes a "NOTICE" text file as part of its 107 | distribution, then any Derivative Works that You distribute must 108 | include a readable copy of the attribution notices contained 109 | within such NOTICE file, excluding those notices that do not 110 | pertain to any part of the Derivative Works, in at least one 111 | of the following places: within a NOTICE text file distributed 112 | as part of the Derivative Works; within the Source form or 113 | documentation, if provided along with the Derivative Works; or, 114 | within a display generated by the Derivative Works, if and 115 | wherever such third-party notices normally appear. The contents 116 | of the NOTICE file are for informational purposes only and 117 | do not modify the License. You may add Your own attribution 118 | notices within Derivative Works that You distribute, alongside 119 | or as an addendum to the NOTICE text from the Work, provided 120 | that such additional attribution notices cannot be construed 121 | as modifying the License. 122 | 123 | You may add Your own copyright statement to Your modifications and 124 | may provide additional or different license terms and conditions 125 | for use, reproduction, or distribution of Your modifications, or 126 | for any such Derivative Works as a whole, provided Your use, 127 | reproduction, and distribution of the Work otherwise complies with 128 | the conditions stated in this License. 129 | 130 | 5. Submission of Contributions. Unless You explicitly state otherwise, 131 | any Contribution intentionally submitted for inclusion in the Work 132 | by You to the Licensor shall be under the terms and conditions of 133 | this License, without any additional terms or conditions. 134 | Notwithstanding the above, nothing herein shall supersede or modify 135 | the terms of any separate license agreement you may have executed 136 | with Licensor regarding such Contributions. 137 | 138 | 6. Trademarks. This License does not grant permission to use the trade 139 | names, trademarks, service marks, or product names of the Licensor, 140 | except as required for reasonable and customary use in describing the 141 | origin of the Work and reproducing the content of the NOTICE file. 142 | 143 | 7. Disclaimer of Warranty. Unless required by applicable law or 144 | agreed to in writing, Licensor provides the Work (and each 145 | Contributor provides its Contributions) on an "AS IS" BASIS, 146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 147 | implied, including, without limitation, any warranties or conditions 148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 149 | PARTICULAR PURPOSE. You are solely responsible for determining the 150 | appropriateness of using or redistributing the Work and assume any 151 | risks associated with Your exercise of permissions under this License. 152 | 153 | 8. Limitation of Liability. In no event and under no legal theory, 154 | whether in tort (including negligence), contract, or otherwise, 155 | unless required by applicable law (such as deliberate and grossly 156 | negligent acts) or agreed to in writing, shall any Contributor be 157 | liable to You for damages, including any direct, indirect, special, 158 | incidental, or consequential damages of any character arising as a 159 | result of this License or out of the use or inability to use the 160 | Work (including but not limited to damages for loss of goodwill, 161 | work stoppage, computer failure or malfunction, or any and all 162 | other commercial damages or losses), even if such Contributor 163 | has been advised of the possibility of such damages. 164 | 165 | 9. Accepting Warranty or Additional Liability. While redistributing 166 | the Work or Derivative Works thereof, You may choose to offer, 167 | and charge a fee for, acceptance of support, warranty, indemnity, 168 | or other liability obligations and/or rights consistent with this 169 | License. However, in accepting such obligations, You may act only 170 | on Your own behalf and on Your sole responsibility, not on behalf 171 | of any other Contributor, and only if You agree to indemnify, 172 | defend, and hold each Contributor harmless for any liability 173 | incurred by, or claims asserted against, such Contributor by reason 174 | of your accepting any such warranty or additional liability. 175 | 176 | END OF TERMS AND CONDITIONS 177 | 178 | APPENDIX: How to apply the Apache License to your work. 179 | 180 | To apply the Apache License to your work, attach the following 181 | boilerplate notice, with the fields enclosed by brackets "{}" 182 | replaced with your own identifying information. (Don't include 183 | the brackets!) The text should be enclosed in the appropriate 184 | comment syntax for the file format. We also recommend that a 185 | file or class name and description of purpose be included on the 186 | same "printed page" as the copyright notice for easier 187 | identification within third-party archives. 188 | 189 | Copyright {yyyy} {name of copyright owner} 190 | 191 | Licensed under the Apache License, Version 2.0 (the "License"); 192 | you may not use this file except in compliance with the License. 193 | You may obtain a copy of the License at 194 | 195 | http://www.apache.org/licenses/LICENSE-2.0 196 | 197 | Unless required by applicable law or agreed to in writing, software 198 | distributed under the License is distributed on an "AS IS" BASIS, 199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 200 | See the License for the specific language governing permissions and 201 | limitations under the License. 202 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | Originally forked from [here](https://github.com/ingoncalves/escpos-xml) 2 | 3 | # ESC/POS XML 4 | 5 | Cross platform JavaScript library that implements the thermal printer ESC / POS protocol and provides an XML interface for preparing templates for printing. 6 | 7 | ## Features 8 | 9 | - [x] Text 10 | - [x] Text line 11 | - [x] Feed line 12 | - [x] Bold text 13 | - [x] Underline text 14 | - [x] Font size 15 | - [x] Small mode 16 | - [x] White mode 17 | - [x] Align 18 | - [x] Barcode 19 | - [x] QRcode 20 | - [x] Paper cut node 21 | - [x] Image (base64) (png only) 22 | - [x] XML with mustache 23 | 24 | 25 | ## Tested manually on following environments or platforms 26 | 27 | - [x] React Native (Android) 28 | - [x] React Native (iOS) 29 | - [x] React Native Web 30 | - [x] Server side (NodeJs) 31 | - [x] Desktop applications (nwjs & electron) 32 | - [x] Other node environment (terminal) 33 | 34 | 35 | ## Installation 36 | 37 | ```bash 38 | yarn add @tillpos/xml-escpos-helper 39 | ``` 40 | 41 | ## Examples 42 | 43 | ### With an XML template + plain object input (regular text). 44 | 45 | ```ts 46 | 47 | import { EscPos } from '@tillpos/xml-escpos-helper'; 48 | 49 | // store this template somewhere `s3` or as `static asset` based on your preference 50 | const template = ` 51 | 52 | 53 | 54 | 55 | {{title}} 56 | 57 | 58 | 59 | {{#thankyouNote}} 60 | 61 | {{{thankyouNote}}} 62 | 63 | 64 | 65 | 66 | 67 | 68 | `; 69 | 70 | const input = { 71 | title: 'Sample', 72 | thankyouNote: 'Welcome...!' 73 | }; 74 | 75 | const buffer = EscPos.getBufferFromTemplate(template, input); 76 | // send this buffer to a stream (eg.: bluetooth or wifi) 77 | 78 | ``` 79 | 80 | ### With an XML template + png image (base64) 81 | 82 | ```ts 83 | const template = ` 84 | 85 | 86 | 87 | {{title}} 88 | 89 | 90 | 91 | {{base64PngImage}} 92 | 93 | 94 | `; 95 | 96 | const input = { 97 | title: 'PNG - base64', 98 | base64PngImage: `` 99 | }; 100 | 101 | const buffer = EscPos.getBufferFromTemplate(template, input); 102 | ``` 103 | 104 | --- 105 | 106 | ## TODO 107 | 108 | - [ ] Font styles (font family) 109 | - [ ] Image bitmap conversion improvements 110 | - [ ] jpeg support 111 | - [ ] Add example apps to repo 112 | - [ ] Removed uglify for some reason, need to bring it back 113 | - [ ] Improve image rendering 114 | 115 | ## Common issues 116 | 117 | - If there is any delay you observe while printing with this library it is mostly due to image manipulations (try without image :mask: ) 118 | 119 | 120 | ## Useful links / resources 121 | 122 | - [ESC / POS Commands manual](./resources/ESCPOS_Command_Manual.pdf) 123 | - A [blog post](https://www.visuality.pl/posts/thermal-printer-protocols-for-image-and-text#:~:text=How%20can%20we%20print%20an,command%20language%20of%20thermal%20printers) explaiing about printing images with ESCPOS 124 | - Similar library for serverside - [node-escpos](https://github.com/song940/node-escpos). 125 | 126 | > Limitations on the react-native framework 127 | 128 | - [FileReader.readAsArrayBuffer](https://github.com/facebook/react-native/issues/21209) was not implemented. 129 | - Most of popular image manupulation libraries does not have support for react-native. eg : [jimp](https://www.npmjs.com/package/jimp), [jpeg-js](https://www.npmjs.com/package/jpeg-js) and [sharp](https://www.npmjs.com/package/sharp). We can use these libraries with some native node lib implemented in react native (some sort of polyfill). 130 | - For png this [library](https://github.com/photopea/UPNG.js) seems to be faster, but when tested this library with it, it is not retaining pixels at some places) 131 | - Use this [node-libs-react-native](https://www.npmjs.com/package/node-libs-react-native) if we need to use this library in react native (adds some mock or js implementation for fs, stream etc) 132 | 133 | --- 134 | 135 | Contributions of any kind welcome! :heart: 136 | 137 | -------------------------------------------------------------------------------- /examples/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "examples", 3 | "version": "1.0.0", 4 | "description": "Using in Nodejs env", 5 | "main": "index.js", 6 | "repository": "http://github.com/oolio-group/xml-escpos-helper.git", 7 | "license": "MIT", 8 | "private": false, 9 | "scripts": { 10 | "clean": "rm -rf ./dist && rm -rf ./tsconfig.build.tsbuildinfo", 11 | "compile": "yarn clean && tsc -p ./tsconfig.json", 12 | "print:basic": "yarn compile && yarn node ./dist/basic.js", 13 | "print:image": "yarn compile && yarn node ./dist/printImage.js", 14 | "print:table": "yarn compile && yarn node ./dist/printTable.js", 15 | "print:barcode": "yarn compile && yarn node ./dist/printBarcode.js" 16 | }, 17 | "dependencies": { 18 | "@tillpos/xml-escpos-helper": "^0.1.8", 19 | "table": "^6.6.0" 20 | }, 21 | "devDependencies": { 22 | "typescript": "^4.2.4" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /examples/src/basic.ts: -------------------------------------------------------------------------------- 1 | import { sendDataToPrinter } from './network'; 2 | import { TEMPLATES } from './templates'; 3 | 4 | const printBasic = async () => { 5 | const input = { 6 | title: 'Sample', 7 | thankyouNote: 'Welcome...!' 8 | }; 9 | 10 | await sendDataToPrinter(input, TEMPLATES.BASIC); 11 | }; 12 | 13 | printBasic(); 14 | -------------------------------------------------------------------------------- /examples/src/network.ts: -------------------------------------------------------------------------------- 1 | import net from 'net'; 2 | import { EscPos } from '@tillpos/xml-escpos-helper'; 3 | const PRINTERS = [{ device_name: 'Epson', host: '192.168.0.8', port: 9100 }]; 4 | 5 | const connectToPrinter = ( 6 | host: string, 7 | port: number, 8 | buffer: Buffer, 9 | ): Promise => { 10 | return new Promise((res: (value: unknown) => void, rej) => { 11 | let device = new net.Socket(); 12 | 13 | device.on('close', () => { 14 | if (device) { 15 | device.destroy(); 16 | device = null; 17 | } 18 | res(true); 19 | return; 20 | }); 21 | 22 | device.on('error', rej); 23 | 24 | device.connect(port, host, () => { 25 | device.write(buffer); 26 | device.emit('close'); 27 | }); 28 | }); 29 | }; 30 | 31 | export const sendDataToPrinter = async (input: any, template: string) => { 32 | const { host, port } = PRINTERS[0]; 33 | const buffer = EscPos.getBufferFromTemplate(template, input); 34 | try { 35 | await connectToPrinter(host, port, (buffer as unknown) as Buffer); 36 | } catch (err) { 37 | console.log('some error', err); 38 | } 39 | }; 40 | -------------------------------------------------------------------------------- /examples/src/printBarCode.ts: -------------------------------------------------------------------------------- 1 | import { sendDataToPrinter } from './network'; 2 | import { TEMPLATES } from './templates'; 3 | 4 | const printBarCode = async () => { 5 | const input = { 6 | barcode: '12345678', 7 | }; 8 | 9 | await sendDataToPrinter(input, TEMPLATES.BAR_CODE); 10 | }; 11 | 12 | printBarCode(); 13 | -------------------------------------------------------------------------------- /examples/src/printImage.ts: -------------------------------------------------------------------------------- 1 | import { sendDataToPrinter } from './network'; 2 | import { TEMPLATES } from './templates'; 3 | 4 | const BASE64_PNG_IMAGE = ``; 5 | 6 | const printImage = async () => { 7 | const input = { 8 | title: 'Github', 9 | logo: BASE64_PNG_IMAGE 10 | }; 11 | 12 | await sendDataToPrinter(input, TEMPLATES.PNG_IMAGE); 13 | }; 14 | 15 | printImage(); 16 | -------------------------------------------------------------------------------- /examples/src/printTable.ts: -------------------------------------------------------------------------------- 1 | import { sendDataToPrinter } from './network'; 2 | import { TEMPLATES } from './templates'; 3 | import { getBorderCharacters, table } from 'table'; 4 | 5 | const printTable = async () => { 6 | const versions = [ 7 | ['React', '16.8'], 8 | ['Angular', '9'], 9 | ['Ember', '3.16'], 10 | ]; 11 | 12 | const tableData = table(versions, { 13 | border: getBorderCharacters(`void`), 14 | drawHorizontalLine: () => { 15 | return false; 16 | }, 17 | }); 18 | 19 | await sendDataToPrinter({ tableData }, TEMPLATES.TABLE); 20 | }; 21 | 22 | printTable(); 23 | -------------------------------------------------------------------------------- /examples/src/templates.ts: -------------------------------------------------------------------------------- 1 | export enum Templates { 2 | PNG_IMAGE = 'PNG_IMAGE', 3 | BASIC = 'BASIC', 4 | QR_CODE = 'QR_CODE', 5 | BAR_CODE = 'BAR_CODE', 6 | TABLE = 'TABLE', 7 | } 8 | 9 | export const TEMPLATES = { 10 | [Templates.BASIC]: ` 11 | 12 | 13 | 14 | {{title}} 15 | 16 | 17 | 18 | {{#thankyouNote}} 19 | 20 | {{{thankyouNote}}} 21 | 22 | {{/thankyouNote}} 23 | 24 | 25 | 26 | 27 | `, 28 | [Templates.PNG_IMAGE]: ` 29 | 30 | 31 | 32 | 33 | {{title}} 34 | 35 | 36 | 37 | {{logo}} 38 | 39 | 40 | `, 41 | [Templates.QR_CODE]: ` 42 | 43 | 44 | 45 | {{qrcode}} 46 | 47 | 48 | 49 | 50 | `, 51 | [Templates.BAR_CODE]: ` 52 | 53 | 54 | 55 | 56 | {{barcode}} 57 | 58 | 59 | 60 | `, 61 | [Templates.TABLE]: ` 62 | 63 | 64 | {{tableData}} 65 | 66 | `, 67 | } 68 | -------------------------------------------------------------------------------- /examples/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "module": "commonjs", 4 | "esModuleInterop": true, 5 | "allowSyntheticDefaultImports": true, 6 | "target": "es6", 7 | "noImplicitAny": true, 8 | "moduleResolution": "node", 9 | "sourceMap": true, 10 | "outDir": "dist", 11 | "baseUrl": ".", 12 | "paths": { 13 | "*": [ 14 | "node_modules/*", 15 | "src/types/*" 16 | ] 17 | } 18 | }, 19 | "include": [ 20 | "src/**/*" 21 | ] 22 | } 23 | -------------------------------------------------------------------------------- /examples/yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@tillpos/xml-escpos-helper@^0.1.8": 6 | version "0.1.8" 7 | resolved "https://registry.yarnpkg.com/@tillpos/xml-escpos-helper/-/xml-escpos-helper-0.1.8.tgz#d3a664d776fa022516f7c85b0280706892e7fa5b" 8 | integrity sha512-2VdutlzFUcCMYxoJ7LmdK4l6/lWXA3TInwoWOdZ2cQKB0V0T3qogrHQGxKgRrlhFds2co0kksaa4ePj5QlFLCw== 9 | dependencies: 10 | buffer "^6.0.3" 11 | mustache "^4.1.0" 12 | mutable-buffer "2.0.3" 13 | ndarray "^1.0.19" 14 | pngjs "^6.0.0" 15 | xml-parser "^1.2.1" 16 | 17 | ajv@^8.0.1: 18 | version "8.2.0" 19 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.2.0.tgz#c89d3380a784ce81b2085f48811c4c101df4c602" 20 | integrity sha512-WSNGFuyWd//XO8n/m/EaOlNLtO0yL8EXT/74LqT4khdhpZjP7lkj/kT5uwRmGitKEVp/Oj7ZUHeGfPtgHhQ5CA== 21 | dependencies: 22 | fast-deep-equal "^3.1.1" 23 | json-schema-traverse "^1.0.0" 24 | require-from-string "^2.0.2" 25 | uri-js "^4.2.2" 26 | 27 | ansi-regex@^5.0.0: 28 | version "5.0.0" 29 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.0.tgz#388539f55179bf39339c81af30a654d69f87cb75" 30 | integrity sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg== 31 | 32 | ansi-styles@^4.0.0: 33 | version "4.3.0" 34 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" 35 | integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== 36 | dependencies: 37 | color-convert "^2.0.1" 38 | 39 | astral-regex@^2.0.0: 40 | version "2.0.0" 41 | resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" 42 | integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== 43 | 44 | base64-js@^1.3.1: 45 | version "1.5.1" 46 | resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" 47 | integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== 48 | 49 | buffer@^6.0.3: 50 | version "6.0.3" 51 | resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6" 52 | integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA== 53 | dependencies: 54 | base64-js "^1.3.1" 55 | ieee754 "^1.2.1" 56 | 57 | color-convert@^2.0.1: 58 | version "2.0.1" 59 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" 60 | integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== 61 | dependencies: 62 | color-name "~1.1.4" 63 | 64 | color-name@~1.1.4: 65 | version "1.1.4" 66 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" 67 | integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== 68 | 69 | debug@^2.2.0: 70 | version "2.6.9" 71 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" 72 | integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== 73 | dependencies: 74 | ms "2.0.0" 75 | 76 | emoji-regex@^8.0.0: 77 | version "8.0.0" 78 | resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" 79 | integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== 80 | 81 | fast-deep-equal@^3.1.1: 82 | version "3.1.3" 83 | resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" 84 | integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== 85 | 86 | ieee754@^1.2.1: 87 | version "1.2.1" 88 | resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" 89 | integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== 90 | 91 | iota-array@^1.0.0: 92 | version "1.0.0" 93 | resolved "https://registry.yarnpkg.com/iota-array/-/iota-array-1.0.0.tgz#81ef57fe5d05814cd58c2483632a99c30a0e8087" 94 | integrity sha1-ge9X/l0FgUzVjCSDYyqZwwoOgIc= 95 | 96 | is-buffer@^1.0.2: 97 | version "1.1.6" 98 | resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" 99 | integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== 100 | 101 | is-fullwidth-code-point@^3.0.0: 102 | version "3.0.0" 103 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" 104 | integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== 105 | 106 | json-schema-traverse@^1.0.0: 107 | version "1.0.0" 108 | resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2" 109 | integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== 110 | 111 | lodash.clonedeep@^4.5.0: 112 | version "4.5.0" 113 | resolved "https://registry.yarnpkg.com/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz#e23f3f9c4f8fbdde872529c1071857a086e5ccef" 114 | integrity sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8= 115 | 116 | lodash.flatten@^4.4.0: 117 | version "4.4.0" 118 | resolved "https://registry.yarnpkg.com/lodash.flatten/-/lodash.flatten-4.4.0.tgz#f31c22225a9632d2bbf8e4addbef240aa765a61f" 119 | integrity sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8= 120 | 121 | lodash.truncate@^4.4.2: 122 | version "4.4.2" 123 | resolved "https://registry.yarnpkg.com/lodash.truncate/-/lodash.truncate-4.4.2.tgz#5a350da0b1113b837ecfffd5812cbe58d6eae193" 124 | integrity sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM= 125 | 126 | ms@2.0.0: 127 | version "2.0.0" 128 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" 129 | integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g= 130 | 131 | mustache@^4.1.0: 132 | version "4.2.0" 133 | resolved "https://registry.yarnpkg.com/mustache/-/mustache-4.2.0.tgz#e5892324d60a12ec9c2a73359edca52972bf6f64" 134 | integrity sha512-71ippSywq5Yb7/tVYyGbkBggbU8H3u5Rz56fH60jGFgr8uHwxs+aSKeqmluIVzM0m0kB7xQjKS6qPfd0b2ZoqQ== 135 | 136 | mutable-buffer@2.0.3: 137 | version "2.0.3" 138 | resolved "https://registry.yarnpkg.com/mutable-buffer/-/mutable-buffer-2.0.3.tgz#67bca5568bf15d3f6035fb716e3899dc44901c8c" 139 | integrity sha1-Z7ylVovxXT9gNftxbjiZ3ESQHIw= 140 | 141 | ndarray@^1.0.19: 142 | version "1.0.19" 143 | resolved "https://registry.yarnpkg.com/ndarray/-/ndarray-1.0.19.tgz#6785b5f5dfa58b83e31ae5b2a058cfd1ab3f694e" 144 | integrity sha512-B4JHA4vdyZU30ELBw3g7/p9bZupyew5a7tX1Y/gGeF2hafrPaQZhgrGQfsvgfYbgdFZjYwuEcnaobeM/WMW+HQ== 145 | dependencies: 146 | iota-array "^1.0.0" 147 | is-buffer "^1.0.2" 148 | 149 | pngjs@^6.0.0: 150 | version "6.0.0" 151 | resolved "https://registry.yarnpkg.com/pngjs/-/pngjs-6.0.0.tgz#ca9e5d2aa48db0228a52c419c3308e87720da821" 152 | integrity sha512-TRzzuFRRmEoSW/p1KVAmiOgPco2Irlah+bGFCeNfJXxxYGwSw7YwAOAcd7X28K/m5bjBWKsC29KyoMfHbypayg== 153 | 154 | punycode@^2.1.0: 155 | version "2.1.1" 156 | resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.1.tgz#b58b010ac40c22c5657616c8d2c2c02c7bf479ec" 157 | integrity sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A== 158 | 159 | require-from-string@^2.0.2: 160 | version "2.0.2" 161 | resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909" 162 | integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== 163 | 164 | slice-ansi@^4.0.0: 165 | version "4.0.0" 166 | resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-4.0.0.tgz#500e8dd0fd55b05815086255b3195adf2a45fe6b" 167 | integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ== 168 | dependencies: 169 | ansi-styles "^4.0.0" 170 | astral-regex "^2.0.0" 171 | is-fullwidth-code-point "^3.0.0" 172 | 173 | string-width@^4.2.0: 174 | version "4.2.2" 175 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.2.tgz#dafd4f9559a7585cfba529c6a0a4f73488ebd4c5" 176 | integrity sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA== 177 | dependencies: 178 | emoji-regex "^8.0.0" 179 | is-fullwidth-code-point "^3.0.0" 180 | strip-ansi "^6.0.0" 181 | 182 | strip-ansi@^6.0.0: 183 | version "6.0.0" 184 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.0.tgz#0b1571dd7669ccd4f3e06e14ef1eed26225ae532" 185 | integrity sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w== 186 | dependencies: 187 | ansi-regex "^5.0.0" 188 | 189 | table@^6.6.0: 190 | version "6.6.0" 191 | resolved "https://registry.yarnpkg.com/table/-/table-6.6.0.tgz#905654b79df98d9e9a973de1dd58682532c40e8e" 192 | integrity sha512-iZMtp5tUvcnAdtHpZTWLPF0M7AgiQsURR2DwmxnJwSy8I3+cY+ozzVvYha3BOLG2TB+L0CqjIz+91htuj6yCXg== 193 | dependencies: 194 | ajv "^8.0.1" 195 | lodash.clonedeep "^4.5.0" 196 | lodash.flatten "^4.4.0" 197 | lodash.truncate "^4.4.2" 198 | slice-ansi "^4.0.0" 199 | string-width "^4.2.0" 200 | strip-ansi "^6.0.0" 201 | 202 | typescript@^4.2.4: 203 | version "4.2.4" 204 | resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.2.4.tgz#8610b59747de028fda898a8aef0e103f156d0961" 205 | integrity sha512-V+evlYHZnQkaz8TRBuxTA92yZBPotr5H+WhQ7bD3hZUndx5tGOa1fuCgeSjxAzM1RiN5IzvadIXTVefuuwZCRg== 206 | 207 | uri-js@^4.2.2: 208 | version "4.4.1" 209 | resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" 210 | integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== 211 | dependencies: 212 | punycode "^2.1.0" 213 | 214 | xml-parser@^1.2.1: 215 | version "1.2.1" 216 | resolved "https://registry.yarnpkg.com/xml-parser/-/xml-parser-1.2.1.tgz#c31f4c34f2975db82ad013222120592736156fcd" 217 | integrity sha1-wx9MNPKXXbgq0BMiISBZJzYVb80= 218 | dependencies: 219 | debug "^2.2.0" 220 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@tillpos/xml-escpos-helper", 3 | "description": "ESC/POS with XML interface", 4 | "keywords": [ 5 | "escpos", 6 | "esc", 7 | "pos", 8 | "xml", 9 | "printer" 10 | ], 11 | "version": "0.3.1", 12 | "files": [ 13 | "lib", 14 | "LICENSE", 15 | "README.md" 16 | ], 17 | "main": "lib/index", 18 | "types": "lib", 19 | "engines": { 20 | "node": ">=4.0.0" 21 | }, 22 | "license": "Apache-2.0", 23 | "scripts": { 24 | "build": "yarn clean && yarn compile", 25 | "build:watch": "yarn compile -- -w", 26 | "clean": "rm -rf ./lib && rm -rf tsconfig.build.tsbuildinfo", 27 | "compile": "tsc -p ./tsconfig.build.json" 28 | }, 29 | "repository": { 30 | "type": "git", 31 | "url": "http://github.com/oolio-group/xml-escpos-helper.git" 32 | }, 33 | "contributors": [ 34 | "Snehit Velma " 35 | ], 36 | "bugs": { 37 | "url": "https://github.com/oolio-group/xml-escpos-helper/issues" 38 | }, 39 | "dependencies": { 40 | "buffer": "^6.0.3", 41 | "mustache": "^4.1.0", 42 | "mutable-buffer": "2.0.3", 43 | "ndarray": "^1.0.19", 44 | "upng-js": "2.1.0", 45 | "xml-parser": "^1.2.1" 46 | }, 47 | "devDependencies": { 48 | "@types/node": "^14.14.20", 49 | "@types/xml-parser": "^1.2.29", 50 | "ts-loader": "^2.3.4", 51 | "tslint": "^5.7.0", 52 | "tslint-eslint-rules": "^4.1.1", 53 | "typescript": "^4.1.3" 54 | } 55 | } -------------------------------------------------------------------------------- /resources/ESCPOS_Command_Manual.pdf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/oolio-group/xml-escpos-helper/a80bd47b56ffe6400d9297c0bb0124300f46bd9a/resources/ESCPOS_Command_Manual.pdf -------------------------------------------------------------------------------- /src/buffer-builder.ts: -------------------------------------------------------------------------------- 1 | import { Command } from "./command"; 2 | import { MutableBuffer } from "mutable-buffer"; 3 | import Image from "./image"; 4 | export class BufferBuilder { 5 | private buffer: MutableBuffer; 6 | private hasGSCommand: boolean; 7 | private doEmphasise: boolean; 8 | 9 | constructor(private defaultSettings: boolean = true) { 10 | this.buffer = new MutableBuffer(); 11 | this.hasGSCommand = true; 12 | this.doEmphasise = false; 13 | 14 | } 15 | 16 | public end(): BufferBuilder { 17 | return this; 18 | } 19 | 20 | public resetCharacterCodeTable(): BufferBuilder { 21 | this.buffer.write(Command.ESC_t(0)); 22 | return this; 23 | } 24 | 25 | public setPrintColor(color: 'black' | 'red' = 'black'): BufferBuilder { 26 | this.buffer.write(Command.ESC_r(color === 'red' ? 1 : 0)); 27 | return this; 28 | } 29 | 30 | public setCharacterSize( 31 | width: number = 0, 32 | height: number = 0 33 | ): BufferBuilder { 34 | if(this.hasGSCommand) 35 | { 36 | let size = (width << 4) + height; 37 | this.buffer.write(Command.GS_exclamation(size)); 38 | } 39 | else{ 40 | let mode = (width>0?(1<<5):0)+(height>0?(1<<4):0)+(this.doEmphasise?(1<<3):0)+1; 41 | this.buffer.write(Command.ESC_exclamation(mode)); 42 | } 43 | 44 | return this; 45 | } 46 | 47 | public setPrintMode(setting:boolean): BufferBuilder { 48 | this.hasGSCommand = setting; 49 | if(setting) this.resetCharacterEncoding(); 50 | this.resetCharacterSize(); 51 | return this; 52 | } 53 | 54 | public resetCharacterSize(): BufferBuilder { 55 | if(this.hasGSCommand) 56 | { 57 | this.buffer.write(Command.GS_exclamation(0)); 58 | } 59 | else 60 | { 61 | this.buffer.write(Command.ESC_exclamation(1)); 62 | } 63 | return this; 64 | } 65 | 66 | public resetCharacterEncoding(): BufferBuilder { 67 | this.buffer.write(Command.FS_ob_C_fe_utf); 68 | return this; 69 | } 70 | 71 | public startCompressedCharacter(): BufferBuilder { 72 | this.buffer.write(Command.ESC_M(1)); 73 | return this; 74 | } 75 | 76 | public endCompressedCharacter(): BufferBuilder { 77 | this.buffer.write(Command.ESC_M(0)); 78 | return this; 79 | } 80 | 81 | public startBold(): BufferBuilder { 82 | this.hasGSCommand? this.buffer.write(Command.ESC_E(1)):this.doEmphasise=true; 83 | return this; 84 | } 85 | 86 | public endBold(): BufferBuilder { 87 | this.hasGSCommand? this.buffer.write(Command.ESC_E(0)):this.doEmphasise=false; 88 | return this; 89 | } 90 | 91 | public startUnderline( 92 | underlineMode: UNDERLINE_MODE = UNDERLINE_MODE.TWO_POINTS_OF_COARSE 93 | ): BufferBuilder { 94 | this.buffer.write(Command.ESC_minus(underlineMode)); 95 | return this; 96 | } 97 | 98 | public endUnderline(): BufferBuilder { 99 | this.buffer.write(Command.ESC_minus(48)); 100 | return this; 101 | } 102 | 103 | public startAlign(alignment: ALIGNMENT): BufferBuilder { 104 | this.buffer.write(Command.ESC_a(alignment)); 105 | return this; 106 | } 107 | 108 | public resetAlign(): BufferBuilder { 109 | return this.startAlign(ALIGNMENT.LEFT); 110 | } 111 | 112 | public startWhiteMode(): BufferBuilder { 113 | this.buffer.write(Command.GS_B(1)); 114 | return this; 115 | } 116 | 117 | public endWhiteMode(): BufferBuilder { 118 | this.buffer.write(Command.GS_B(0)); 119 | return this; 120 | } 121 | 122 | public startReverseMode(): BufferBuilder { 123 | this.buffer.write(Command.ESC_rev(1)); 124 | return this; 125 | } 126 | 127 | public endReverseMode(): BufferBuilder { 128 | this.buffer.write(Command.ESC_rev(0)); 129 | return this; 130 | } 131 | 132 | public printBarcode( 133 | data: string, 134 | barcodeSystem: BARCODE_SYSTEM, 135 | width: BARCODE_WIDTH = BARCODE_WIDTH.DOT_375, 136 | height: number = 162, 137 | labelFont: BARCODE_LABEL_FONT = BARCODE_LABEL_FONT.FONT_A, 138 | labelPosition: BARCODE_LABEL_POSITION = BARCODE_LABEL_POSITION.BOTTOM, 139 | leftSpacing: number = 0 140 | ): BufferBuilder { 141 | this.buffer.write(Command.GS_w(width)); // width 142 | this.buffer.write(Command.GS_h(height)); // height 143 | this.buffer.write(Command.GS_x(leftSpacing)); // left spacing 144 | this.buffer.write(Command.GS_f(labelFont)); // HRI font 145 | this.buffer.write(Command.GS_H(labelPosition)); // HRI font 146 | this.buffer.write(Command.GS_K(barcodeSystem, data.length)); // data is a string in UTF-8 147 | this.buffer.write(data, "ascii"); 148 | return this; 149 | } 150 | 151 | public printQRcode(data: string, model: number, size: number, ecLevel: number): BufferBuilder { 152 | let x = data.length + 3; 153 | let pL = Math.floor(x % 256); 154 | let pH = Math.floor(x / 256); 155 | 156 | this.buffer.write(Command.QR_MODEL(model)); 157 | this.buffer.write(Command.QR_SIZE(size)); 158 | this.buffer.write(Command.EC_LEVEL(ecLevel)); 159 | this.buffer.write(Command.STORE_QR(pL, pH)); 160 | this.buffer.write(data, "ascii"); 161 | this.buffer.write(Command.PRINT_QR()); 162 | 163 | return this; 164 | } 165 | 166 | public printBitmap( 167 | image: number[], 168 | width: number, 169 | height: number, 170 | scale: BITMAP_SCALE = BITMAP_SCALE.NORMAL 171 | ): BufferBuilder { 172 | //TODO 173 | return this; 174 | } 175 | 176 | public printText(text: string): BufferBuilder { 177 | this.buffer.write(text, "utf8"); 178 | return this; 179 | } 180 | 181 | public printTextLine(text: string): BufferBuilder { 182 | return this.printText(text).breakLine(); 183 | } 184 | 185 | public breakLine(lines: number = 0): BufferBuilder { 186 | this.buffer.write(Command.ESC_d(lines)); 187 | return this; 188 | } 189 | 190 | public lineFeed(): BufferBuilder { 191 | this.buffer.write(Command.LF); 192 | return this; 193 | } 194 | 195 | public transmitStatus(statusType: STATUS_TYPE): BufferBuilder { 196 | this.buffer.write(Command.DLE_EOT(statusType)); 197 | return this; 198 | } 199 | 200 | public build(): number[] { 201 | if (this.defaultSettings) { 202 | this.lineFeed(); 203 | this.buffer.write(Command.ESC_init); 204 | } 205 | 206 | return this.buffer.flush(); 207 | } 208 | 209 | /** 210 | * Register Paper Cut Action 211 | * @return BufferBuilder 212 | */ 213 | public paperCut(): BufferBuilder { 214 | this.buffer.write(Command.GS_v(66,50)); 215 | return this; 216 | } 217 | 218 | /** 219 | * Register open cash drawer action 220 | * @return BufferBuilder 221 | */ 222 | public openCashDrawer(): BufferBuilder { 223 | // kick drawer 1, pin 2 224 | this.buffer.write(Command.CD_KICK_1()); 225 | // kick drawer 2, pin 5 226 | this.buffer.write(Command.CD_KICK_2()); 227 | return this; 228 | } 229 | 230 | public printImage(image: Image, mode: RASTER_MODE): BufferBuilder { 231 | if (!(image instanceof Image)) { 232 | throw new TypeError("not supported"); 233 | } 234 | const raster = image.toRaster(); 235 | this.buffer.write(Command.GS_v0(mode)); 236 | this.buffer.writeUInt16LE(raster.width); 237 | this.buffer.writeUInt16LE(raster.height); 238 | this.buffer.write(raster.data); 239 | return this; 240 | } 241 | } 242 | 243 | export enum UNDERLINE_MODE { 244 | ONE_POINT_OF_COARSE = 49, 245 | TWO_POINTS_OF_COARSE = 50, 246 | } 247 | 248 | export enum ALIGNMENT { 249 | LEFT = 48, 250 | CENTER = 49, 251 | RIGHT = 50, 252 | } 253 | 254 | export enum BARCODE_SYSTEM { 255 | UPC_A = 65, 256 | UPC_E = 66, 257 | EAN_13 = 67, 258 | EAN_8 = 68, 259 | CODE_39 = 69, 260 | ITF = 70, 261 | CODABAR = 71, 262 | CODE_93 = 72, 263 | CODE_128 = 73, 264 | } 265 | 266 | export enum BARCODE_WIDTH { 267 | DOT_250 = 2, 268 | DOT_375 = 3, 269 | DOT_560 = 4, 270 | DOT_625 = 5, 271 | DOT_750 = 6, 272 | } 273 | 274 | export enum BARCODE_LABEL_FONT { 275 | FONT_A = 48, 276 | FONT_B = 49, 277 | } 278 | 279 | export enum BARCODE_LABEL_POSITION { 280 | NOT_PRINT = 48, 281 | ABOVE = 49, 282 | BOTTOM = 50, 283 | ABOVE_BOTTOM = 51, 284 | } 285 | 286 | export enum QR_EC_LEVEL { 287 | L = 0, 288 | M = 1, 289 | Q = 2, 290 | H = 3, 291 | } 292 | 293 | export enum BITMAP_SCALE { 294 | NORMAL = 48, 295 | DOUBLE_WIDTH = 49, 296 | DOUBLE_HEIGHT = 50, 297 | FOUR_TIMES = 51, 298 | } 299 | 300 | export enum STATUS_TYPE { 301 | PRINTER_STATUS = 1, 302 | OFFLINE_STATUS = 2, 303 | ERROR_STATUS = 3, 304 | PAPER_ROLL_SENSOR_STATUS = 4, 305 | } 306 | 307 | export enum RASTER_MODE { 308 | NORMAL = 0, 309 | DOUBLE_WIDTH = 1, 310 | DOUBLE_HEIGHT = 2, 311 | DOUBLE_WIDTH_HEIGHT = 3, 312 | } 313 | -------------------------------------------------------------------------------- /src/command.ts: -------------------------------------------------------------------------------- 1 | export class Command { 2 | 3 | public static ESC: number = 0x1B; 4 | public static FF: number = 0x0C; 5 | public static FS: number = 0x1C; 6 | public static GS: number = 0x1D; 7 | public static DC1: number = 0x11; 8 | public static DC4: number = 0x14; 9 | public static DLE: number = 0x10; 10 | public static NL: number = 0x0A; 11 | public static SP: number = 0x20; 12 | public static US: number = 0x1F; 13 | 14 | public static DLE_EOT = (n: number): number[] => [Command.DLE, 0x04, n]; // DLEEOTn 15 | 16 | public static ESC_init: number[] = [Command.ESC, 0x40]; //ESC@ 17 | public static ESC_exclamation = (n: number): number[] => [Command.ESC, 0x21, n]; // ESC!n 18 | public static ESC_minus = (n: number): number[] => [Command.ESC, 0x2D, n]; // ESC-n 19 | public static ESC_rev = (n: number): number[] => [Command.ESC, 0x7B, n]; // ESC{n 20 | public static ESC_a = (n: number): number[] => [Command.ESC, 0x61, n]; // ESCan 21 | public static ESC_d = (n: number): number[] => [Command.ESC, 0x64, n]; // ESCdn 22 | public static ESC_E = (n: number): number[] => [Command.ESC, 0x45, n]; // ESCEn 23 | public static ESC_G = (n: number): number[] => [Command.ESC, 0x47, n]; // ESCGn 24 | public static ESC_J = (n: number): number[] => [Command.ESC, 0x4A, n]; // ESCJn 25 | public static ESC_M = (n: number): number[] => [Command.ESC, 0x4D, n]; // ESCMn 26 | public static ESC_t = (n: number): number[] => [Command.ESC, 0x07, n]; // ESCtn 27 | public static ESC_Z = (m: number, n: number, k: number): number[] => [Command.ESC, 0x5A, m, n, k]; // ESCZmnk 28 | public static ESC_r = (n: number): number[] => [Command.ESC, 0x72, n]; // ESCR 29 | 30 | 31 | public static FS_and: number[] = [Command.FS, 0x40]; //ESC@ 32 | public static FS_ob_C_fe_utf = [Command.FS, 0x28, 0x43, 0x02, 0x00, 0x30, 0x02]; //UTF-8 encoding 33 | 34 | public static GS_exclamation = (n: number): number[] => [Command.GS, 0x21, n]; // ESC!n 35 | public static GS_B = (n: number): number[] => [Command.GS, 0x42, n]; // GSBn 36 | public static GS_f = (n: number): number[] => [Command.GS, 0x66, n]; // GSfn 37 | public static GS_h = (n: number): number[] => [Command.GS, 0x68, n]; // GShn 38 | public static GS_H = (n: number): number[] => [Command.GS, 0x48, n]; // GSHn 39 | public static GS_K = (m: number, n: number): number[] => [Command.GS, 0x6B, m, n]; // GSKmn 40 | public static GS_v0 = (m: number): number[] => [Command.GS, 0x76, 0x30, m]; // GSv0m 41 | public static GS_w = (n: number): number[] => [Command.GS, 0x77, n]; // GSwn 42 | public static GS_x = (n: number): number[] => [Command.GS, 0x78, n]; // GSxn 43 | public static GS_v = (m: number, n: number): number[] => [Command.GS, 0x56, m, n]; // GSv 44 | public static ESC_ak = (n: number): number[] => [Command.ESC, 0x2A, n]; // ESC*n 45 | public static ESC_akp = (m: number, nL: number,nH: number ): number[] => [Command.ESC, 0x2A, m, nL, nH]; // ESC*n 46 | 47 | public static LF: number[] = [Command.NL]; 48 | 49 | 50 | // Cash Drawer 51 | public static CD_KICK_1 = (): number[] => [Command.ESC, 0x70, 0x00]; // Sends a pulse to pin 2 52 | public static CD_KICK_2 = (): number[] => [Command.ESC, 0x70, 0x01]; // Sends a pulse to pin 5 53 | 54 | //QR Code 55 | public static QR_MODEL =(n: number): number[]=> [29, 40, 107, 4, 0, 49, 65, n, 0]; //Select QR model, n = 49/50 56 | //[29 40 107 4 0 49 65 n1 n2] 57 | public static QR_SIZE =(n: number): number[] => [29, 40, 107, 3, 0, 49, 67, n]; //Set QR size, n = 8? 58 | //[29 40 107 3 0 49 67 n] 59 | public static EC_LEVEL =(n: number): number[] => [29, 40, 107, 3, 0, 49, 69, n]; //Set error correction level for QR, n = 48/49/50/51 60 | //[29 40 107 3 0 49 69 n] 61 | public static STORE_QR =(pL, pH): number[] => [29, 40, 107, pL, pH, 49, 80, 48]; //Store QR data in symbol storage area 62 | //[29 40 107 pL pH 49 80 48 d1…dk] 63 | public static PRINT_QR =(): number[] => [29, 40, 107, 3, 0, 49, 81, 48]; //Print QR from data in symbol storage area 64 | //[29 40 107 3 0 49 81 m] 65 | } 66 | -------------------------------------------------------------------------------- /src/escpos.ts: -------------------------------------------------------------------------------- 1 | import { TemplateParser } from './template-parser'; 2 | import { XMLParser } from './xml-parser'; 3 | import { BufferBuilder } from './buffer-builder'; 4 | 5 | export class EscPos { 6 | 7 | public static getBufferFromTemplate(template: string, data: any): number[] { 8 | let templateParser = new TemplateParser(); 9 | return templateParser.parser(template, data).build(); 10 | } 11 | 12 | public static getBufferFromXML(xml: string): number[] { 13 | let xmlParser = new XMLParser(); 14 | return xmlParser.parser(xml).build(); 15 | } 16 | 17 | public static getBufferBuilder(): BufferBuilder { 18 | return new BufferBuilder(); 19 | } 20 | 21 | } 22 | -------------------------------------------------------------------------------- /src/image.ts: -------------------------------------------------------------------------------- 1 | class Image { 2 | public pixels; 3 | public data; 4 | constructor(pixels) { 5 | this.pixels = pixels; 6 | 7 | this.data = []; 8 | function rgb(pixel) { 9 | return { 10 | r: pixel[0], 11 | g: pixel[1], 12 | b: pixel[2], 13 | a: pixel[3], 14 | }; 15 | } 16 | 17 | var self = this; 18 | for (var i = 0; i < this.pixels.data.length; i += this.size.colors) { 19 | this.data.push( 20 | rgb( 21 | new Array(this.size.colors).fill(0).map(function (_, b) { 22 | return self.pixels.data[i + b]; 23 | }) 24 | ) 25 | ); 26 | } 27 | 28 | this.data = this.data.map(function (pixel) { 29 | if (pixel.a == 0) return 0; 30 | var shouldBeWhite = pixel.r > 200 && pixel.g > 200 && pixel.b > 200; 31 | return shouldBeWhite ? 0 : 1; 32 | }); 33 | } 34 | 35 | /** 36 | * [description] 37 | * @return {[type]} [description] 38 | */ 39 | get size() { 40 | return { 41 | width: this.pixels.shape[0], 42 | height: this.pixels.shape[1], 43 | colors: this.pixels.shape[2], 44 | }; 45 | } 46 | 47 | /** 48 | * [toBitmap description] 49 | * @param {[type]} density [description] 50 | * @return {[type]} [description] 51 | */ 52 | public toBitmap(density) { 53 | density = density || 24; 54 | 55 | var ld, 56 | result = []; 57 | var x, y, b, l, i; 58 | var c = density / 8; 59 | 60 | // n blocks of lines 61 | var n = Math.ceil(this.size.height / density); 62 | 63 | for (y = 0; y < n; y++) { 64 | // line data 65 | ld = result[y] = []; 66 | 67 | for (x = 0; x < this.size.width; x++) { 68 | for (b = 0; b < density; b++) { 69 | i = x * c + (b >> 3); 70 | 71 | if (ld[i] === undefined) { 72 | ld[i] = 0; 73 | } 74 | 75 | l = y * density + b; 76 | if (l < this.size.height) { 77 | if (this.data[l * this.size.width + x]) { 78 | ld[i] += 0x80 >> (b & 0x7); 79 | } 80 | } 81 | } 82 | } 83 | } 84 | 85 | return { 86 | data: result, 87 | density: density, 88 | }; 89 | } 90 | 91 | /** 92 | * [toRaster description] 93 | * @return {[type]} [description] 94 | */ 95 | public toRaster() { 96 | var result = []; 97 | var width = this.size.width; 98 | var height = this.size.height; 99 | var data = this.data; 100 | 101 | // n blocks of lines 102 | var n = Math.ceil(width / 8); 103 | var x, y, b, c, i; 104 | 105 | for (y = 0; y < height; y++) { 106 | for (x = 0; x < n; x++) { 107 | for (b = 0; b < 8; b++) { 108 | i = x * 8 + b; 109 | 110 | if (result[y * n + x] === undefined) { 111 | result[y * n + x] = 0; 112 | } 113 | 114 | c = x * 8 + b; 115 | if (c < width) { 116 | if (data[y * width + i]) { 117 | result[y * n + x] += 0x80 >> (b & 0x7); 118 | } 119 | } 120 | } 121 | } 122 | } 123 | return { 124 | data: result, 125 | width: n, 126 | height: height, 127 | }; 128 | } 129 | } 130 | 131 | export default Image; 132 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | global.Buffer = global.Buffer || require('buffer').Buffer; 2 | export * from './template-parser'; 3 | export * from './xml-parser'; 4 | export * from './buffer-builder'; 5 | export * from './escpos'; 6 | -------------------------------------------------------------------------------- /src/node-factory.ts: -------------------------------------------------------------------------------- 1 | import AlignNode from './nodes/align-node'; 2 | import BarcodeNode from './nodes/barcode-node'; 3 | import BoldNode from './nodes/bold-node'; 4 | import BreakLineNode from './nodes/break-line-node'; 5 | import DocumentNode from './nodes/document-node'; 6 | import LineFeedNode from './nodes/line-feed-node'; 7 | import QRcodeNode from './nodes/qrcode-node'; 8 | import SmallNode from './nodes/small-node'; 9 | import TextNode from './nodes/text-node'; 10 | import TextLineNode from './nodes/text-line-node'; 11 | import UnderlineNode from './nodes/underline-node'; 12 | import WhiteModeNode from './nodes/white-mode-node'; 13 | import PaperCutNode from './nodes/paper-cut-node'; 14 | import ImageNode from './nodes/image-node'; 15 | import OpenCashDrawerNode from './nodes/open-cash-drawer-node'; 16 | import PrintModeNode from './nodes/print-mode'; 17 | 18 | export class NodeFactory { 19 | 20 | public static create(nodeType: String, node) { 21 | switch (nodeType) { 22 | case 'align': return new AlignNode(node); 23 | case 'barcode': return new BarcodeNode(node); 24 | case 'bold': return new BoldNode(node); 25 | case 'break-line': return new BreakLineNode(node); 26 | case 'document': return new DocumentNode(node); 27 | case 'line-feed': return new LineFeedNode(node); 28 | case 'qrcode': return new QRcodeNode(node); 29 | case 'small': return new SmallNode(node); 30 | case 'text': return new TextNode(node); 31 | case 'text-line': return new TextLineNode(node); 32 | case 'underline': return new UnderlineNode(node); 33 | case 'white-mode': return new WhiteModeNode(node); 34 | case 'paper-cut': return new PaperCutNode(node); 35 | case 'open-cash-drawer': return new OpenCashDrawerNode(node); 36 | case 'image': return new ImageNode(node); 37 | case 'print-mode': return new PrintModeNode(node); 38 | 39 | default: return null; 40 | } 41 | } 42 | 43 | } 44 | -------------------------------------------------------------------------------- /src/nodes/align-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder, ALIGNMENT } from '../buffer-builder'; 3 | 4 | export default class AlignNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | switch (this.attributes.mode) { 12 | case 'center': 13 | bufferBuilder.startAlign(ALIGNMENT.CENTER); break; 14 | case 'left': 15 | bufferBuilder.startAlign(ALIGNMENT.LEFT); break; 16 | case 'right': 17 | bufferBuilder.startAlign(ALIGNMENT.RIGHT); break; 18 | } 19 | return bufferBuilder; 20 | } 21 | 22 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 23 | bufferBuilder.resetAlign(); 24 | return bufferBuilder; 25 | } 26 | 27 | } 28 | -------------------------------------------------------------------------------- /src/nodes/barcode-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder, BARCODE_SYSTEM, BARCODE_WIDTH, BARCODE_LABEL_FONT, BARCODE_LABEL_POSITION } from '../buffer-builder'; 3 | 4 | export default class BarcodeNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | let system, width, height, labelFont, labelPosition, leftSpacing; 12 | 13 | switch (this.attributes.system) { 14 | case 'UPC_A': 15 | system = BARCODE_SYSTEM.UPC_A; break; 16 | case 'UPC_E': 17 | system = BARCODE_SYSTEM.UPC_E; break; 18 | case 'EAN_13': 19 | system = BARCODE_SYSTEM.EAN_13; break; 20 | case 'EAN_8': 21 | system = BARCODE_SYSTEM.EAN_8; break; 22 | case 'CODE_39': 23 | system = BARCODE_SYSTEM.CODE_39; break; 24 | case 'ITF': 25 | system = BARCODE_SYSTEM.ITF; break; 26 | case 'CODABAR': 27 | system = BARCODE_SYSTEM.CODABAR; break; 28 | case 'CODE_93': 29 | system = BARCODE_SYSTEM.CODE_93; break; 30 | case 'CODE_128': 31 | system = BARCODE_SYSTEM.CODE_128; break; 32 | } 33 | 34 | switch (this.attributes.width) { 35 | case 'DOT_250': 36 | width = BARCODE_WIDTH.DOT_250; break; 37 | case 'DOT_375': 38 | width = BARCODE_WIDTH.DOT_375; break; 39 | case 'DOT_560': 40 | width = BARCODE_WIDTH.DOT_560; break; 41 | case 'DOT_625': 42 | width = BARCODE_WIDTH.DOT_625; break; 43 | case 'DOT_750': 44 | width = BARCODE_WIDTH.DOT_750; break; 45 | default: 46 | width = BARCODE_WIDTH.DOT_375; 47 | } 48 | 49 | switch (this.attributes.labelFont) { 50 | case 'FONT_A': 51 | labelFont = BARCODE_LABEL_FONT.FONT_A; break; 52 | case 'FONT_B': 53 | labelFont = BARCODE_LABEL_FONT.FONT_B; break; 54 | default: 55 | labelFont = BARCODE_LABEL_FONT.FONT_A; 56 | } 57 | 58 | switch (this.attributes.labelPosition) { 59 | case 'NOT_PRINT': 60 | labelPosition = BARCODE_LABEL_POSITION.NOT_PRINT; break; 61 | case 'ABOVE': 62 | labelPosition = BARCODE_LABEL_POSITION.ABOVE; break; 63 | case 'BOTTOM': 64 | labelPosition = BARCODE_LABEL_POSITION.BOTTOM; break; 65 | case 'ABOVE_BOTTOM': 66 | labelPosition = BARCODE_LABEL_POSITION.ABOVE_BOTTOM; break; 67 | default: 68 | labelPosition = BARCODE_LABEL_POSITION.BOTTOM; 69 | } 70 | 71 | if (/\d+/.test(this.attributes.height)) { 72 | height = parseInt(this.attributes.height); 73 | } else { 74 | height = 162; 75 | } 76 | 77 | if (/\d+/.test(this.attributes.leftSpacing)) { 78 | leftSpacing = parseInt(this.attributes.leftSpacing); 79 | } else { 80 | leftSpacing = 0; 81 | } 82 | 83 | if (system && this.content) { 84 | bufferBuilder.printBarcode(this.content, system, width, height, labelFont, labelPosition, leftSpacing); 85 | } 86 | 87 | return bufferBuilder; 88 | } 89 | 90 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 91 | return bufferBuilder; 92 | } 93 | 94 | } 95 | -------------------------------------------------------------------------------- /src/nodes/bold-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class BoldNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | bufferBuilder.startBold(); 12 | return bufferBuilder; 13 | } 14 | 15 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 16 | bufferBuilder.endBold(); 17 | return bufferBuilder; 18 | } 19 | 20 | } 21 | -------------------------------------------------------------------------------- /src/nodes/break-line-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class BreakLineNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | let lines: number = 0; 12 | if (/\d+/.test(this.attributes.lines)) 13 | lines = parseInt(this.attributes.lines) - 1; 14 | 15 | return bufferBuilder.breakLine(lines); 16 | } 17 | 18 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 19 | return bufferBuilder; 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /src/nodes/document-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class DocumentNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | if (this.attributes.reverse) 12 | bufferBuilder.startReverseMode(); 13 | 14 | return bufferBuilder; 15 | } 16 | 17 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 18 | bufferBuilder.endReverseMode(); 19 | return bufferBuilder; 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /src/nodes/image-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder, RASTER_MODE } from '../buffer-builder'; 3 | import ndarray from 'ndarray'; 4 | import Image from '../image'; 5 | import upngjs from 'upng-js'; 6 | 7 | export default class ImageNode extends XMLNode { 8 | constructor(node: any) { 9 | super(node); 10 | } 11 | 12 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 13 | const img_data = upngjs.decode( 14 | Buffer.from(this.content.replace(///g, '/'), 'base64'), 15 | ); 16 | 17 | const pixels = ndarray( 18 | new Uint8Array(img_data.data), 19 | [img_data.width | 0, img_data.height | 0, 4], 20 | [4, (4 * img_data.width) | 0, 1], 21 | 0, 22 | ); 23 | 24 | let mode; 25 | switch (this.attributes.mode) { 26 | case 'NORMAL': 27 | mode = RASTER_MODE.NORMAL; 28 | break; 29 | case 'DW': 30 | mode = RASTER_MODE.DOUBLE_WIDTH; 31 | break; 32 | case 'DH': 33 | mode = RASTER_MODE.DOUBLE_HEIGHT; 34 | break; 35 | case 'DWH': 36 | mode = RASTER_MODE.DOUBLE_WIDTH_HEIGHT; 37 | break; 38 | default: 39 | mode = RASTER_MODE.NORMAL; 40 | } 41 | 42 | bufferBuilder.printImage(new Image(pixels), mode); 43 | return bufferBuilder; 44 | } 45 | 46 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 47 | return bufferBuilder; 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /src/nodes/line-feed-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class LineFeedNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | return bufferBuilder.lineFeed(); 12 | } 13 | 14 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 15 | return bufferBuilder; 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /src/nodes/open-cash-drawer-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class OpenCashDrawerNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | return bufferBuilder.openCashDrawer(); 12 | } 13 | 14 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 15 | return bufferBuilder; 16 | } 17 | 18 | } -------------------------------------------------------------------------------- /src/nodes/paper-cut-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class PaperCutNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | return bufferBuilder.paperCut(); 12 | } 13 | 14 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 15 | return bufferBuilder; 16 | } 17 | 18 | } -------------------------------------------------------------------------------- /src/nodes/print-mode.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class PrintModeNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | if(this.attributes.mode === 'U220') 12 | { 13 | bufferBuilder.setPrintMode(false); 14 | } 15 | else 16 | { 17 | bufferBuilder.setPrintMode(true); 18 | } 19 | return bufferBuilder; 20 | } 21 | 22 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 23 | bufferBuilder.resetAlign(); 24 | return bufferBuilder; 25 | } 26 | 27 | } 28 | -------------------------------------------------------------------------------- /src/nodes/qrcode-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder, QR_EC_LEVEL } from '../buffer-builder'; 3 | 4 | export default class QRcodeNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | let qrModel, qrSize, ecLevel; 12 | 13 | switch (this.attributes.model) { 14 | case '1': 15 | qrModel = 49; break; 16 | case '2': 17 | qrModel = 50; break; 18 | default: 19 | qrModel = 50; 20 | } 21 | 22 | if (/\d+/.test(this.attributes.size)) { 23 | qrSize = parseInt(this.attributes.size); 24 | } 25 | else { 26 | qrSize = 8; 27 | } 28 | 29 | switch (this.attributes.ecl) { 30 | case 'L': 31 | ecLevel = 48; break; 32 | case 'M': 33 | ecLevel = 49; break; 34 | case 'Q': 35 | ecLevel = 50; break; 36 | case 'H': 37 | ecLevel = 51; break; 38 | default: 39 | ecLevel = 48; 40 | } 41 | 42 | if (this.content) { 43 | let url = this.content.replace(/ /g, ' ').replace(/&/g, '&').replace(/=/g, '=').replace(///g, '/').replace(/</g, '<').replace(/>/g, '>').replace(/'/g, "'").replace(/"/g, '"') 44 | bufferBuilder.printQRcode(url, qrModel, qrSize, ecLevel); 45 | } 46 | 47 | return bufferBuilder; 48 | } 49 | 50 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 51 | return bufferBuilder; 52 | } 53 | 54 | } 55 | -------------------------------------------------------------------------------- /src/nodes/small-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class SmallNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | bufferBuilder.startCompressedCharacter(); 12 | return bufferBuilder; 13 | } 14 | 15 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 16 | bufferBuilder.endCompressedCharacter(); 17 | return bufferBuilder; 18 | } 19 | 20 | } 21 | -------------------------------------------------------------------------------- /src/nodes/text-line-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | import TextNode from './text-node'; 4 | 5 | export default class TextLineNode extends XMLNode { 6 | 7 | private textNode: TextNode; 8 | 9 | constructor(node: any) { 10 | super(node); 11 | this.textNode = new TextNode(node); 12 | } 13 | 14 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 15 | return this.textNode.open(bufferBuilder); 16 | } 17 | 18 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 19 | return this.textNode.close(bufferBuilder).breakLine(); 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /src/nodes/text-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class TextNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | 12 | if (/\d+:\d+/.test(this.attributes.size)) { 13 | let size: number[] = new String(this.attributes.size).split(':').map(entry => parseInt(entry)); 14 | bufferBuilder.setCharacterSize(size[0], size[1]); 15 | } 16 | 17 | bufferBuilder.setPrintColor(this.attributes.color); 18 | 19 | let text = this.getContent().replace(/ /g, ' ').replace(/<tab>/g, ' ').replace(/&/g, '&').replace(/=/g, '=').replace(///g, '/').replace(/</g, '<').replace(/>/g, '>').replace(/'/g, "'").replace(/"/g, '"'); 20 | 21 | bufferBuilder.printText(text); 22 | return bufferBuilder; 23 | } 24 | 25 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 26 | bufferBuilder.resetCharacterSize(); 27 | return bufferBuilder; 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /src/nodes/underline-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder, UNDERLINE_MODE } from '../buffer-builder'; 3 | 4 | export default class UnderlineNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | switch (this.attributes.mode) { 12 | case 'one-point': 13 | bufferBuilder.startUnderline(UNDERLINE_MODE.ONE_POINT_OF_COARSE); break; 14 | case 'two-points': 15 | bufferBuilder.startUnderline(UNDERLINE_MODE.TWO_POINTS_OF_COARSE); break; 16 | default: 17 | bufferBuilder.startUnderline(); 18 | } 19 | return bufferBuilder; 20 | } 21 | 22 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 23 | bufferBuilder.endUnderline(); 24 | return bufferBuilder; 25 | } 26 | 27 | } 28 | -------------------------------------------------------------------------------- /src/nodes/white-mode-node.ts: -------------------------------------------------------------------------------- 1 | import { XMLNode } from '../xml-node'; 2 | import { BufferBuilder } from '../buffer-builder'; 3 | 4 | export default class WhiteModeNode extends XMLNode { 5 | 6 | constructor(node: any) { 7 | super(node); 8 | } 9 | 10 | public open(bufferBuilder: BufferBuilder): BufferBuilder { 11 | bufferBuilder.startWhiteMode(); 12 | return bufferBuilder; 13 | } 14 | 15 | public close(bufferBuilder: BufferBuilder): BufferBuilder { 16 | bufferBuilder.endWhiteMode(); 17 | return bufferBuilder; 18 | } 19 | 20 | } 21 | -------------------------------------------------------------------------------- /src/template-parser.ts: -------------------------------------------------------------------------------- 1 | import Mustache from 'mustache'; 2 | import { XMLParser } from './xml-parser'; 3 | import { BufferBuilder } from './buffer-builder'; 4 | 5 | export class TemplateParser { 6 | private mustache: any; 7 | 8 | constructor() { 9 | this.mustache = Mustache; 10 | } 11 | 12 | public parser(template, scope): BufferBuilder { 13 | const xml = this.mustache.render(template, scope); 14 | return new XMLParser().parser(xml); 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /src/xml-node.ts: -------------------------------------------------------------------------------- 1 | import { BufferBuilder } from './buffer-builder'; 2 | 3 | export abstract class XMLNode { 4 | 5 | protected attributes: any; 6 | protected content: string; 7 | protected children: XMLNode[]; 8 | 9 | constructor(node: any) { 10 | this.attributes = node.attributes || {}; 11 | this.content = node.content; 12 | this.children = []; 13 | } 14 | 15 | public addChild(child: XMLNode) { 16 | if (child) 17 | this.children.push(child); 18 | } 19 | 20 | protected getContent(): string { 21 | return this.content; 22 | } 23 | 24 | public abstract open(bufferBuilder: BufferBuilder): BufferBuilder | Promise; 25 | 26 | public abstract close(bufferBuilder: BufferBuilder): BufferBuilder; 27 | 28 | public draw(bufferBuilder: BufferBuilder): BufferBuilder { 29 | 30 | // open tag 31 | this.open(bufferBuilder); 32 | 33 | if (this.children.length > 0) { 34 | this.children.forEach(child => child.draw(bufferBuilder)); 35 | } 36 | 37 | // close tag 38 | this.close(bufferBuilder); 39 | 40 | return bufferBuilder; 41 | } 42 | 43 | } 44 | -------------------------------------------------------------------------------- /src/xml-parser.ts: -------------------------------------------------------------------------------- 1 | import parser from 'xml-parser'; 2 | import { BufferBuilder } from './buffer-builder'; 3 | import { XMLNode } from './xml-node'; 4 | import { NodeFactory } from './node-factory'; 5 | 6 | export class XMLParser { 7 | 8 | public parser(xml: string): BufferBuilder { 9 | let parsedXML = parser(xml); 10 | return this.compile(parsedXML); 11 | } 12 | 13 | private compile(parsedXML: any): BufferBuilder { 14 | let bufferBuilder = new BufferBuilder(); 15 | let rootNode = this.adapter(parsedXML.root, null); 16 | return rootNode.draw(bufferBuilder); 17 | } 18 | 19 | private adapter(node: any, parentNode): XMLNode { 20 | let xmlNode: XMLNode = NodeFactory.create(node.name, node); 21 | if (parentNode) parentNode.addChild(xmlNode); 22 | if (node.children.length > 0) { 23 | node.children.forEach(child => { 24 | this.adapter(child, xmlNode); 25 | }); 26 | } 27 | return xmlNode; 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /tsconfig.build.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "module": "commonjs", 4 | "target": "ES2019", 5 | "outDir": "lib", 6 | "declaration": true, 7 | "esModuleInterop": true, 8 | "moduleResolution": "node", 9 | "emitDecoratorMetadata": true, 10 | "experimentalDecorators": true, 11 | "resolveJsonModule": true, 12 | "skipLibCheck": true 13 | }, 14 | 15 | "exclude": ["node_modules", "dist", "examples"] 16 | } 17 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "./tsconfig.build.json", 3 | 4 | "compilerOptions": { 5 | "baseUrl": ".", 6 | "esModuleInterop": true, 7 | "resolveJsonModule": true 8 | } 9 | } 10 | -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "rules": { 3 | "no-duplicate-variable": true, 4 | "no-unused-variable": [ 5 | true 6 | ] 7 | }, 8 | "rulesDirectory": [ 9 | "node_modules/tslint-eslint-rules/dist/rules" 10 | ] 11 | } 12 | -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@babel/code-frame@^7.0.0": 6 | version "7.12.11" 7 | resolved "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.12.11.tgz" 8 | integrity sha512-Zt1yodBx1UcyiePMSkWnU4hPqhwq7hGi2nFL1LeA3EUl+q2LQx16MISgJ0+z7dnmgvP9QtIleuETGOiOH1RcIw== 9 | dependencies: 10 | "@babel/highlight" "^7.10.4" 11 | 12 | "@babel/helper-validator-identifier@^7.10.4": 13 | version "7.12.11" 14 | resolved "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.12.11.tgz" 15 | integrity sha512-np/lG3uARFybkoHokJUmf1QfEvRVCPbmQeUQpKow5cQ3xWrV9i3rUHodKDJPQfTVX61qKi+UdYk8kik84n7XOw== 16 | 17 | "@babel/highlight@^7.10.4": 18 | version "7.10.4" 19 | resolved "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz" 20 | integrity sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA== 21 | dependencies: 22 | "@babel/helper-validator-identifier" "^7.10.4" 23 | chalk "^2.0.0" 24 | js-tokens "^4.0.0" 25 | 26 | "@types/node@^14.14.20": 27 | version "14.14.20" 28 | resolved "https://registry.npmjs.org/@types/node/-/node-14.14.20.tgz" 29 | integrity sha512-Y93R97Ouif9JEOWPIUyU+eyIdyRqQR0I8Ez1dzku4hDx34NWh4HbtIc3WNzwB1Y9ULvNGeu5B8h8bVL5cAk4/A== 30 | 31 | "@types/xml-parser@^1.2.29": 32 | version "1.2.29" 33 | resolved "https://registry.npmjs.org/@types/xml-parser/-/xml-parser-1.2.29.tgz" 34 | integrity sha512-l5ID65aPDctN/dZYkDgLOEBuoHrD8S9TyfD5soORUtVHKyOs7Wr66iNxAtcmT/tER1GeYqp51jR6l08gmsRcZg== 35 | 36 | ansi-styles@^3.2.1: 37 | version "3.2.1" 38 | resolved "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz" 39 | integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== 40 | dependencies: 41 | color-convert "^1.9.0" 42 | 43 | argparse@^1.0.7: 44 | version "1.0.10" 45 | resolved "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz" 46 | integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg== 47 | dependencies: 48 | sprintf-js "~1.0.2" 49 | 50 | balanced-match@^1.0.0: 51 | version "1.0.0" 52 | resolved "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz" 53 | integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c= 54 | 55 | base64-js@^1.3.1: 56 | version "1.5.1" 57 | resolved "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz" 58 | integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== 59 | 60 | big.js@^5.2.2: 61 | version "5.2.2" 62 | resolved "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz" 63 | integrity sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ== 64 | 65 | brace-expansion@^1.1.7: 66 | version "1.1.11" 67 | resolved "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz" 68 | integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== 69 | dependencies: 70 | balanced-match "^1.0.0" 71 | concat-map "0.0.1" 72 | 73 | buffer@^6.0.3: 74 | version "6.0.3" 75 | resolved "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz" 76 | integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA== 77 | dependencies: 78 | base64-js "^1.3.1" 79 | ieee754 "^1.2.1" 80 | 81 | builtin-modules@^1.1.1: 82 | version "1.1.1" 83 | resolved "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz" 84 | integrity sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8= 85 | 86 | chalk@^2.0.0, chalk@^2.0.1, chalk@^2.3.0: 87 | version "2.4.2" 88 | resolved "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz" 89 | integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== 90 | dependencies: 91 | ansi-styles "^3.2.1" 92 | escape-string-regexp "^1.0.5" 93 | supports-color "^5.3.0" 94 | 95 | color-convert@^1.9.0: 96 | version "1.9.3" 97 | resolved "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz" 98 | integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== 99 | dependencies: 100 | color-name "1.1.3" 101 | 102 | color-name@1.1.3: 103 | version "1.1.3" 104 | resolved "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz" 105 | integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU= 106 | 107 | commander@^2.12.1: 108 | version "2.20.3" 109 | resolved "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz" 110 | integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== 111 | 112 | concat-map@0.0.1: 113 | version "0.0.1" 114 | resolved "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz" 115 | integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= 116 | 117 | core-util-is@~1.0.0: 118 | version "1.0.2" 119 | resolved "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz" 120 | integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac= 121 | 122 | debug@^2.2.0: 123 | version "2.6.9" 124 | resolved "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz" 125 | integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== 126 | dependencies: 127 | ms "2.0.0" 128 | 129 | diff@^4.0.1: 130 | version "4.0.2" 131 | resolved "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz" 132 | integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== 133 | 134 | doctrine@^0.7.2: 135 | version "0.7.2" 136 | resolved "https://registry.npmjs.org/doctrine/-/doctrine-0.7.2.tgz" 137 | integrity sha1-fLhgNZujvpDgQLJrcpzkv6ZUxSM= 138 | dependencies: 139 | esutils "^1.1.6" 140 | isarray "0.0.1" 141 | 142 | emojis-list@^3.0.0: 143 | version "3.0.0" 144 | resolved "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz" 145 | integrity sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q== 146 | 147 | enhanced-resolve@^3.0.0: 148 | version "3.4.1" 149 | resolved "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-3.4.1.tgz" 150 | integrity sha1-BCHjOf1xQZs9oT0Smzl5BAIwR24= 151 | dependencies: 152 | graceful-fs "^4.1.2" 153 | memory-fs "^0.4.0" 154 | object-assign "^4.0.1" 155 | tapable "^0.2.7" 156 | 157 | errno@^0.1.3: 158 | version "0.1.8" 159 | resolved "https://registry.npmjs.org/errno/-/errno-0.1.8.tgz" 160 | integrity sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A== 161 | dependencies: 162 | prr "~1.0.1" 163 | 164 | escape-string-regexp@^1.0.5: 165 | version "1.0.5" 166 | resolved "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz" 167 | integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ= 168 | 169 | esprima@^4.0.0: 170 | version "4.0.1" 171 | resolved "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz" 172 | integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== 173 | 174 | esutils@^1.1.6: 175 | version "1.1.6" 176 | resolved "https://registry.npmjs.org/esutils/-/esutils-1.1.6.tgz" 177 | integrity sha1-wBzKqa5LiXxtDD4hCuUvPHqEQ3U= 178 | 179 | fs.realpath@^1.0.0: 180 | version "1.0.0" 181 | resolved "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz" 182 | integrity sha1-FQStJSMVjKpA20onh8sBQRmU6k8= 183 | 184 | function-bind@^1.1.1: 185 | version "1.1.1" 186 | resolved "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz" 187 | integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== 188 | 189 | glob@^7.1.1: 190 | version "7.1.6" 191 | resolved "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz" 192 | integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA== 193 | dependencies: 194 | fs.realpath "^1.0.0" 195 | inflight "^1.0.4" 196 | inherits "2" 197 | minimatch "^3.0.4" 198 | once "^1.3.0" 199 | path-is-absolute "^1.0.0" 200 | 201 | graceful-fs@^4.1.2: 202 | version "4.2.4" 203 | resolved "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz" 204 | integrity sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw== 205 | 206 | has-flag@^3.0.0: 207 | version "3.0.0" 208 | resolved "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz" 209 | integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0= 210 | 211 | has@^1.0.3: 212 | version "1.0.3" 213 | resolved "https://registry.npmjs.org/has/-/has-1.0.3.tgz" 214 | integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw== 215 | dependencies: 216 | function-bind "^1.1.1" 217 | 218 | ieee754@^1.2.1: 219 | version "1.2.1" 220 | resolved "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz" 221 | integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== 222 | 223 | inflight@^1.0.4: 224 | version "1.0.6" 225 | resolved "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz" 226 | integrity sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk= 227 | dependencies: 228 | once "^1.3.0" 229 | wrappy "1" 230 | 231 | inherits@2, inherits@~2.0.3: 232 | version "2.0.4" 233 | resolved "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz" 234 | integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== 235 | 236 | iota-array@^1.0.0: 237 | version "1.0.0" 238 | resolved "https://registry.npmjs.org/iota-array/-/iota-array-1.0.0.tgz" 239 | integrity sha1-ge9X/l0FgUzVjCSDYyqZwwoOgIc= 240 | 241 | is-buffer@^1.0.2: 242 | version "1.1.6" 243 | resolved "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz" 244 | integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== 245 | 246 | is-core-module@^2.1.0: 247 | version "2.2.0" 248 | resolved "https://registry.npmjs.org/is-core-module/-/is-core-module-2.2.0.tgz" 249 | integrity sha512-XRAfAdyyY5F5cOXn7hYQDqh2Xmii+DEfIcQGxK/uNwMHhIkPWO0g8msXcbzLe+MpGoR951MlqM/2iIlU4vKDdQ== 250 | dependencies: 251 | has "^1.0.3" 252 | 253 | isarray@0.0.1: 254 | version "0.0.1" 255 | resolved "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz" 256 | integrity sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8= 257 | 258 | isarray@~1.0.0: 259 | version "1.0.0" 260 | resolved "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz" 261 | integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= 262 | 263 | js-tokens@^4.0.0: 264 | version "4.0.0" 265 | resolved "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz" 266 | integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== 267 | 268 | js-yaml@^3.13.1: 269 | version "3.14.1" 270 | resolved "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz" 271 | integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g== 272 | dependencies: 273 | argparse "^1.0.7" 274 | esprima "^4.0.0" 275 | 276 | json5@^1.0.1: 277 | version "1.0.1" 278 | resolved "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz" 279 | integrity sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow== 280 | dependencies: 281 | minimist "^1.2.0" 282 | 283 | loader-utils@^1.0.2: 284 | version "1.4.0" 285 | resolved "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz" 286 | integrity sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA== 287 | dependencies: 288 | big.js "^5.2.2" 289 | emojis-list "^3.0.0" 290 | json5 "^1.0.1" 291 | 292 | memory-fs@^0.4.0: 293 | version "0.4.1" 294 | resolved "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz" 295 | integrity sha1-OpoguEYlI+RHz7x+i7gO1me/xVI= 296 | dependencies: 297 | errno "^0.1.3" 298 | readable-stream "^2.0.1" 299 | 300 | minimatch@^3.0.4: 301 | version "3.0.4" 302 | resolved "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz" 303 | integrity sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA== 304 | dependencies: 305 | brace-expansion "^1.1.7" 306 | 307 | minimist@^1.2.0, minimist@^1.2.5: 308 | version "1.2.5" 309 | resolved "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz" 310 | integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw== 311 | 312 | mkdirp@^0.5.1: 313 | version "0.5.5" 314 | resolved "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz" 315 | integrity sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ== 316 | dependencies: 317 | minimist "^1.2.5" 318 | 319 | ms@2.0.0: 320 | version "2.0.0" 321 | resolved "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz" 322 | integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g= 323 | 324 | mustache@^4.1.0: 325 | version "4.1.0" 326 | resolved "https://registry.npmjs.org/mustache/-/mustache-4.1.0.tgz" 327 | integrity sha512-0FsgP/WVq4mKyjolIyX+Z9Bd+3WS8GOwoUTyKXT5cTYMGeauNTi2HPCwERqseC1IHAy0Z7MDZnJBfjabd4O8GQ== 328 | 329 | mutable-buffer@2.0.3: 330 | version "2.0.3" 331 | resolved "https://registry.npmjs.org/mutable-buffer/-/mutable-buffer-2.0.3.tgz" 332 | integrity sha1-Z7ylVovxXT9gNftxbjiZ3ESQHIw= 333 | 334 | ndarray@^1.0.19: 335 | version "1.0.19" 336 | resolved "https://registry.npmjs.org/ndarray/-/ndarray-1.0.19.tgz" 337 | integrity sha512-B4JHA4vdyZU30ELBw3g7/p9bZupyew5a7tX1Y/gGeF2hafrPaQZhgrGQfsvgfYbgdFZjYwuEcnaobeM/WMW+HQ== 338 | dependencies: 339 | iota-array "^1.0.0" 340 | is-buffer "^1.0.2" 341 | 342 | object-assign@^4.0.1: 343 | version "4.1.1" 344 | resolved "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz" 345 | integrity sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM= 346 | 347 | once@^1.3.0: 348 | version "1.4.0" 349 | resolved "https://registry.npmjs.org/once/-/once-1.4.0.tgz" 350 | integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E= 351 | dependencies: 352 | wrappy "1" 353 | 354 | pako@^1.0.5: 355 | version "1.0.11" 356 | resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf" 357 | integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw== 358 | 359 | path-is-absolute@^1.0.0: 360 | version "1.0.1" 361 | resolved "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz" 362 | integrity sha1-F0uSaHNVNP+8es5r9TpanhtcX18= 363 | 364 | path-parse@^1.0.6: 365 | version "1.0.6" 366 | resolved "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz" 367 | integrity sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw== 368 | 369 | process-nextick-args@~2.0.0: 370 | version "2.0.1" 371 | resolved "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz" 372 | integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== 373 | 374 | prr@~1.0.1: 375 | version "1.0.1" 376 | resolved "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz" 377 | integrity sha1-0/wRS6BplaRexok/SEzrHXj19HY= 378 | 379 | readable-stream@^2.0.1: 380 | version "2.3.7" 381 | resolved "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz" 382 | integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw== 383 | dependencies: 384 | core-util-is "~1.0.0" 385 | inherits "~2.0.3" 386 | isarray "~1.0.0" 387 | process-nextick-args "~2.0.0" 388 | safe-buffer "~5.1.1" 389 | string_decoder "~1.1.1" 390 | util-deprecate "~1.0.1" 391 | 392 | resolve@^1.3.2: 393 | version "1.19.0" 394 | resolved "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz" 395 | integrity sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg== 396 | dependencies: 397 | is-core-module "^2.1.0" 398 | path-parse "^1.0.6" 399 | 400 | safe-buffer@~5.1.0, safe-buffer@~5.1.1: 401 | version "5.1.2" 402 | resolved "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz" 403 | integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== 404 | 405 | semver@^5.0.1, semver@^5.3.0: 406 | version "5.7.1" 407 | resolved "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz" 408 | integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== 409 | 410 | sprintf-js@~1.0.2: 411 | version "1.0.3" 412 | resolved "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz" 413 | integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw= 414 | 415 | string_decoder@~1.1.1: 416 | version "1.1.1" 417 | resolved "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz" 418 | integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== 419 | dependencies: 420 | safe-buffer "~5.1.0" 421 | 422 | supports-color@^5.3.0: 423 | version "5.5.0" 424 | resolved "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz" 425 | integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== 426 | dependencies: 427 | has-flag "^3.0.0" 428 | 429 | tapable@^0.2.7: 430 | version "0.2.9" 431 | resolved "https://registry.npmjs.org/tapable/-/tapable-0.2.9.tgz" 432 | integrity sha512-2wsvQ+4GwBvLPLWsNfLCDYGsW6xb7aeC6utq2Qh0PFwgEy7K7dsma9Jsmb2zSQj7GvYAyUGSntLtsv++GmgL1A== 433 | 434 | ts-loader@^2.3.4: 435 | version "2.3.7" 436 | resolved "https://registry.npmjs.org/ts-loader/-/ts-loader-2.3.7.tgz" 437 | integrity sha512-8t3bu2FcEkXb+D4L+Cn8qiK2E2C6Ms4/GQChvz6IMbVurcFHLXrhW4EMtfaol1a1ASQACZGDUGit4NHnX9g7hQ== 438 | dependencies: 439 | chalk "^2.0.1" 440 | enhanced-resolve "^3.0.0" 441 | loader-utils "^1.0.2" 442 | semver "^5.0.1" 443 | 444 | tslib@^1.0.0, tslib@^1.8.0, tslib@^1.8.1: 445 | version "1.14.1" 446 | resolved "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz" 447 | integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg== 448 | 449 | tslint-eslint-rules@^4.1.1: 450 | version "4.1.1" 451 | resolved "https://registry.npmjs.org/tslint-eslint-rules/-/tslint-eslint-rules-4.1.1.tgz" 452 | integrity sha1-fDDniC8mvCdr/5HSOEl1xp2viLo= 453 | dependencies: 454 | doctrine "^0.7.2" 455 | tslib "^1.0.0" 456 | tsutils "^1.4.0" 457 | 458 | tslint@^5.7.0: 459 | version "5.20.1" 460 | resolved "https://registry.npmjs.org/tslint/-/tslint-5.20.1.tgz" 461 | integrity sha512-EcMxhzCFt8k+/UP5r8waCf/lzmeSyVlqxqMEDQE7rWYiQky8KpIBz1JAoYXfROHrPZ1XXd43q8yQnULOLiBRQg== 462 | dependencies: 463 | "@babel/code-frame" "^7.0.0" 464 | builtin-modules "^1.1.1" 465 | chalk "^2.3.0" 466 | commander "^2.12.1" 467 | diff "^4.0.1" 468 | glob "^7.1.1" 469 | js-yaml "^3.13.1" 470 | minimatch "^3.0.4" 471 | mkdirp "^0.5.1" 472 | resolve "^1.3.2" 473 | semver "^5.3.0" 474 | tslib "^1.8.0" 475 | tsutils "^2.29.0" 476 | 477 | tsutils@^1.4.0: 478 | version "1.9.1" 479 | resolved "https://registry.npmjs.org/tsutils/-/tsutils-1.9.1.tgz" 480 | integrity sha1-ufmrROVa+WgYMdXyjQrur1x1DLA= 481 | 482 | tsutils@^2.29.0: 483 | version "2.29.0" 484 | resolved "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz" 485 | integrity sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA== 486 | dependencies: 487 | tslib "^1.8.1" 488 | 489 | typescript@^4.1.3: 490 | version "4.1.3" 491 | resolved "https://registry.npmjs.org/typescript/-/typescript-4.1.3.tgz" 492 | integrity sha512-B3ZIOf1IKeH2ixgHhj6la6xdwR9QrLC5d1VKeCSY4tvkqhF2eqd9O7txNlS0PO3GrBAFIdr3L1ndNwteUbZLYg== 493 | 494 | upng-js@2.1.0: 495 | version "2.1.0" 496 | resolved "https://registry.yarnpkg.com/upng-js/-/upng-js-2.1.0.tgz#7176e73973db361ca95d0fa14f958385db6b9dd2" 497 | integrity sha512-d3xzZzpMP64YkjP5pr8gNyvBt7dLk/uGI67EctzDuVp4lCZyVMo0aJO6l/VDlgbInJYDY6cnClLoBp29eKWI6g== 498 | dependencies: 499 | pako "^1.0.5" 500 | 501 | util-deprecate@~1.0.1: 502 | version "1.0.2" 503 | resolved "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz" 504 | integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8= 505 | 506 | wrappy@1: 507 | version "1.0.2" 508 | resolved "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz" 509 | integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= 510 | 511 | xml-parser@^1.2.1: 512 | version "1.2.1" 513 | resolved "https://registry.npmjs.org/xml-parser/-/xml-parser-1.2.1.tgz" 514 | integrity sha1-wx9MNPKXXbgq0BMiISBZJzYVb80= 515 | dependencies: 516 | debug "^2.2.0" 517 | --------------------------------------------------------------------------------