@pdfme/schemas 3.1.5-dev.15 → 3.1.5-dev.17

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 (65) hide show
  1. package/dist/cjs/src/barcodes/index.js +2 -2
  2. package/dist/cjs/src/barcodes/index.js.map +1 -1
  3. package/dist/cjs/src/barcodes/propPanel.js +1 -1
  4. package/dist/cjs/src/barcodes/propPanel.js.map +1 -1
  5. package/dist/cjs/src/barcodes/uiRender.js +2 -2
  6. package/dist/cjs/src/barcodes/uiRender.js.map +1 -1
  7. package/dist/cjs/src/graphics/helper.js +115 -0
  8. package/dist/cjs/src/graphics/helper.js.map +1 -0
  9. package/dist/cjs/src/graphics/image.js +20 -30
  10. package/dist/cjs/src/graphics/image.js.map +1 -1
  11. package/dist/cjs/src/graphics/svg.js +21 -7
  12. package/dist/cjs/src/graphics/svg.js.map +1 -1
  13. package/dist/cjs/src/index.js +30 -4
  14. package/dist/cjs/src/index.js.map +1 -1
  15. package/dist/cjs/src/shapes/line.js +2 -2
  16. package/dist/cjs/src/shapes/line.js.map +1 -1
  17. package/dist/cjs/src/text/index.js +16 -2
  18. package/dist/cjs/src/text/index.js.map +1 -1
  19. package/dist/cjs/src/text/uiRender.js +1 -1
  20. package/dist/cjs/src/text/uiRender.js.map +1 -1
  21. package/dist/cjs/src/utils.js +1 -1
  22. package/dist/cjs/src/utils.js.map +1 -1
  23. package/dist/esm/src/barcodes/index.js +2 -2
  24. package/dist/esm/src/barcodes/index.js.map +1 -1
  25. package/dist/esm/src/barcodes/propPanel.js +1 -1
  26. package/dist/esm/src/barcodes/propPanel.js.map +1 -1
  27. package/dist/esm/src/barcodes/uiRender.js +2 -2
  28. package/dist/esm/src/barcodes/uiRender.js.map +1 -1
  29. package/dist/esm/src/graphics/helper.js +111 -0
  30. package/dist/esm/src/graphics/helper.js.map +1 -0
  31. package/dist/esm/src/graphics/image.js +20 -28
  32. package/dist/esm/src/graphics/image.js.map +1 -1
  33. package/dist/esm/src/graphics/svg.js +20 -7
  34. package/dist/esm/src/graphics/svg.js.map +1 -1
  35. package/dist/esm/src/index.js +4 -4
  36. package/dist/esm/src/index.js.map +1 -1
  37. package/dist/esm/src/shapes/line.js +2 -2
  38. package/dist/esm/src/shapes/line.js.map +1 -1
  39. package/dist/esm/src/text/index.js +15 -2
  40. package/dist/esm/src/text/index.js.map +1 -1
  41. package/dist/esm/src/text/uiRender.js +1 -1
  42. package/dist/esm/src/text/uiRender.js.map +1 -1
  43. package/dist/esm/src/utils.js +1 -1
  44. package/dist/esm/src/utils.js.map +1 -1
  45. package/dist/types/src/barcodes/index.d.ts +2 -2
  46. package/dist/types/src/graphics/helper.d.ts +5 -0
  47. package/dist/types/src/graphics/image.d.ts +3 -2
  48. package/dist/types/src/graphics/svg.d.ts +4 -3
  49. package/dist/types/src/index.d.ts +4 -4
  50. package/dist/types/src/shapes/line.d.ts +2 -2
  51. package/dist/types/src/shapes/rectAndEllipse.d.ts +6 -0
  52. package/dist/types/src/text/index.d.ts +3 -2
  53. package/dist/types/src/utils.d.ts +1 -1
  54. package/package.json +3 -4
  55. package/src/barcodes/index.ts +2 -2
  56. package/src/barcodes/propPanel.ts +1 -1
  57. package/src/barcodes/uiRender.ts +2 -2
  58. package/src/graphics/helper.ts +147 -0
  59. package/src/graphics/image.ts +32 -33
  60. package/src/graphics/svg.ts +22 -8
  61. package/src/index.ts +6 -3
  62. package/src/shapes/line.ts +2 -2
  63. package/src/text/index.ts +17 -2
  64. package/src/text/uiRender.ts +1 -1
  65. package/src/utils.ts +2 -1
@@ -19,6 +19,7 @@ export declare const rectangle: {
19
19
  width: number;
20
20
  height: number;
21
21
  readOnly?: boolean | undefined;
22
+ readOnlyValue?: string | undefined;
22
23
  rotate?: number | undefined;
23
24
  opacity?: number | undefined;
24
25
  };
@@ -38,6 +39,7 @@ export declare const rectangle: {
38
39
  opacity?: number | undefined;
39
40
  rotate?: number | undefined;
40
41
  readOnly?: boolean | undefined;
42
+ readOnlyValue?: string | undefined;
41
43
  };
42
44
  activeElements: HTMLElement[];
43
45
  changeSchemas: import("@pdfme/common").ChangeSchemas;
@@ -55,6 +57,7 @@ export declare const rectangle: {
55
57
  opacity?: number | undefined;
56
58
  rotate?: number | undefined;
57
59
  readOnly?: boolean | undefined;
60
+ readOnlyValue?: string | undefined;
58
61
  }[];
59
62
  pageSize: {
60
63
  width: number;
@@ -84,6 +87,7 @@ export declare const ellipse: {
84
87
  width: number;
85
88
  height: number;
86
89
  readOnly?: boolean | undefined;
90
+ readOnlyValue?: string | undefined;
87
91
  rotate?: number | undefined;
88
92
  opacity?: number | undefined;
89
93
  };
@@ -103,6 +107,7 @@ export declare const ellipse: {
103
107
  opacity?: number | undefined;
104
108
  rotate?: number | undefined;
105
109
  readOnly?: boolean | undefined;
110
+ readOnlyValue?: string | undefined;
106
111
  };
107
112
  activeElements: HTMLElement[];
108
113
  changeSchemas: import("@pdfme/common").ChangeSchemas;
@@ -120,6 +125,7 @@ export declare const ellipse: {
120
125
  opacity?: number | undefined;
121
126
  rotate?: number | undefined;
122
127
  readOnly?: boolean | undefined;
128
+ readOnlyValue?: string | undefined;
123
129
  }[];
124
130
  pageSize: {
125
131
  width: number;
@@ -1,4 +1,5 @@
1
1
  import type { Plugin } from '@pdfme/common';
2
2
  import type { TextSchema } from './types';
3
- declare const schema: Plugin<TextSchema>;
4
- export default schema;
3
+ declare const textSchema: Plugin<TextSchema>;
4
+ export default textSchema;
5
+ export declare const readOnlyText: Plugin<TextSchema>;
@@ -24,7 +24,7 @@ export declare const rotatePoint: (point: {
24
24
  y: number;
25
25
  };
26
26
  export declare const addAlphaToHex: (hex: string, alphaPercentage: number) => string;
27
- export declare const isEditable: (mode: Mode) => boolean;
27
+ export declare const isEditable: (mode: Mode, schema: Schema) => boolean;
28
28
  export declare const hex2RgbColor: (hexString: string | undefined) => import("@pdfme/pdf-lib").RGB | undefined;
29
29
  export declare const readFile: (input: File | FileList | null) => Promise<string | ArrayBuffer>;
30
30
  export declare const createErrorElm: () => HTMLDivElement;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pdfme/schemas",
3
- "version": "3.1.5-dev.15",
3
+ "version": "3.1.5-dev.17",
4
4
  "sideEffects": false,
5
5
  "author": "hand-dot",
6
6
  "license": "MIT",
@@ -43,11 +43,10 @@
43
43
  "prettier": "prettier --write 'src/**/*.ts'"
44
44
  },
45
45
  "dependencies": {
46
- "@pdfme/pdf-lib": "^1.18.1",
46
+ "@pdfme/pdf-lib": "^1.18.3",
47
47
  "bwip-js": "^4.1.1",
48
48
  "fast-xml-parser": "^4.3.2",
49
- "fontkit": "^2.0.2",
50
- "image-size": "^1.0.2"
49
+ "fontkit": "^2.0.2"
51
50
  },
52
51
  "devDependencies": {
53
52
  "@pdfme/common": "file:../common",
@@ -5,7 +5,7 @@ import type { BarcodeSchema, BarcodeTypes } from './types';
5
5
  import { BARCODE_TYPES } from './constants.js';
6
6
  import { Plugin } from '@pdfme/common';
7
7
 
8
- const schemas = BARCODE_TYPES.reduce(
8
+ const barcodes = BARCODE_TYPES.reduce(
9
9
  (acc, type) =>
10
10
  Object.assign(acc, {
11
11
  [type]: { pdf: pdfRender, ui: uiRender, propPanel: getPropPanelByBarcodeType(type) },
@@ -13,4 +13,4 @@ const schemas = BARCODE_TYPES.reduce(
13
13
  {} as Record<BarcodeTypes, Plugin<BarcodeSchema>>
14
14
  );
15
15
 
16
- export default schemas;
16
+ export default barcodes;
@@ -154,7 +154,7 @@ export const getPropPanelByBarcodeType = (barcodeType: string): PropPanel<Barcod
154
154
 
155
155
  const defaults = barcodeDefaults.find(({ defaultSchema }) => defaultSchema.type === barcodeType);
156
156
 
157
- if (!defaults) throw new Error(`[@pdfme/schemas] No default for barcode type ${barcodeType}`);
157
+ if (!defaults) throw new Error(`[@pdfme/schemas/barcodes] No default for barcode type ${barcodeType}`);
158
158
 
159
159
  return {
160
160
  schema: ({ i18n }) => ({
@@ -47,7 +47,7 @@ export const uiRender = async (arg: UIRenderProps<BarcodeSchema>) => {
47
47
  };
48
48
  Object.assign(container.style, containerStyle);
49
49
  rootElement.appendChild(container);
50
- const editable = isEditable(mode);
50
+ const editable = isEditable(mode, schema);
51
51
  if (editable) {
52
52
  const input = document.createElement('input');
53
53
  const inputStyle: CSS.Properties = {
@@ -82,7 +82,7 @@ export const uiRender = async (arg: UIRenderProps<BarcodeSchema>) => {
82
82
  if (!value) return;
83
83
  try {
84
84
  if (!validateBarcodeInput(schema.type, value))
85
- throw new Error('[@pdfme/schemas] Invalid barcode input');
85
+ throw new Error('[@pdfme/schemas/barcodes] Invalid barcode input');
86
86
  const imgElm = await createBarcodeImageElm(schema, value);
87
87
  container.appendChild(imgElm);
88
88
  } catch (err) {
@@ -0,0 +1,147 @@
1
+ // ref: https://github.com/image-size/image-size ----------------------------
2
+ // The following code is adapted from the image-size code. Unnecessary formats and dependencies on Node have been removed.
3
+ type IImage = {
4
+ validate: (input: Uint8Array) => boolean;
5
+ calculate: (input: Uint8Array) => { width: number; height: number } | undefined;
6
+ };
7
+
8
+ const decoder = new TextDecoder();
9
+ const toUTF8String = (input: Uint8Array, start = 0, end = input.length) =>
10
+ decoder.decode(input.slice(start, end));
11
+
12
+ const toHexString = (input: Uint8Array, start = 0, end = input.length) =>
13
+ input.slice(start, end).reduce((memo, i) => memo + ('0' + i.toString(16)).slice(-2), '');
14
+
15
+ const readUInt16BE = (input: Uint8Array, offset = 0) => input[offset] * 2 ** 8 + input[offset + 1];
16
+
17
+ const readUInt32BE = (input: Uint8Array, offset = 0) =>
18
+ input[offset] * 2 ** 24 +
19
+ input[offset + 1] * 2 ** 16 +
20
+ input[offset + 2] * 2 ** 8 +
21
+ input[offset + 3];
22
+
23
+ const extractSize = (input: Uint8Array, index: number) => {
24
+ return {
25
+ height: readUInt16BE(input, index),
26
+ width: readUInt16BE(input, index + 2),
27
+ };
28
+ };
29
+
30
+ const validateInput = (input: Uint8Array, index: number): void => {
31
+ // index should be within buffer limits
32
+ if (index > input.length) {
33
+ throw new TypeError('Corrupt JPG, exceeded buffer limits');
34
+ }
35
+ // Every JPEG block must begin with a 0xFF
36
+ if (input[index] !== 0xff) {
37
+ throw new TypeError('Invalid JPG, marker table corrupted');
38
+ }
39
+ };
40
+
41
+ const JPG: IImage = {
42
+ validate: (input) => toHexString(input, 0, 2) === 'ffd8',
43
+
44
+ calculate(input) {
45
+ // Skip 4 chars, they are for signature
46
+ input = input.slice(4);
47
+
48
+ let next: number;
49
+ while (input.length) {
50
+ // read length of the next block
51
+ const i = readUInt16BE(input, 0);
52
+
53
+ // ensure correct format
54
+ validateInput(input, i);
55
+
56
+ // 0xFFC0 is baseline standard(SOF)
57
+ // 0xFFC1 is baseline optimized(SOF)
58
+ // 0xFFC2 is progressive(SOF2)
59
+ next = input[i + 1];
60
+ if (next === 0xc0 || next === 0xc1 || next === 0xc2) {
61
+ const size = extractSize(input, i + 5);
62
+
63
+ return size;
64
+ }
65
+
66
+ // move to the next block
67
+ input = input.slice(i + 2);
68
+ }
69
+
70
+ throw new TypeError('Invalid JPG, no size found');
71
+ },
72
+ };
73
+
74
+ const pngSignature = 'PNG\r\n\x1a\n';
75
+ const pngImageHeaderChunkName = 'IHDR';
76
+
77
+ // Used to detect "fried" png's: http://www.jongware.com/pngdefry.html
78
+ const pngFriedChunkName = 'CgBI';
79
+
80
+ const PNG: IImage = {
81
+ validate(input) {
82
+ if (pngSignature === toUTF8String(input, 1, 8)) {
83
+ let chunkName = toUTF8String(input, 12, 16);
84
+ if (chunkName === pngFriedChunkName) {
85
+ chunkName = toUTF8String(input, 28, 32);
86
+ }
87
+ if (chunkName !== pngImageHeaderChunkName) {
88
+ throw new TypeError('Invalid PNG');
89
+ }
90
+ return true;
91
+ }
92
+ return false;
93
+ },
94
+
95
+ calculate(input) {
96
+ if (toUTF8String(input, 12, 16) === pngFriedChunkName) {
97
+ return {
98
+ height: readUInt32BE(input, 36),
99
+ width: readUInt32BE(input, 32),
100
+ };
101
+ }
102
+ return {
103
+ height: readUInt32BE(input, 20),
104
+ width: readUInt32BE(input, 16),
105
+ };
106
+ },
107
+ };
108
+
109
+ const typeHandlers = {
110
+ jpg: JPG,
111
+ png: PNG,
112
+ };
113
+
114
+ type imageType = keyof typeof typeHandlers;
115
+
116
+ function detector(input: Uint8Array): imageType | undefined {
117
+ const firstBytes: { [byte: number]: imageType } = {
118
+ 0x89: 'png',
119
+ 0xff: 'jpg',
120
+ };
121
+ const byte = input[0];
122
+ if (byte in firstBytes) {
123
+ const type = firstBytes[byte];
124
+ if (type && typeHandlers[type].validate(input)) {
125
+ return type;
126
+ }
127
+ }
128
+
129
+ const keys = Object.keys(typeHandlers) as imageType[];
130
+ return keys.find((key: imageType) => typeHandlers[key].validate(input));
131
+ }
132
+
133
+ export const imageSize = (imgBuffer: Buffer): { height: number; width: number } => {
134
+ const type = detector(imgBuffer);
135
+
136
+ if (typeof type !== 'undefined' && type in typeHandlers) {
137
+ const size = typeHandlers[type].calculate(imgBuffer);
138
+ if (size !== undefined) {
139
+ return size;
140
+ }
141
+ }
142
+
143
+ throw new TypeError(
144
+ '[@pdfme/schemas/images] Unsupported file type: ' + (type === undefined ? 'undefined' : type)
145
+ );
146
+ };
147
+ // ----------------------------
@@ -3,16 +3,11 @@ import type { PDFImage } from '@pdfme/pdf-lib';
3
3
  import type { Plugin } from '@pdfme/common';
4
4
  import type { PDFRenderProps, Schema } from '@pdfme/common';
5
5
  import type * as CSS from 'csstype';
6
- import sizeOf from 'image-size';
7
6
  import { Buffer } from 'buffer';
8
7
  import { UIRenderProps } from '@pdfme/common';
9
- import {
10
- convertForPdfLayoutProps,
11
- addAlphaToHex,
12
- isEditable,
13
- readFile,
14
- } from '../utils.js';
8
+ import { convertForPdfLayoutProps, addAlphaToHex, isEditable, readFile } from '../utils.js';
15
9
  import { DEFAULT_OPACITY } from '../constants.js';
10
+ import { imageSize } from './helper.js';
16
11
 
17
12
  const px2mm = (px: number): number => {
18
13
  // http://www.endmemo.com/sconvert/millimeterpixel.php
@@ -20,27 +15,6 @@ const px2mm = (px: number): number => {
20
15
  return parseFloat(String(px)) * ratio;
21
16
  };
22
17
 
23
- const getDimension = (imgBuffer: Buffer): Promise<{ height: number; width: number }> => {
24
- if (typeof window !== 'undefined') {
25
- return new Promise((resolve, reject) => {
26
- const blob = new Blob([imgBuffer]);
27
- const url = URL.createObjectURL(blob);
28
- const img = new Image();
29
- img.onload = () => {
30
- resolve({ width: img.width, height: img.height });
31
- URL.revokeObjectURL(url);
32
- };
33
- img.onerror = (e) => {
34
- reject(e);
35
- };
36
- img.src = url;
37
- });
38
- } else {
39
- const dimensions = sizeOf(imgBuffer);
40
- return Promise.resolve({ width: dimensions.width ?? 0, height: dimensions.height ?? 0 });
41
- }
42
- };
43
-
44
18
  const getCacheKey = (schema: Schema, input: string) => `${schema.type}${input}`;
45
19
  const fullSize = { width: '100%', height: '100%' };
46
20
  const defaultValue =
@@ -48,7 +22,7 @@ const defaultValue =
48
22
 
49
23
  interface ImageSchema extends Schema {}
50
24
 
51
- const schema: Plugin<ImageSchema> = {
25
+ const imageSchema: Plugin<ImageSchema> = {
52
26
  pdf: async (arg: PDFRenderProps<ImageSchema>) => {
53
27
  const { value, schema, pdfDoc, page, _cache } = arg;
54
28
  if (!value || !value.startsWith('data:image/')) return;
@@ -65,7 +39,7 @@ const schema: Plugin<ImageSchema> = {
65
39
  const dataUriPrefix = ';base64,';
66
40
  const idx = value.indexOf(dataUriPrefix);
67
41
  const imgBase64 = value.substring(idx + dataUriPrefix.length, value.length);
68
- const dimension = await getDimension(Buffer.from(imgBase64, 'base64'));
42
+ const dimension = imageSize(Buffer.from(imgBase64, 'base64'));
69
43
 
70
44
  const imageWidth = px2mm(dimension.width);
71
45
  const imageHeight = px2mm(dimension.height);
@@ -92,8 +66,18 @@ const schema: Plugin<ImageSchema> = {
92
66
  page.drawImage(image, { x, y, rotate, width, height, opacity });
93
67
  },
94
68
  ui: (arg: UIRenderProps<ImageSchema>) => {
95
- const { value, rootElement, mode, onChange, stopEditing, tabIndex, placeholder, theme } = arg;
96
- const editable = isEditable(mode);
69
+ const {
70
+ value,
71
+ rootElement,
72
+ mode,
73
+ onChange,
74
+ stopEditing,
75
+ tabIndex,
76
+ placeholder,
77
+ theme,
78
+ schema,
79
+ } = arg;
80
+ const editable = isEditable(mode, schema);
97
81
  const isDefault = value === defaultValue;
98
82
 
99
83
  const container = document.createElement('div');
@@ -205,4 +189,19 @@ const schema: Plugin<ImageSchema> = {
205
189
  },
206
190
  },
207
191
  };
208
- export default schema;
192
+
193
+ export default imageSchema;
194
+
195
+ export const readOnlyImage: Plugin<ImageSchema> = {
196
+ pdf: imageSchema.pdf,
197
+ ui: imageSchema.ui,
198
+ propPanel: {
199
+ ...imageSchema.propPanel,
200
+ defaultSchema: {
201
+ ...imageSchema.propPanel.defaultSchema,
202
+ type: 'readOnlyImage',
203
+ readOnly: true,
204
+ readOnlyValue: defaultValue,
205
+ },
206
+ },
207
+ };
@@ -4,22 +4,22 @@ import { convertForPdfLayoutProps, isEditable, addAlphaToHex, createErrorElm } f
4
4
 
5
5
  const isValidSVG = (svgString: string) => XMLValidator.validate(svgString) === true;
6
6
 
7
- const svgData = `<svg viewBox="0 0 488 600" version="1.1" xmlns="http://www.w3.org/2000/svg">
7
+ const defaultValue = `<svg viewBox="0 0 488 600" version="1.1" xmlns="http://www.w3.org/2000/svg">
8
8
  <g transform="matrix(1,0,0,1,-56,0)" fill="#000000" stroke="none">
9
9
  <path d="M228.667,0L56,172.667L56.267,345.334L56.667,518L59.733,527.334C65.867,545.467 72.933,557.067 86,570.134C96.133,580.4 100,583.2 110.667,588.4C134.533,600.134 120,599.334 300,599.334C480,599.334 465.467,600.134 489.334,588.4C500,583.2 503.867,580.4 514,570.134C527.334,556.8 534.534,544.8 540.267,526.667L543.334,516.667L543.334,83.333L540.267,73.333C534.534,55.2 527.334,43.2 514,29.867C503.867,19.6 500,16.8 489.334,11.6C465.734,0 475.467,0.8 344.667,0.267L228.667,0ZM466.4,41.6C483.334,48.933 496.267,61.867 502.934,78.4L506,86L506,514L502.934,521.734C496,538.934 480.267,553.867 463.334,559.334C455.6,561.867 450.8,562 300,562C149.2,562 144.4,561.867 136.667,559.334C119.733,553.867 104,538.934 97.067,521.734L94,514L93.6,351.067L93.333,188.133L149.067,187.733L204.667,187.333L213.6,182.933C224.8,177.467 235.867,165.867 240.267,155.067C243.333,147.467 243.333,146.4 243.733,92.267L244.133,37.2L458,38L466.4,41.6ZM195.067,304C175.6,306.8 164,320.667 165.6,339.467C166,343.6 167.6,348.667 169.733,352.4C174.4,360.267 185.2,365.734 201.867,368.534C208.4,369.734 215.067,371.467 216.8,372.667C224,377.334 221.467,389.067 212.533,392C205.6,394.4 193.733,392.934 185.6,388.8C173.333,382.534 164,385.334 164,395.2C164,400.934 170.133,406.667 180.267,410.134C190.933,413.867 217.067,413.734 225.467,409.867C238.933,403.6 246.667,390 244.8,375.6C242.667,359.734 232.8,351.334 212.267,347.867C193.6,344.8 189.333,342.4 189.333,334.533C189.333,324.267 201.867,320.933 218.267,326.667C228.667,330.267 232.533,330.133 235.867,325.867C242.133,318 237.6,310.667 224.267,306.8C213.333,303.6 204.267,302.8 195.067,304ZM386,304.133C377.6,305.333 374,306.8 367.334,311.6C355.734,320.133 351.2,336.4 352.4,365.334C353.2,385.334 356,394.4 364.134,402.534C372.267,410.667 381.734,413.734 396.667,413.067C406.8,412.667 409.734,412 415.734,408.667C429.2,401.334 434.534,390.934 435.6,370.667C436.4,353.734 436,353.067 420.934,352.267C401.867,351.334 396,353.467 396,361.867C396,367.867 399.467,370.667 407.067,370.667C413.2,370.667 413.334,370.667 413.334,374.934C413.334,394 386.267,400.534 378.534,383.467C374.934,375.334 374.934,341.867 378.534,333.733C382,326.4 387.467,323.467 396.8,324.267C403.067,324.8 404.667,325.6 410.534,331.067C414.267,334.533 418.4,337.333 419.867,337.333C427.334,337.333 433.334,330.267 431.334,323.733C427.2,310.133 406.4,301.2 386,304.133ZM258.4,308C255.067,311.467 254.533,312.8 255.2,316.4C257.067,326.667 285.333,405.867 288.133,408.8C289.733,410.534 293.067,412.267 295.333,412.8C303.867,414.4 310.667,407.867 314.4,394.667C315.067,392.134 321.2,374.134 327.867,354.8C334.8,334.533 340,317.467 340,314.533C340,303.733 325.067,299.867 319.867,309.467C318.533,312.133 309.467,340.933 302.667,364C301.067,369.467 299.333,374.4 298.8,375.067C298.267,375.6 292.933,360.8 286.933,342C275.333,306 274.133,304 266.267,304C263.867,304 261.067,305.467 258.4,308Z" style="fill-rule:nonzero;"/>
10
10
  </g>
11
11
  </svg>`;
12
12
 
13
- interface SVG extends Schema {}
13
+ interface SVGSchema extends Schema {}
14
14
 
15
- const schema: Plugin<SVG> = {
15
+ const svgSchema: Plugin<SVGSchema> = {
16
16
  ui: (arg) => {
17
- const { rootElement, value, mode, onChange, theme } = arg;
18
- const container = document.createElement(isEditable(mode) ? 'textarea' : 'div');
17
+ const { rootElement, value, mode, onChange, theme, schema } = arg;
18
+ const container = document.createElement(isEditable(mode, schema) ? 'textarea' : 'div');
19
19
  container.style.width = '100%';
20
20
  container.style.height = '100%';
21
21
  container.style.boxSizing = 'border-box';
22
- if (isEditable(mode)) {
22
+ if (isEditable(mode, schema)) {
23
23
  const textarea = container as HTMLTextAreaElement;
24
24
  textarea.value = value;
25
25
  textarea.style.position = 'absolute';
@@ -71,7 +71,7 @@ const schema: Plugin<SVG> = {
71
71
  },
72
72
  propPanel: {
73
73
  schema: {},
74
- defaultValue: svgData,
74
+ defaultValue,
75
75
  defaultSchema: {
76
76
  type: 'svg',
77
77
  position: { x: 0, y: 0 },
@@ -81,4 +81,18 @@ const schema: Plugin<SVG> = {
81
81
  },
82
82
  };
83
83
 
84
- export default schema;
84
+ export default svgSchema;
85
+
86
+ export const readOnlySvg: Plugin<SVGSchema> = {
87
+ pdf: svgSchema.pdf,
88
+ ui: svgSchema.ui,
89
+ propPanel: {
90
+ ...svgSchema.propPanel,
91
+ defaultSchema: {
92
+ ...svgSchema.propPanel.defaultSchema,
93
+ type: 'readOnlySvg',
94
+ readOnly: true,
95
+ readOnlyValue: defaultValue,
96
+ },
97
+ },
98
+ };
package/src/index.ts CHANGED
@@ -1,6 +1,6 @@
1
- import text from './text/index.js';
2
- import image from './graphics/image.js';
3
- import svg from './graphics/svg.js';
1
+ import text, { readOnlyText } from './text/index.js';
2
+ import image, { readOnlyImage } from './graphics/image.js';
3
+ import svg, { readOnlySvg } from './graphics/svg.js';
4
4
  import barcodes from './barcodes/index.js';
5
5
  import line from './shapes/line.js';
6
6
  import { rectangle, ellipse } from './shapes/rectAndEllipse.js';
@@ -10,8 +10,11 @@ const builtInPlugins = { Text: text };
10
10
 
11
11
  export {
12
12
  text,
13
+ readOnlyText,
13
14
  image,
15
+ readOnlyImage,
14
16
  svg,
17
+ readOnlySvg,
15
18
  barcodes,
16
19
  line,
17
20
  rectangle,
@@ -8,7 +8,7 @@ interface LineSchema extends Schema {
8
8
  color: string;
9
9
  }
10
10
 
11
- const schema: Plugin<LineSchema> = {
11
+ const lineSchema: Plugin<LineSchema> = {
12
12
  pdf: (arg: PDFRenderProps<LineSchema>) => {
13
13
  const { page, schema } = arg;
14
14
  const pageHeight = page.getHeight();
@@ -64,4 +64,4 @@ const schema: Plugin<LineSchema> = {
64
64
  },
65
65
  },
66
66
  };
67
- export default schema;
67
+ export default lineSchema;
package/src/text/index.ts CHANGED
@@ -4,5 +4,20 @@ import { propPanel } from './propPanel.js';
4
4
  import { uiRender } from './uiRender.js';
5
5
  import type { TextSchema } from './types';
6
6
 
7
- const schema: Plugin<TextSchema> = { pdf: pdfRender, ui: uiRender, propPanel };
8
- export default schema;
7
+ const textSchema: Plugin<TextSchema> = { pdf: pdfRender, ui: uiRender, propPanel };
8
+
9
+ export default textSchema;
10
+
11
+ export const readOnlyText: Plugin<TextSchema> = {
12
+ pdf: textSchema.pdf,
13
+ ui: textSchema.ui,
14
+ propPanel: {
15
+ ...textSchema.propPanel,
16
+ defaultSchema: {
17
+ ...textSchema.propPanel.defaultSchema,
18
+ type: 'readOnlyText',
19
+ readOnly: true,
20
+ readOnlyValue: textSchema.propPanel.defaultValue,
21
+ },
22
+ },
23
+ };
@@ -117,7 +117,7 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
117
117
  const textBlock = document.createElement('div');
118
118
  Object.assign(textBlock.style, textBlockStyle);
119
119
 
120
- if (isEditable(mode)) {
120
+ if (isEditable(mode, schema)) {
121
121
  textBlock.contentEditable = 'plaintext-only';
122
122
  textBlock.tabIndex = tabIndex || 0;
123
123
  textBlock.innerText = value;
package/src/utils.ts CHANGED
@@ -70,7 +70,8 @@ export const addAlphaToHex = (hex: string, alphaPercentage: number) => {
70
70
  return hex + alphaHex;
71
71
  };
72
72
 
73
- export const isEditable = (mode: Mode) => mode === 'form' || mode === 'designer';
73
+ export const isEditable = (mode: Mode, schema: Schema) =>
74
+ mode === 'designer' || (mode === 'form' && schema.readOnly !== true);
74
75
 
75
76
  const hex2rgb = (hex: string) => {
76
77
  if (hex.slice(0, 1) === '#') hex = hex.slice(1);