@pdfme/manipulator 0.0.0
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.
- package/README.md +5 -0
- package/dist/cjs/__tests__/e2e/insert.e2e.test.js +32 -0
- package/dist/cjs/__tests__/e2e/insert.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/merge.e2e.test.js +20 -0
- package/dist/cjs/__tests__/e2e/merge.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/move.e2e.test.js +18 -0
- package/dist/cjs/__tests__/e2e/move.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/organize-complex.e2e.test.js +96 -0
- package/dist/cjs/__tests__/e2e/organize-complex.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/organize-single.e2e.test.js +68 -0
- package/dist/cjs/__tests__/e2e/organize-single.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/remove.e2e.test.js +28 -0
- package/dist/cjs/__tests__/e2e/remove.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/rotate.e2e.test.js +38 -0
- package/dist/cjs/__tests__/e2e/rotate.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/e2e/split.e2e.test.js +27 -0
- package/dist/cjs/__tests__/e2e/split.e2e.test.js.map +1 -0
- package/dist/cjs/__tests__/insert.test.js +18 -0
- package/dist/cjs/__tests__/insert.test.js.map +1 -0
- package/dist/cjs/__tests__/merge.test.js +16 -0
- package/dist/cjs/__tests__/merge.test.js.map +1 -0
- package/dist/cjs/__tests__/move.test.js +16 -0
- package/dist/cjs/__tests__/move.test.js.map +1 -0
- package/dist/cjs/__tests__/organize.test.js +75 -0
- package/dist/cjs/__tests__/organize.test.js.map +1 -0
- package/dist/cjs/__tests__/remove.test.js +20 -0
- package/dist/cjs/__tests__/remove.test.js.map +1 -0
- package/dist/cjs/__tests__/rotate.test.js +17 -0
- package/dist/cjs/__tests__/rotate.test.js.map +1 -0
- package/dist/cjs/__tests__/split.test.js +21 -0
- package/dist/cjs/__tests__/split.test.js.map +1 -0
- package/dist/cjs/__tests__/test-helpers.js +44 -0
- package/dist/cjs/__tests__/test-helpers.js.map +1 -0
- package/dist/cjs/__tests__/utils.js +29 -0
- package/dist/cjs/__tests__/utils.js.map +1 -0
- package/dist/cjs/src/index.js +161 -0
- package/dist/cjs/src/index.js.map +1 -0
- package/dist/esm/__tests__/e2e/insert.e2e.test.js +30 -0
- package/dist/esm/__tests__/e2e/insert.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/merge.e2e.test.js +18 -0
- package/dist/esm/__tests__/e2e/merge.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/move.e2e.test.js +16 -0
- package/dist/esm/__tests__/e2e/move.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/organize-complex.e2e.test.js +94 -0
- package/dist/esm/__tests__/e2e/organize-complex.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/organize-single.e2e.test.js +66 -0
- package/dist/esm/__tests__/e2e/organize-single.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/remove.e2e.test.js +26 -0
- package/dist/esm/__tests__/e2e/remove.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/rotate.e2e.test.js +36 -0
- package/dist/esm/__tests__/e2e/rotate.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/e2e/split.e2e.test.js +25 -0
- package/dist/esm/__tests__/e2e/split.e2e.test.js.map +1 -0
- package/dist/esm/__tests__/insert.test.js +16 -0
- package/dist/esm/__tests__/insert.test.js.map +1 -0
- package/dist/esm/__tests__/merge.test.js +14 -0
- package/dist/esm/__tests__/merge.test.js.map +1 -0
- package/dist/esm/__tests__/move.test.js +14 -0
- package/dist/esm/__tests__/move.test.js.map +1 -0
- package/dist/esm/__tests__/organize.test.js +73 -0
- package/dist/esm/__tests__/organize.test.js.map +1 -0
- package/dist/esm/__tests__/remove.test.js +18 -0
- package/dist/esm/__tests__/remove.test.js.map +1 -0
- package/dist/esm/__tests__/rotate.test.js +15 -0
- package/dist/esm/__tests__/rotate.test.js.map +1 -0
- package/dist/esm/__tests__/split.test.js +19 -0
- package/dist/esm/__tests__/split.test.js.map +1 -0
- package/dist/esm/__tests__/test-helpers.js +32 -0
- package/dist/esm/__tests__/test-helpers.js.map +1 -0
- package/dist/esm/__tests__/utils.js +23 -0
- package/dist/esm/__tests__/utils.js.map +1 -0
- package/dist/esm/src/index.js +152 -0
- package/dist/esm/src/index.js.map +1 -0
- package/dist/types/__tests__/e2e/insert.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/merge.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/move.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/organize-complex.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/organize-single.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/remove.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/rotate.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/e2e/split.e2e.test.d.ts +1 -0
- package/dist/types/__tests__/insert.test.d.ts +1 -0
- package/dist/types/__tests__/merge.test.d.ts +1 -0
- package/dist/types/__tests__/move.test.d.ts +1 -0
- package/dist/types/__tests__/organize.test.d.ts +1 -0
- package/dist/types/__tests__/remove.test.d.ts +1 -0
- package/dist/types/__tests__/rotate.test.d.ts +1 -0
- package/dist/types/__tests__/split.test.d.ts +1 -0
- package/dist/types/__tests__/test-helpers.d.ts +6 -0
- package/dist/types/__tests__/utils.d.ts +3 -0
- package/dist/types/src/index.d.ts +46 -0
- package/eslint.config.mjs +22 -0
- package/jest.setup.js +2 -0
- package/package.json +81 -0
- package/src/index.ts +240 -0
- package/tsconfig.cjs.json +10 -0
- package/tsconfig.esm.json +11 -0
- package/tsconfig.json +6 -0
package/package.json
ADDED
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@pdfme/manipulator",
|
|
3
|
+
"version": "0.0.0",
|
|
4
|
+
"sideEffects": false,
|
|
5
|
+
"author": "hand-dot",
|
|
6
|
+
"license": "MIT",
|
|
7
|
+
"keywords": [
|
|
8
|
+
"pdf",
|
|
9
|
+
"pdf-generation",
|
|
10
|
+
"pdf-designer",
|
|
11
|
+
"pdf-viewer",
|
|
12
|
+
"typescript",
|
|
13
|
+
"react"
|
|
14
|
+
],
|
|
15
|
+
"description": "TypeScript base PDF generator and React base UI. Open source, developed by the community, and completely free to use under the MIT license!",
|
|
16
|
+
"homepage": "https://pdfme.com",
|
|
17
|
+
"repository": {
|
|
18
|
+
"type": "git",
|
|
19
|
+
"url": "git@github.com:pdfme/pdfme.git"
|
|
20
|
+
},
|
|
21
|
+
"bugs": {
|
|
22
|
+
"url": "https://github.com/pdfme/pdfme/issues"
|
|
23
|
+
},
|
|
24
|
+
"main": "dist/cjs/src/index.js",
|
|
25
|
+
"module": "dist/esm/src/index.js",
|
|
26
|
+
"types": "dist/types/src/index.d.ts",
|
|
27
|
+
"exports": {
|
|
28
|
+
".": {
|
|
29
|
+
"import": "./dist/esm/src/index.js",
|
|
30
|
+
"require": "./dist/cjs/src/index.js",
|
|
31
|
+
"types": "./dist/types/src/index.d.ts"
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
"scripts": {
|
|
35
|
+
"dev": "tsc -p tsconfig.esm.json -w",
|
|
36
|
+
"build": "run-p build:cjs build:esm",
|
|
37
|
+
"build:cjs": "tsc -p tsconfig.cjs.json",
|
|
38
|
+
"build:esm": "tsc -p tsconfig.esm.json",
|
|
39
|
+
"clean": "rimraf dist",
|
|
40
|
+
"lint": "eslint --ext .ts src --config eslint.config.mjs",
|
|
41
|
+
"test": "jest",
|
|
42
|
+
"test:update-snapshots": "jest --updateSnapshot",
|
|
43
|
+
"prune": "ts-prune src",
|
|
44
|
+
"prettier": "prettier --write 'src/**/*.ts'"
|
|
45
|
+
},
|
|
46
|
+
"dependencies": {
|
|
47
|
+
"@pdfme/pdf-lib": "*"
|
|
48
|
+
},
|
|
49
|
+
"publishConfig": {
|
|
50
|
+
"access": "public"
|
|
51
|
+
},
|
|
52
|
+
"devDependencies": {
|
|
53
|
+
"@pdfme/converter": "*",
|
|
54
|
+
"@types/estree": "^1.0.6"
|
|
55
|
+
},
|
|
56
|
+
"jest": {
|
|
57
|
+
"preset": "ts-jest/presets/default-esm",
|
|
58
|
+
"testEnvironment": "node",
|
|
59
|
+
"testMatch": [
|
|
60
|
+
"**/__tests__/**/*.test.ts"
|
|
61
|
+
],
|
|
62
|
+
"setupFilesAfterEnv": [
|
|
63
|
+
"<rootDir>/jest.setup.js"
|
|
64
|
+
],
|
|
65
|
+
"extensionsToTreatAsEsm": [
|
|
66
|
+
".ts"
|
|
67
|
+
],
|
|
68
|
+
"transform": {
|
|
69
|
+
"^.+\\.tsx?$": [
|
|
70
|
+
"ts-jest",
|
|
71
|
+
{
|
|
72
|
+
"tsconfig": "./tsconfig.json",
|
|
73
|
+
"useESM": true
|
|
74
|
+
}
|
|
75
|
+
]
|
|
76
|
+
},
|
|
77
|
+
"moduleNameMapper": {
|
|
78
|
+
"^(\\.{1,2}/(?:src|__tests__)/.*)\\.js$": "$1.ts"
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
import { PDFDocument, RotationTypes } from '@pdfme/pdf-lib';
|
|
2
|
+
|
|
3
|
+
const merge = async (pdfs: (ArrayBuffer | Uint8Array)[]): Promise<Uint8Array> => {
|
|
4
|
+
if (!pdfs.length) {
|
|
5
|
+
throw new Error('[@pdfme/manipulator] At least one PDF is required for merging');
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
const mergedPdf = await PDFDocument.create();
|
|
9
|
+
for (const buffer of pdfs) {
|
|
10
|
+
const srcDoc = await PDFDocument.load(buffer);
|
|
11
|
+
const copiedPages = await mergedPdf.copyPages(srcDoc, srcDoc.getPageIndices());
|
|
12
|
+
copiedPages.forEach((page) => mergedPdf.addPage(page));
|
|
13
|
+
}
|
|
14
|
+
return mergedPdf.save();
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
const split = async (
|
|
18
|
+
pdf: ArrayBuffer | Uint8Array,
|
|
19
|
+
ranges: { start?: number; end?: number }[],
|
|
20
|
+
): Promise<Uint8Array[]> => {
|
|
21
|
+
if (!ranges.length) {
|
|
22
|
+
throw new Error('[@pdfme/manipulator] At least one range is required for splitting');
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const originalPdf = await PDFDocument.load(pdf);
|
|
26
|
+
const numPages = originalPdf.getPages().length;
|
|
27
|
+
const result: Uint8Array[] = [];
|
|
28
|
+
|
|
29
|
+
for (const { start = 0, end = numPages - 1 } of ranges) {
|
|
30
|
+
if (start < 0 || end >= numPages || start > end) {
|
|
31
|
+
throw new Error(
|
|
32
|
+
`[@pdfme/manipulator] Invalid range: start=${start}, end=${end}, total pages=${numPages}`,
|
|
33
|
+
);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const newPdf = await PDFDocument.create();
|
|
37
|
+
const pages = await newPdf.copyPages(
|
|
38
|
+
originalPdf,
|
|
39
|
+
Array.from({ length: end - start + 1 }, (_, i) => i + start),
|
|
40
|
+
);
|
|
41
|
+
pages.forEach((page) => newPdf.addPage(page));
|
|
42
|
+
result.push(await newPdf.save());
|
|
43
|
+
}
|
|
44
|
+
return result;
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
const remove = async (pdf: ArrayBuffer | Uint8Array, pages: number[]): Promise<Uint8Array> => {
|
|
48
|
+
if (!pages.length) {
|
|
49
|
+
throw new Error('[@pdfme/manipulator] At least one page number is required for removal');
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
const pdfDoc = await PDFDocument.load(pdf);
|
|
53
|
+
const numPages = pdfDoc.getPageCount();
|
|
54
|
+
|
|
55
|
+
if (pages.some((page) => page < 0 || page >= numPages)) {
|
|
56
|
+
throw new Error(
|
|
57
|
+
`[@pdfme/manipulator] Invalid page number: pages must be between 0 and ${numPages - 1}`,
|
|
58
|
+
);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
pages.sort((a, b) => b - a).forEach((pageIndex) => pdfDoc.removePage(pageIndex));
|
|
62
|
+
return pdfDoc.save();
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
const insert = async (
|
|
66
|
+
basePdf: ArrayBuffer | Uint8Array,
|
|
67
|
+
inserts: { pdf: ArrayBuffer | Uint8Array; position: number }[],
|
|
68
|
+
): Promise<Uint8Array> => {
|
|
69
|
+
inserts.sort((a, b) => a.position - b.position);
|
|
70
|
+
|
|
71
|
+
let currentPdf = basePdf;
|
|
72
|
+
let offset = 0;
|
|
73
|
+
|
|
74
|
+
for (let i = 0; i < inserts.length; i++) {
|
|
75
|
+
const { pdf, position } = inserts[i];
|
|
76
|
+
const actualPos = position + offset;
|
|
77
|
+
|
|
78
|
+
const basePdfDoc = await PDFDocument.load(currentPdf);
|
|
79
|
+
const insertDoc = await PDFDocument.load(pdf);
|
|
80
|
+
const numPages = basePdfDoc.getPageCount();
|
|
81
|
+
|
|
82
|
+
if (actualPos < 0 || actualPos > numPages) {
|
|
83
|
+
throw new Error(`[@pdfme/manipulator] Invalid position: must be between 0 and ${numPages}`);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
const newPdfDoc = await PDFDocument.create();
|
|
87
|
+
|
|
88
|
+
if (actualPos > 0) {
|
|
89
|
+
const beforePages = await newPdfDoc.copyPages(
|
|
90
|
+
basePdfDoc,
|
|
91
|
+
Array.from({ length: actualPos }, (_, idx) => idx),
|
|
92
|
+
);
|
|
93
|
+
beforePages.forEach((page) => newPdfDoc.addPage(page));
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
const insertPages = await newPdfDoc.copyPages(insertDoc, insertDoc.getPageIndices());
|
|
97
|
+
insertPages.forEach((page) => newPdfDoc.addPage(page));
|
|
98
|
+
|
|
99
|
+
if (actualPos < numPages) {
|
|
100
|
+
const afterPages = await newPdfDoc.copyPages(
|
|
101
|
+
basePdfDoc,
|
|
102
|
+
Array.from({ length: numPages - actualPos }, (_, idx) => idx + actualPos),
|
|
103
|
+
);
|
|
104
|
+
afterPages.forEach((page) => newPdfDoc.addPage(page));
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
currentPdf = await newPdfDoc.save();
|
|
108
|
+
|
|
109
|
+
offset += insertDoc.getPageCount();
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
const pdfDoc = await PDFDocument.load(currentPdf);
|
|
113
|
+
return pdfDoc.save();
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
const rotate = async (
|
|
117
|
+
pdf: ArrayBuffer | Uint8Array,
|
|
118
|
+
degrees: 0 | 90 | 180 | 270 | 360,
|
|
119
|
+
pageNumbers?: number[],
|
|
120
|
+
): Promise<Uint8Array> => {
|
|
121
|
+
if (!Number.isInteger(degrees) || degrees % 90 !== 0) {
|
|
122
|
+
throw new Error('[@pdfme/manipulator] Rotation degrees must be a multiple of 90');
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const pdfDoc = await PDFDocument.load(pdf);
|
|
126
|
+
const pages = pdfDoc.getPages();
|
|
127
|
+
|
|
128
|
+
if (!pages.length) {
|
|
129
|
+
throw new Error('[@pdfme/manipulator] PDF has no pages to rotate');
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
const normalizedDegrees = ((degrees % 360) + 360) % 360;
|
|
133
|
+
|
|
134
|
+
if (normalizedDegrees % 90 !== 0) {
|
|
135
|
+
throw new Error('[@pdfme/manipulator] Rotation degrees must be a multiple of 90');
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (pageNumbers) {
|
|
139
|
+
if (pageNumbers.some((page) => page < 0 || page >= pages.length)) {
|
|
140
|
+
throw new Error(
|
|
141
|
+
`[@pdfme/manipulator] Invalid page number: pages must be between 0 and ${pages.length - 1}`,
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const pagesToRotate = pageNumbers || pages.map((_, i) => i);
|
|
147
|
+
pagesToRotate.forEach((pageNum) => {
|
|
148
|
+
const page = pages[pageNum];
|
|
149
|
+
if (page) {
|
|
150
|
+
page.setRotation({
|
|
151
|
+
type: RotationTypes.Degrees,
|
|
152
|
+
angle: normalizedDegrees % 360,
|
|
153
|
+
});
|
|
154
|
+
}
|
|
155
|
+
});
|
|
156
|
+
return pdfDoc.save();
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
const move = async (
|
|
160
|
+
pdf: ArrayBuffer | Uint8Array,
|
|
161
|
+
operation: { from: number; to: number },
|
|
162
|
+
): Promise<Uint8Array> => {
|
|
163
|
+
const { from, to } = operation;
|
|
164
|
+
const pdfDoc = await PDFDocument.load(pdf);
|
|
165
|
+
const currentPageCount = pdfDoc.getPageCount();
|
|
166
|
+
|
|
167
|
+
if (from < 0 || from >= currentPageCount || to < 0 || to >= currentPageCount) {
|
|
168
|
+
throw new Error(
|
|
169
|
+
`[@pdfme/manipulator] Invalid page number: from=${from}, to=${to}, total pages=${currentPageCount}`,
|
|
170
|
+
);
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
if (from === to) {
|
|
174
|
+
return pdfDoc.save();
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
const page = pdfDoc.getPage(from);
|
|
178
|
+
pdfDoc.removePage(from);
|
|
179
|
+
|
|
180
|
+
const adjustedTo = from < to ? to - 1 : to;
|
|
181
|
+
pdfDoc.insertPage(adjustedTo, page);
|
|
182
|
+
|
|
183
|
+
return pdfDoc.save();
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
const organize = async (
|
|
187
|
+
pdf: ArrayBuffer | Uint8Array,
|
|
188
|
+
actions: Array<
|
|
189
|
+
| { type: 'remove'; data: { position: number } }
|
|
190
|
+
| { type: 'insert'; data: { pdf: ArrayBuffer | Uint8Array; position: number } }
|
|
191
|
+
| { type: 'replace'; data: { pdf: ArrayBuffer | Uint8Array; position: number } }
|
|
192
|
+
| { type: 'rotate'; data: { position: number; degrees: 0 | 90 | 180 | 270 | 360 } }
|
|
193
|
+
| { type: 'move'; data: { from: number; to: number } }
|
|
194
|
+
>,
|
|
195
|
+
): Promise<Uint8Array> => {
|
|
196
|
+
if (!actions.length) {
|
|
197
|
+
throw new Error('[@pdfme/manipulator] At least one action is required');
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
let currentPdf = await PDFDocument.load(pdf);
|
|
201
|
+
|
|
202
|
+
for (const action of actions) {
|
|
203
|
+
const currentBuffer = await currentPdf.save();
|
|
204
|
+
|
|
205
|
+
switch (action.type) {
|
|
206
|
+
case 'remove':
|
|
207
|
+
currentPdf = await PDFDocument.load(await remove(currentBuffer, [action.data.position]));
|
|
208
|
+
break;
|
|
209
|
+
|
|
210
|
+
case 'insert':
|
|
211
|
+
currentPdf = await PDFDocument.load(await insert(currentBuffer, [action.data]));
|
|
212
|
+
break;
|
|
213
|
+
|
|
214
|
+
case 'replace': {
|
|
215
|
+
const withoutTarget = await remove(currentBuffer, [action.data.position]);
|
|
216
|
+
currentPdf = await PDFDocument.load(await insert(withoutTarget, [action.data]));
|
|
217
|
+
break;
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
case 'rotate':
|
|
221
|
+
currentPdf = await PDFDocument.load(
|
|
222
|
+
await rotate(currentBuffer, action.data.degrees, [action.data.position]),
|
|
223
|
+
);
|
|
224
|
+
break;
|
|
225
|
+
|
|
226
|
+
case 'move':
|
|
227
|
+
currentPdf = await PDFDocument.load(await move(currentBuffer, action.data));
|
|
228
|
+
break;
|
|
229
|
+
|
|
230
|
+
default:
|
|
231
|
+
throw new Error(
|
|
232
|
+
`[@pdfme/manipulator] Unknown action type: ${(action as { type: string }).type}`,
|
|
233
|
+
);
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
return currentPdf.save();
|
|
238
|
+
};
|
|
239
|
+
|
|
240
|
+
export { merge, split, remove, insert, rotate, move, organize };
|