├── .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 |
--------------------------------------------------------------------------------