pdfmake-acroforms 0.3.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/CHANGELOG.md +60 -0
  2. package/LICENSE +21 -0
  3. package/README.md +75 -0
  4. package/build/fonts/Roboto/Roboto-Italic.ttf +0 -0
  5. package/build/fonts/Roboto/Roboto-Medium.ttf +0 -0
  6. package/build/fonts/Roboto/Roboto-MediumItalic.ttf +0 -0
  7. package/build/fonts/Roboto/Roboto-Regular.ttf +0 -0
  8. package/build/fonts/Roboto.js +27 -0
  9. package/build/pdfmake.js +67806 -0
  10. package/build/pdfmake.js.map +1 -0
  11. package/build/pdfmake.min.js +3 -0
  12. package/build/pdfmake.min.js.map +1 -0
  13. package/build/standard-fonts/Courier.js +27 -0
  14. package/build/standard-fonts/Helvetica.js +27 -0
  15. package/build/standard-fonts/Symbol.js +21 -0
  16. package/build/standard-fonts/Times.js +27 -0
  17. package/build/standard-fonts/ZapfDingbats.js +21 -0
  18. package/build/vfs_fonts.js +6 -0
  19. package/build-vfs.js +44 -0
  20. package/fonts/Roboto/Roboto-Italic.ttf +0 -0
  21. package/fonts/Roboto/Roboto-Medium.ttf +0 -0
  22. package/fonts/Roboto/Roboto-MediumItalic.ttf +0 -0
  23. package/fonts/Roboto/Roboto-Regular.ttf +0 -0
  24. package/fonts/Roboto.js +8 -0
  25. package/js/3rd-party/svg-to-pdfkit/source.js +3823 -0
  26. package/js/3rd-party/svg-to-pdfkit.js +7 -0
  27. package/js/DocMeasure.js +675 -0
  28. package/js/DocPreprocessor.js +258 -0
  29. package/js/DocumentContext.js +310 -0
  30. package/js/ElementWriter.js +399 -0
  31. package/js/LayoutBuilder.js +1202 -0
  32. package/js/Line.js +101 -0
  33. package/js/OutputDocument.js +64 -0
  34. package/js/OutputDocumentServer.js +29 -0
  35. package/js/PDFDocument.js +145 -0
  36. package/js/PageElementWriter.js +164 -0
  37. package/js/PageSize.js +74 -0
  38. package/js/Printer.js +288 -0
  39. package/js/Renderer.js +513 -0
  40. package/js/SVGMeasure.js +92 -0
  41. package/js/StyleContextStack.js +191 -0
  42. package/js/TableProcessor.js +562 -0
  43. package/js/TextBreaker.js +179 -0
  44. package/js/TextDecorator.js +152 -0
  45. package/js/TextInlines.js +212 -0
  46. package/js/URLResolver.js +43 -0
  47. package/js/base.js +59 -0
  48. package/js/browser-extensions/OutputDocumentBrowser.js +82 -0
  49. package/js/browser-extensions/fonts/Roboto.js +38 -0
  50. package/js/browser-extensions/index.js +53 -0
  51. package/js/browser-extensions/pdfMake.js +3 -0
  52. package/js/browser-extensions/standard-fonts/Courier.js +38 -0
  53. package/js/browser-extensions/standard-fonts/Helvetica.js +38 -0
  54. package/js/browser-extensions/standard-fonts/Symbol.js +23 -0
  55. package/js/browser-extensions/standard-fonts/Times.js +38 -0
  56. package/js/browser-extensions/standard-fonts/ZapfDingbats.js +23 -0
  57. package/js/browser-extensions/virtual-fs-cjs.js +3 -0
  58. package/js/columnCalculator.js +148 -0
  59. package/js/helpers/node.js +122 -0
  60. package/js/helpers/tools.js +46 -0
  61. package/js/helpers/variableType.js +59 -0
  62. package/js/index.js +15 -0
  63. package/js/qrEnc.js +721 -0
  64. package/js/standardPageSizes.js +56 -0
  65. package/js/tableLayouts.js +98 -0
  66. package/js/virtual-fs.js +60 -0
  67. package/package.json +94 -0
  68. package/src/3rd-party/svg-to-pdfkit/LICENSE +9 -0
  69. package/src/3rd-party/svg-to-pdfkit/source.js +2745 -0
  70. package/src/3rd-party/svg-to-pdfkit.js +3 -0
  71. package/src/DocMeasure.js +768 -0
  72. package/src/DocPreprocessor.js +289 -0
  73. package/src/DocumentContext.js +345 -0
  74. package/src/ElementWriter.js +468 -0
  75. package/src/LayoutBuilder.js +1366 -0
  76. package/src/Line.js +108 -0
  77. package/src/OutputDocument.js +64 -0
  78. package/src/OutputDocumentServer.js +32 -0
  79. package/src/PDFDocument.js +178 -0
  80. package/src/PageElementWriter.js +191 -0
  81. package/src/PageSize.js +53 -0
  82. package/src/Printer.js +306 -0
  83. package/src/Renderer.js +546 -0
  84. package/src/SVGMeasure.js +109 -0
  85. package/src/StyleContextStack.js +208 -0
  86. package/src/TableProcessor.js +620 -0
  87. package/src/TextBreaker.js +181 -0
  88. package/src/TextDecorator.js +175 -0
  89. package/src/TextInlines.js +229 -0
  90. package/src/URLResolver.js +43 -0
  91. package/src/base.js +70 -0
  92. package/src/browser-extensions/OutputDocumentBrowser.js +80 -0
  93. package/src/browser-extensions/fonts/Roboto.js +27 -0
  94. package/src/browser-extensions/index.js +55 -0
  95. package/src/browser-extensions/pdfMake.js +1 -0
  96. package/src/browser-extensions/standard-fonts/Courier.js +27 -0
  97. package/src/browser-extensions/standard-fonts/Helvetica.js +27 -0
  98. package/src/browser-extensions/standard-fonts/Symbol.js +21 -0
  99. package/src/browser-extensions/standard-fonts/Times.js +27 -0
  100. package/src/browser-extensions/standard-fonts/ZapfDingbats.js +21 -0
  101. package/src/browser-extensions/virtual-fs-cjs.js +1 -0
  102. package/src/columnCalculator.js +154 -0
  103. package/src/helpers/node.js +134 -0
  104. package/src/helpers/tools.js +44 -0
  105. package/src/helpers/variableType.js +50 -0
  106. package/src/index.js +16 -0
  107. package/src/qrEnc.js +796 -0
  108. package/src/standardPageSizes.js +52 -0
  109. package/src/tableLayouts.js +100 -0
  110. package/src/virtual-fs.js +66 -0
  111. package/standard-fonts/Courier.js +8 -0
  112. package/standard-fonts/Helvetica.js +8 -0
  113. package/standard-fonts/Symbol.js +5 -0
  114. package/standard-fonts/Times.js +8 -0
  115. package/standard-fonts/ZapfDingbats.js +5 -0
@@ -0,0 +1,181 @@
1
+ import LineBreaker from 'linebreak';
2
+ import { isObject } from './helpers/variableType';
3
+ import StyleContextStack from './StyleContextStack';
4
+
5
+ /**
6
+ * @param {string} text
7
+ * @param {boolean} noWrap
8
+ * @param {boolean} breakAll
9
+ * @returns {Array}
10
+ */
11
+ const splitWords = (text, noWrap, breakAll = false) => {
12
+ let words = [];
13
+ if (text === undefined || text === null) {
14
+ text = '';
15
+ } else {
16
+ text = String(text);
17
+ }
18
+
19
+ if (noWrap) {
20
+ words.push({ text: text });
21
+ return words;
22
+ }
23
+ if (breakAll) {
24
+ return text.split('').map(c => {
25
+ if(c.match(/^\n$|^\r$/)) { // new line
26
+ return { text: '', lineEnd: true };
27
+ }
28
+ return { text: c };
29
+ });
30
+ }
31
+
32
+ if (text.length === 0) {
33
+ words.push({ text: '' });
34
+ return words;
35
+ }
36
+
37
+ let breaker = new LineBreaker(text);
38
+ let last = 0;
39
+ let bk;
40
+
41
+ while ((bk = breaker.nextBreak())) {
42
+ let word = text.slice(last, bk.position);
43
+
44
+ if (bk.required || word.match(/\r?\n$|\r$/)) { // new line
45
+ word = word.replace(/\r?\n$|\r$/, '');
46
+ words.push({ text: word, lineEnd: true });
47
+ } else {
48
+ words.push({ text: word });
49
+ }
50
+
51
+ last = bk.position;
52
+ }
53
+
54
+ return words;
55
+ };
56
+
57
+ /**
58
+ * @param {Array} words
59
+ * @param {boolean} noWrap
60
+ * @returns {?string}
61
+ */
62
+ const getFirstWord = (words, noWrap) => {
63
+ let word = words[0];
64
+ if (word === undefined) {
65
+ return null;
66
+ }
67
+
68
+ if (noWrap) { // text was not wrapped, we need only first word
69
+ let tmpWords = splitWords(word.text, false);
70
+ if (tmpWords[0] === undefined) {
71
+ return null;
72
+ }
73
+ word = tmpWords[0];
74
+ }
75
+
76
+ return word.text;
77
+ };
78
+
79
+ /**
80
+ * @param {Array} words
81
+ * @param {boolean} noWrap
82
+ * @returns {?string}
83
+ */
84
+ const getLastWord = (words, noWrap) => {
85
+ let word = words[words.length - 1];
86
+ if (word === undefined) {
87
+ return null;
88
+ }
89
+
90
+ if (word.lineEnd) {
91
+ return null;
92
+ }
93
+
94
+ if (noWrap) { // text was not wrapped, we need only last word
95
+ let tmpWords = splitWords(word.text, false);
96
+ if (tmpWords[tmpWords.length - 1] === undefined) {
97
+ return null;
98
+ }
99
+ word = tmpWords[tmpWords.length - 1];
100
+ }
101
+
102
+ return word.text;
103
+ };
104
+
105
+ class TextBreaker {
106
+ /**
107
+ * @param {string|Array} texts
108
+ * @param {StyleContextStack} styleContextStack
109
+ * @returns {Array}
110
+ */
111
+ getBreaks(texts, styleContextStack) {
112
+ let results = [];
113
+
114
+ if (!Array.isArray(texts)) {
115
+ texts = [texts];
116
+ }
117
+
118
+ let lastWord = null;
119
+ for (let i = 0, l = texts.length; i < l; i++) {
120
+ let item = texts[i];
121
+ let style = null;
122
+ let words;
123
+ let breakAll = StyleContextStack.getStyleProperty(item || {}, styleContextStack, 'wordBreak', 'normal') === 'break-all';
124
+ let noWrap = StyleContextStack.getStyleProperty(item || {}, styleContextStack, 'noWrap', false);
125
+ if (isObject(item)) {
126
+ if (item.text) {
127
+ if (item._textRef && item._textRef._textNodeRef.text) {
128
+ item.text = item._textRef._textNodeRef.text;
129
+ }
130
+ words = splitWords(item.text, noWrap, breakAll);
131
+ } else if (item.acroform) {
132
+ words = [item];
133
+ }
134
+ style = StyleContextStack.copyStyle(item);
135
+ } else {
136
+ words = splitWords(item, noWrap, breakAll);
137
+ }
138
+
139
+ if (lastWord && words.length && !lastWord.acroform) {
140
+ let firstWord = getFirstWord(words, noWrap);
141
+
142
+ let wrapWords = splitWords(lastWord + firstWord, false);
143
+ if (wrapWords.length === 1) {
144
+ results[results.length - 1].noNewLine = true;
145
+ }
146
+ }
147
+
148
+ for (let i2 = 0, l2 = words.length; i2 < l2; i2++) {
149
+ let result = {};
150
+ if (words[0].acroform) {
151
+ result = words[0];
152
+ } else {
153
+ result = {
154
+ text: words[i2].text
155
+ };
156
+ }
157
+
158
+ if (words[i2].lineEnd) {
159
+ result.lineEnd = true;
160
+ }
161
+
162
+ StyleContextStack.copyStyle(style, result);
163
+
164
+ results.push(result);
165
+ }
166
+
167
+ lastWord = null;
168
+ if (i + 1 < l) {
169
+ if (words[0].acroform) {
170
+ lastWord = words[0];
171
+ } else {
172
+ lastWord = getLastWord(words, noWrap);
173
+ }
174
+ }
175
+ }
176
+
177
+ return results;
178
+ }
179
+ }
180
+
181
+ export default TextBreaker;
@@ -0,0 +1,175 @@
1
+ import { isNumber } from './helpers/variableType';
2
+
3
+ const groupDecorations = line => {
4
+ let groups = [];
5
+ let currentGroup = null;
6
+ for (let i = 0, l = line.inlines.length; i < l; i++) {
7
+ let inline = line.inlines[i];
8
+ let decoration = inline.decoration;
9
+ if (!decoration) {
10
+ currentGroup = null;
11
+ continue;
12
+ }
13
+ if (!Array.isArray(decoration)) {
14
+ decoration = [decoration];
15
+ }
16
+ let color = inline.decorationColor || inline.color || 'black';
17
+ let style = inline.decorationStyle || 'solid';
18
+ let thickness = isNumber(inline.decorationThickness) ? inline.decorationThickness : null;
19
+ for (let ii = 0, ll = decoration.length; ii < ll; ii++) {
20
+ let decorationItem = decoration[ii];
21
+ if (!currentGroup || decorationItem !== currentGroup.decoration ||
22
+ style !== currentGroup.decorationStyle || color !== currentGroup.decorationColor) {
23
+
24
+ currentGroup = {
25
+ line: line,
26
+ decoration: decorationItem,
27
+ decorationColor: color,
28
+ decorationStyle: style,
29
+ decorationThickness: thickness,
30
+ inlines: [inline]
31
+ };
32
+ groups.push(currentGroup);
33
+ } else {
34
+ currentGroup.inlines.push(inline);
35
+ }
36
+ }
37
+ }
38
+
39
+ return groups;
40
+ };
41
+
42
+ class TextDecorator {
43
+
44
+ constructor(pdfDocument) {
45
+ this.pdfDocument = pdfDocument;
46
+ }
47
+
48
+ drawBackground(line, x, y) {
49
+ let height = line.getHeight();
50
+ for (let i = 0, l = line.inlines.length; i < l; i++) {
51
+ let inline = line.inlines[i];
52
+ if (!inline.background) {
53
+ continue;
54
+ }
55
+
56
+ let color = inline.background;
57
+ let patternColor = this.pdfDocument.providePattern(inline.background);
58
+ if (patternColor !== null) {
59
+ color = patternColor;
60
+ }
61
+
62
+ let justifyShift = (inline.justifyShift || 0);
63
+ this.pdfDocument.fillColor(color)
64
+ .rect(x + inline.x - justifyShift, y, inline.width + justifyShift, height)
65
+ .fill();
66
+ }
67
+ }
68
+
69
+ drawDecorations(line, x, y) {
70
+ let groups = groupDecorations(line);
71
+ for (let i = 0, l = groups.length; i < l; i++) {
72
+ this._drawDecoration(groups[i], x, y);
73
+ }
74
+ }
75
+
76
+ _drawDecoration(group, x, y) {
77
+ const maxInline = () => {
78
+ let max = 0;
79
+ for (let i = 0, l = group.inlines.length; i < l; i++) {
80
+ let inline = group.inlines[i];
81
+ max = inline.fontSize > max ? i : max;
82
+ }
83
+ return group.inlines[max];
84
+ };
85
+
86
+ const width = () => {
87
+ let sum = 0;
88
+ for (let i = 0, l = group.inlines.length; i < l; i++) {
89
+ let justifyShift = (group.inlines[i].justifyShift || 0);
90
+ sum += group.inlines[i].width + justifyShift;
91
+ }
92
+ return sum;
93
+ };
94
+
95
+ let firstInline = group.inlines[0];
96
+ let biggerInline = maxInline();
97
+ let totalWidth = width();
98
+ let lineAscent = group.line.getAscenderHeight();
99
+ let ascent = biggerInline.font.ascender / 1000 * biggerInline.fontSize;
100
+ let height = biggerInline.height;
101
+ let descent = height - ascent;
102
+
103
+ let lw = isNumber(group.decorationThickness)
104
+ ? group.decorationThickness
105
+ : 0.5 + Math.floor(Math.max(biggerInline.fontSize - 8, 0) / 2) * 0.12;
106
+
107
+ switch (group.decoration) {
108
+ case 'underline':
109
+ y += lineAscent + descent * 0.45;
110
+ break;
111
+ case 'overline':
112
+ y += lineAscent - (ascent * 0.85);
113
+ break;
114
+ case 'lineThrough':
115
+ y += lineAscent - (ascent * 0.25);
116
+ break;
117
+ default:
118
+ throw new Error(`Unknown decoration : ${group.decoration}`);
119
+ }
120
+
121
+ if (group.inlines[0].sup) {
122
+ y -= group.inlines[0].fontSize * 0.75;
123
+ }
124
+ if (group.inlines[0].sub) {
125
+ y += group.inlines[0].fontSize * 0.35;
126
+ }
127
+
128
+ this.pdfDocument.save();
129
+
130
+ if (group.decorationStyle === 'double') {
131
+ let gap = Math.max(0.5, lw, biggerInline.fontSize * 0.15);
132
+ this.pdfDocument.fillColor(group.decorationColor)
133
+ .rect(x + firstInline.x, y - lw / 2, totalWidth, lw / 2).fill()
134
+ .rect(x + firstInline.x, y + gap - lw / 2, totalWidth, lw / 2).fill();
135
+ } else if (group.decorationStyle === 'dashed') {
136
+ let nbDashes = Math.ceil(totalWidth / (3.96 + 2.84));
137
+ let rdx = x + firstInline.x;
138
+ this.pdfDocument.rect(rdx, y, totalWidth, lw).clip();
139
+ this.pdfDocument.fillColor(group.decorationColor);
140
+ for (let i = 0; i < nbDashes; i++) {
141
+ this.pdfDocument.rect(rdx, y - lw / 2, 3.96, lw).fill();
142
+ rdx += 3.96 + 2.84;
143
+ }
144
+ } else if (group.decorationStyle === 'dotted') {
145
+ let nbDots = Math.ceil(totalWidth / (lw * 3));
146
+ let rx = x + firstInline.x;
147
+ this.pdfDocument.rect(rx, y, totalWidth, lw).clip();
148
+ this.pdfDocument.fillColor(group.decorationColor);
149
+ for (let i = 0; i < nbDots; i++) {
150
+ this.pdfDocument.rect(rx, y - lw / 2, lw, lw).fill();
151
+ rx += (lw * 3);
152
+ }
153
+ } else if (group.decorationStyle === 'wavy') {
154
+ let sh = 0.7, sv = 1;
155
+ let nbWaves = Math.ceil(totalWidth / (sh * 2)) + 1;
156
+ let rwx = x + firstInline.x - 1;
157
+ this.pdfDocument.rect(x + firstInline.x, y - sv, totalWidth, y + sv).clip();
158
+ this.pdfDocument.lineWidth(lw / 3);
159
+ this.pdfDocument.moveTo(rwx, y);
160
+ for (let i = 0; i < nbWaves; i++) {
161
+ this.pdfDocument.bezierCurveTo(rwx + sh, y - sv, rwx + sh * 2, y - sv, rwx + sh * 3, y)
162
+ .bezierCurveTo(rwx + sh * 4, y + sv, rwx + sh * 5, y + sv, rwx + sh * 6, y);
163
+ rwx += sh * 6;
164
+ }
165
+ this.pdfDocument.stroke(group.decorationColor);
166
+ } else {
167
+ this.pdfDocument.fillColor(group.decorationColor)
168
+ .rect(x + firstInline.x, y - lw / 2, totalWidth, lw)
169
+ .fill();
170
+ }
171
+ this.pdfDocument.restore();
172
+ }
173
+ }
174
+
175
+ export default TextDecorator;
@@ -0,0 +1,229 @@
1
+ import TextBreaker from './TextBreaker';
2
+ import StyleContextStack from './StyleContextStack';
3
+
4
+ const LEADING = /^(\s)+/g;
5
+ const TRAILING = /(\s)+$/g;
6
+
7
+ /**
8
+ * @param {Array} array
9
+ * @returns {Array}
10
+ */
11
+ const flattenTextArray = array => {
12
+ function flatten(array) {
13
+ return array.reduce((prev, cur) => {
14
+ let current = Array.isArray(cur.text) ? flatten(cur.text) : cur;
15
+ let more = [].concat(current).some(Array.isArray);
16
+ return prev.concat(more ? flatten(current) : current);
17
+ }, []);
18
+ }
19
+
20
+ if (!Array.isArray(array)) {
21
+ array = [array];
22
+ }
23
+
24
+ // TODO: Styling in nested text (issue: https://github.com/bpampuch/pdfmake/issues/1174)
25
+
26
+ array = flatten(array);
27
+
28
+ return array;
29
+ };
30
+
31
+
32
+ /**
33
+ * Text measurement utility
34
+ */
35
+ class TextInlines {
36
+
37
+ /**
38
+ * @param {object} pdfDocument object is instance of PDFDocument
39
+ */
40
+ constructor(pdfDocument) {
41
+ this.pdfDocument = pdfDocument;
42
+ }
43
+
44
+ /**
45
+ * Converts an array of strings (or inline-definition-objects) into a collection
46
+ * of inlines and calculated minWidth/maxWidth and their min/max widths
47
+ *
48
+ * @param {Array|object} textArray an array of inline-definition-objects (or strings)
49
+ * @param {StyleContextStack} styleContextStack current style stack
50
+ * @returns {object} collection of inlines, minWidth, maxWidth
51
+ */
52
+ buildInlines(textArray, styleContextStack) {
53
+ const getTrimmedWidth = item => {
54
+ return Math.max(0, item.width - item.leadingCut - item.trailingCut);
55
+ };
56
+
57
+ let minWidth = 0;
58
+ let maxWidth = 0;
59
+ let currentLineWidth;
60
+
61
+ let flattenedTextArray = flattenTextArray(textArray);
62
+
63
+ const textBreaker = new TextBreaker();
64
+ let brokenText = textBreaker.getBreaks(flattenedTextArray, styleContextStack);
65
+
66
+ let measuredText = this.measure(brokenText, styleContextStack);
67
+
68
+ measuredText.forEach(inline => {
69
+ minWidth = Math.max(minWidth, getTrimmedWidth(inline));
70
+
71
+ if (!currentLineWidth) {
72
+ currentLineWidth = { width: 0, leadingCut: inline.leadingCut, trailingCut: 0 };
73
+ }
74
+
75
+ currentLineWidth.width += inline.width;
76
+ currentLineWidth.trailingCut = inline.trailingCut;
77
+
78
+ maxWidth = Math.max(maxWidth, getTrimmedWidth(currentLineWidth));
79
+
80
+ if (inline.lineEnd) {
81
+ currentLineWidth = null;
82
+ }
83
+ });
84
+
85
+ if (StyleContextStack.getStyleProperty({}, styleContextStack, 'noWrap', false)) {
86
+ minWidth = maxWidth;
87
+ }
88
+
89
+ return {
90
+ items: measuredText,
91
+ minWidth: minWidth,
92
+ maxWidth: maxWidth
93
+ };
94
+ }
95
+
96
+ measure(array, styleContextStack) {
97
+ if (array.length) {
98
+ let leadingIndent = StyleContextStack.getStyleProperty(array[0], styleContextStack, 'leadingIndent', 0);
99
+ if (leadingIndent) {
100
+ array[0].leadingCut = -leadingIndent;
101
+ array[0].leadingIndent = leadingIndent;
102
+ }
103
+ }
104
+
105
+ array.forEach(item => {
106
+ let font = StyleContextStack.getStyleProperty(item, styleContextStack, 'font', 'Roboto');
107
+ let bold = StyleContextStack.getStyleProperty(item, styleContextStack, 'bold', false);
108
+ let italics = StyleContextStack.getStyleProperty(item, styleContextStack, 'italics', false);
109
+
110
+ item.font = this.pdfDocument.provideFont(font, bold, italics);
111
+
112
+ item.alignment = StyleContextStack.getStyleProperty(item, styleContextStack, 'alignment', 'left');
113
+ item.fontSize = StyleContextStack.getStyleProperty(item, styleContextStack, 'fontSize', 12);
114
+ item.fontFeatures = StyleContextStack.getStyleProperty(item, styleContextStack, 'fontFeatures', null);
115
+ item.characterSpacing = StyleContextStack.getStyleProperty(item, styleContextStack, 'characterSpacing', 0);
116
+ item.color = StyleContextStack.getStyleProperty(item, styleContextStack, 'color', 'black');
117
+ item.decoration = StyleContextStack.getStyleProperty(item, styleContextStack, 'decoration', null);
118
+ item.decorationColor = StyleContextStack.getStyleProperty(item, styleContextStack, 'decorationColor', null);
119
+ item.decorationStyle = StyleContextStack.getStyleProperty(item, styleContextStack, 'decorationStyle', null);
120
+ item.decorationThickness = StyleContextStack.getStyleProperty(item, styleContextStack, 'decorationThickness', null);
121
+ item.background = StyleContextStack.getStyleProperty(item, styleContextStack, 'background', null);
122
+ item.link = StyleContextStack.getStyleProperty(item, styleContextStack, 'link', null);
123
+ item.linkToPage = StyleContextStack.getStyleProperty(item, styleContextStack, 'linkToPage', null);
124
+ item.linkToDestination = StyleContextStack.getStyleProperty(item, styleContextStack, 'linkToDestination', null);
125
+ item.noWrap = item.acroform ? true: StyleContextStack.getStyleProperty(item, styleContextStack, 'noWrap', null);
126
+ item.opacity = StyleContextStack.getStyleProperty(item, styleContextStack, 'opacity', 1);
127
+ item.sup = StyleContextStack.getStyleProperty(item, styleContextStack, 'sup', false);
128
+ item.sub = StyleContextStack.getStyleProperty(item, styleContextStack, 'sub', false);
129
+
130
+ if (item.sup || item.sub) {
131
+ // font size reduction taken from here: https://en.wikipedia.org/wiki/Subscript_and_superscript#Desktop_publishing
132
+ item.fontSize *= 0.58;
133
+ }
134
+
135
+ let lineHeight = StyleContextStack.getStyleProperty(item, styleContextStack, 'lineHeight', 1);
136
+
137
+ if (item.acroform) {
138
+ item.width = item.width || 25;
139
+ item.height = item.height || 15;
140
+ } else {
141
+ item.width = this.widthOfText(item.text, item);
142
+ item.height = item.font.lineHeight(item.fontSize) * lineHeight;
143
+
144
+ if (!item.leadingCut) {
145
+ item.leadingCut = 0;
146
+ }
147
+
148
+ let preserveLeadingSpaces = StyleContextStack.getStyleProperty(item, styleContextStack, 'preserveLeadingSpaces', false);
149
+ if (!preserveLeadingSpaces) {
150
+ let leadingSpaces = item.text.match(LEADING);
151
+ if (leadingSpaces) {
152
+ item.leadingCut += this.widthOfText(leadingSpaces[0], item);
153
+ }
154
+ }
155
+
156
+ item.trailingCut = 0;
157
+
158
+ let preserveTrailingSpaces = StyleContextStack.getStyleProperty(item, styleContextStack, 'preserveTrailingSpaces', false);
159
+ if (!preserveTrailingSpaces) {
160
+ let trailingSpaces = item.text.match(TRAILING);
161
+ if (trailingSpaces) {
162
+ item.trailingCut = this.widthOfText(trailingSpaces[0], item);
163
+ }
164
+ }
165
+ }
166
+ }, this);
167
+
168
+ return array;
169
+ }
170
+
171
+ /**
172
+ * Width of text
173
+ *
174
+ * @param {string} text
175
+ * @param {object} inline
176
+ * @returns {number}
177
+ */
178
+ widthOfText(text, inline) {
179
+ return inline.font.widthOfString(text, inline.fontSize, inline.fontFeatures) + ((inline.characterSpacing || 0) * (text.length - 1));
180
+ }
181
+
182
+ /**
183
+ * Returns size of the specified string (without breaking it) using the current style
184
+ *
185
+ * @param {string} text text to be measured
186
+ * @param {object} styleContextStack current style stack
187
+ * @returns {object} size of the specified string
188
+ */
189
+ sizeOfText(text, styleContextStack) {
190
+ //TODO: refactor - extract from measure
191
+ let fontName = StyleContextStack.getStyleProperty({}, styleContextStack, 'font', 'Roboto');
192
+ let fontSize = StyleContextStack.getStyleProperty({}, styleContextStack, 'fontSize', 12);
193
+ let fontFeatures = StyleContextStack.getStyleProperty({}, styleContextStack, 'fontFeatures', null);
194
+ let bold = StyleContextStack.getStyleProperty({}, styleContextStack, 'bold', false);
195
+ let italics = StyleContextStack.getStyleProperty({}, styleContextStack, 'italics', false);
196
+ let lineHeight = StyleContextStack.getStyleProperty({}, styleContextStack, 'lineHeight', 1);
197
+ let characterSpacing = StyleContextStack.getStyleProperty({}, styleContextStack, 'characterSpacing', 0);
198
+
199
+ let font = this.pdfDocument.provideFont(fontName, bold, italics);
200
+
201
+ return {
202
+ width: this.widthOfText(text, { font: font, fontSize: fontSize, characterSpacing: characterSpacing, fontFeatures: fontFeatures }),
203
+ height: font.lineHeight(fontSize) * lineHeight,
204
+ fontSize: fontSize,
205
+ lineHeight: lineHeight,
206
+ ascender: font.ascender / 1000 * fontSize,
207
+ descender: font.descender / 1000 * fontSize
208
+ };
209
+ }
210
+
211
+ /**
212
+ * Returns size of the specified rotated string (without breaking it) using the current style
213
+ *
214
+ * @param {string} text text to be measured
215
+ * @param {number} angle
216
+ * @param {object} styleContextStack current style stack
217
+ * @returns {object} size of the specified string
218
+ */
219
+ sizeOfRotatedText(text, angle, styleContextStack) {
220
+ let angleRad = angle * Math.PI / -180;
221
+ let size = this.sizeOfText(text, styleContextStack);
222
+ return {
223
+ width: Math.abs(size.height * Math.sin(angleRad)) + Math.abs(size.width * Math.cos(angleRad)),
224
+ height: Math.abs(size.width * Math.sin(angleRad)) + Math.abs(size.height * Math.cos(angleRad))
225
+ };
226
+ }
227
+ }
228
+
229
+ export default TextInlines;
@@ -0,0 +1,43 @@
1
+ async function fetchUrl(url, headers = {}) {
2
+ try {
3
+ const response = await fetch(url, { headers });
4
+ if (!response.ok) {
5
+ throw new Error(`Failed to fetch (status code: ${response.status}, url: "${url}")`);
6
+ }
7
+ return await response.arrayBuffer();
8
+ } catch (error) {
9
+ throw new Error(`Network request failed (url: "${url}", error: ${error.message})`);
10
+ }
11
+ }
12
+
13
+ class URLResolver {
14
+ constructor(fs) {
15
+ this.fs = fs;
16
+ this.resolving = {};
17
+ }
18
+
19
+ resolve(url, headers = {}) {
20
+ const resolveUrlInternal = async () => {
21
+ if (url.toLowerCase().startsWith('https://') || url.toLowerCase().startsWith('http://')) {
22
+ if (this.fs.existsSync(url)) {
23
+ return; // url was downloaded earlier
24
+ }
25
+ const buffer = await fetchUrl(url, headers);
26
+ this.fs.writeFileSync(url, buffer);
27
+ }
28
+ // else cannot be resolved
29
+ };
30
+
31
+ if (!this.resolving[url]) {
32
+ this.resolving[url] = resolveUrlInternal();
33
+ }
34
+ return this.resolving[url];
35
+ }
36
+
37
+ resolved() {
38
+ return Promise.all(Object.values(this.resolving));
39
+ }
40
+
41
+ }
42
+
43
+ export default URLResolver;
package/src/base.js ADDED
@@ -0,0 +1,70 @@
1
+ import Printer from './Printer';
2
+ import virtualfs from './virtual-fs';
3
+ import { pack } from './helpers/tools';
4
+ import { isObject } from './helpers/variableType';
5
+
6
+ class pdfmake {
7
+
8
+ constructor() {
9
+ this.virtualfs = virtualfs;
10
+ this.urlResolver = null;
11
+ }
12
+
13
+ /**
14
+ * @param {object} docDefinition
15
+ * @param {?object} options
16
+ * @returns {object}
17
+ */
18
+ createPdf(docDefinition, options = {}) {
19
+ if (!isObject(docDefinition)) {
20
+ throw new Error("Parameter 'docDefinition' has an invalid type. Object expected.");
21
+ }
22
+
23
+ if (!isObject(options)) {
24
+ throw new Error("Parameter 'options' has an invalid type. Object expected.");
25
+ }
26
+
27
+ options.progressCallback = this.progressCallback;
28
+ options.tableLayouts = this.tableLayouts;
29
+
30
+ let printer = new Printer(this.fonts, this.virtualfs, this.urlResolver());
31
+ const pdfDocumentPromise = printer.createPdfKitDocument(docDefinition, options);
32
+
33
+ return this._transformToDocument(pdfDocumentPromise);
34
+ }
35
+
36
+ setProgressCallback(callback) {
37
+ this.progressCallback = callback;
38
+ }
39
+
40
+ addTableLayouts(tableLayouts) {
41
+ this.tableLayouts = pack(this.tableLayouts, tableLayouts);
42
+ }
43
+
44
+ setTableLayouts(tableLayouts) {
45
+ this.tableLayouts = tableLayouts;
46
+ }
47
+
48
+ clearTableLayouts() {
49
+ this.tableLayouts = {};
50
+ }
51
+
52
+ addFonts(fonts) {
53
+ this.fonts = pack(this.fonts, fonts);
54
+ }
55
+
56
+ setFonts(fonts) {
57
+ this.fonts = fonts;
58
+ }
59
+
60
+ clearFonts() {
61
+ this.fonts = {};
62
+ }
63
+
64
+ _transformToDocument(doc) {
65
+ return doc;
66
+ }
67
+
68
+ }
69
+
70
+ export default pdfmake;