docxmlater 10.0.2 → 10.0.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 (398) hide show
  1. package/README.md +3 -2
  2. package/dist/constants/legacyCompatFlags.d.ts.map +1 -1
  3. package/dist/constants/legacyCompatFlags.js.map +1 -1
  4. package/dist/constants/limits.d.ts +0 -27
  5. package/dist/constants/limits.d.ts.map +1 -1
  6. package/dist/constants/limits.js +13 -13
  7. package/dist/constants/limits.js.map +1 -1
  8. package/dist/core/Document.d.ts +23 -19
  9. package/dist/core/Document.d.ts.map +1 -1
  10. package/dist/core/Document.js +197 -63
  11. package/dist/core/Document.js.map +1 -1
  12. package/dist/core/DocumentContent.d.ts.map +1 -1
  13. package/dist/core/DocumentContent.js.map +1 -1
  14. package/dist/core/DocumentGenerator.d.ts.map +1 -1
  15. package/dist/core/DocumentGenerator.js +59 -24
  16. package/dist/core/DocumentGenerator.js.map +1 -1
  17. package/dist/core/DocumentIdManager.d.ts.map +1 -1
  18. package/dist/core/DocumentIdManager.js.map +1 -1
  19. package/dist/core/DocumentParser.d.ts +6 -6
  20. package/dist/core/DocumentParser.d.ts.map +1 -1
  21. package/dist/core/DocumentParser.js +86 -55
  22. package/dist/core/DocumentParser.js.map +1 -1
  23. package/dist/core/DocumentValidator.d.ts.map +1 -1
  24. package/dist/core/DocumentValidator.js.map +1 -1
  25. package/dist/core/Relationship.d.ts.map +1 -1
  26. package/dist/core/Relationship.js +1 -1
  27. package/dist/core/Relationship.js.map +1 -1
  28. package/dist/core/RelationshipManager.js +3 -3
  29. package/dist/core/RelationshipManager.js.map +1 -1
  30. package/dist/elements/AlternateContent.js.map +1 -1
  31. package/dist/elements/Bookmark.d.ts.map +1 -1
  32. package/dist/elements/Bookmark.js.map +1 -1
  33. package/dist/elements/BookmarkManager.d.ts.map +1 -1
  34. package/dist/elements/BookmarkManager.js.map +1 -1
  35. package/dist/elements/Comment.js +1 -1
  36. package/dist/elements/Comment.js.map +1 -1
  37. package/dist/elements/CommentManager.d.ts.map +1 -1
  38. package/dist/elements/CommentManager.js +8 -2
  39. package/dist/elements/CommentManager.js.map +1 -1
  40. package/dist/elements/CommonTypes.d.ts.map +1 -1
  41. package/dist/elements/CommonTypes.js +1 -2
  42. package/dist/elements/CommonTypes.js.map +1 -1
  43. package/dist/elements/CustomXml.js.map +1 -1
  44. package/dist/elements/Endnote.d.ts.map +1 -1
  45. package/dist/elements/Endnote.js.map +1 -1
  46. package/dist/elements/EndnoteManager.d.ts.map +1 -1
  47. package/dist/elements/EndnoteManager.js.map +1 -1
  48. package/dist/elements/Field.d.ts.map +1 -1
  49. package/dist/elements/Field.js +31 -28
  50. package/dist/elements/Field.js.map +1 -1
  51. package/dist/elements/FieldHelpers.d.ts.map +1 -1
  52. package/dist/elements/FieldHelpers.js +6 -6
  53. package/dist/elements/FieldHelpers.js.map +1 -1
  54. package/dist/elements/FontManager.d.ts.map +1 -1
  55. package/dist/elements/FontManager.js.map +1 -1
  56. package/dist/elements/Footer.js.map +1 -1
  57. package/dist/elements/Footnote.d.ts.map +1 -1
  58. package/dist/elements/Footnote.js.map +1 -1
  59. package/dist/elements/FootnoteManager.d.ts.map +1 -1
  60. package/dist/elements/FootnoteManager.js.map +1 -1
  61. package/dist/elements/Header.js.map +1 -1
  62. package/dist/elements/HeaderFooterManager.js.map +1 -1
  63. package/dist/elements/Hyperlink.d.ts.map +1 -1
  64. package/dist/elements/Hyperlink.js +5 -5
  65. package/dist/elements/Hyperlink.js.map +1 -1
  66. package/dist/elements/Image.d.ts +2 -2
  67. package/dist/elements/Image.d.ts.map +1 -1
  68. package/dist/elements/Image.js +21 -5
  69. package/dist/elements/Image.js.map +1 -1
  70. package/dist/elements/ImageManager.d.ts.map +1 -1
  71. package/dist/elements/ImageManager.js +2 -2
  72. package/dist/elements/ImageManager.js.map +1 -1
  73. package/dist/elements/ImageRun.js.map +1 -1
  74. package/dist/elements/MathElement.js.map +1 -1
  75. package/dist/elements/Paragraph.d.ts +8 -0
  76. package/dist/elements/Paragraph.d.ts.map +1 -1
  77. package/dist/elements/Paragraph.js +153 -118
  78. package/dist/elements/Paragraph.js.map +1 -1
  79. package/dist/elements/PreservedElement.js.map +1 -1
  80. package/dist/elements/PropertyChangeTypes.js.map +1 -1
  81. package/dist/elements/RangeMarker.js.map +1 -1
  82. package/dist/elements/Revision.d.ts +1 -0
  83. package/dist/elements/Revision.d.ts.map +1 -1
  84. package/dist/elements/Revision.js +44 -5
  85. package/dist/elements/Revision.js.map +1 -1
  86. package/dist/elements/RevisionContent.js.map +1 -1
  87. package/dist/elements/RevisionManager.d.ts.map +1 -1
  88. package/dist/elements/RevisionManager.js.map +1 -1
  89. package/dist/elements/Run.d.ts.map +1 -1
  90. package/dist/elements/Run.js +1 -3
  91. package/dist/elements/Run.js.map +1 -1
  92. package/dist/elements/Section.d.ts.map +1 -1
  93. package/dist/elements/Section.js +127 -118
  94. package/dist/elements/Section.js.map +1 -1
  95. package/dist/elements/Shape.d.ts.map +1 -1
  96. package/dist/elements/Shape.js +21 -0
  97. package/dist/elements/Shape.js.map +1 -1
  98. package/dist/elements/StructuredDocumentTag.d.ts.map +1 -1
  99. package/dist/elements/StructuredDocumentTag.js +20 -8
  100. package/dist/elements/StructuredDocumentTag.js.map +1 -1
  101. package/dist/elements/Table.d.ts +2 -2
  102. package/dist/elements/Table.d.ts.map +1 -1
  103. package/dist/elements/Table.js +29 -35
  104. package/dist/elements/Table.js.map +1 -1
  105. package/dist/elements/TableCell.d.ts +2 -2
  106. package/dist/elements/TableCell.d.ts.map +1 -1
  107. package/dist/elements/TableCell.js +63 -67
  108. package/dist/elements/TableCell.js.map +1 -1
  109. package/dist/elements/TableGridChange.js.map +1 -1
  110. package/dist/elements/TableOfContents.d.ts +6 -6
  111. package/dist/elements/TableOfContents.d.ts.map +1 -1
  112. package/dist/elements/TableOfContents.js.map +1 -1
  113. package/dist/elements/TableOfContentsElement.js.map +1 -1
  114. package/dist/elements/TableRow.d.ts.map +1 -1
  115. package/dist/elements/TableRow.js +65 -47
  116. package/dist/elements/TableRow.js.map +1 -1
  117. package/dist/elements/TextBox.d.ts.map +1 -1
  118. package/dist/elements/TextBox.js +1 -1
  119. package/dist/elements/TextBox.js.map +1 -1
  120. package/dist/formatting/AbstractNumbering.d.ts +1 -1
  121. package/dist/formatting/AbstractNumbering.d.ts.map +1 -1
  122. package/dist/formatting/AbstractNumbering.js +11 -11
  123. package/dist/formatting/AbstractNumbering.js.map +1 -1
  124. package/dist/formatting/NumberingInstance.d.ts.map +1 -1
  125. package/dist/formatting/NumberingInstance.js +4 -4
  126. package/dist/formatting/NumberingInstance.js.map +1 -1
  127. package/dist/formatting/NumberingLevel.d.ts.map +1 -1
  128. package/dist/formatting/NumberingLevel.js +26 -26
  129. package/dist/formatting/NumberingLevel.js.map +1 -1
  130. package/dist/formatting/NumberingManager.d.ts +1 -1
  131. package/dist/formatting/NumberingManager.d.ts.map +1 -1
  132. package/dist/formatting/NumberingManager.js.map +1 -1
  133. package/dist/formatting/Style.d.ts.map +1 -1
  134. package/dist/formatting/Style.js +87 -95
  135. package/dist/formatting/Style.js.map +1 -1
  136. package/dist/formatting/StylesManager.d.ts +3 -3
  137. package/dist/formatting/StylesManager.d.ts.map +1 -1
  138. package/dist/formatting/StylesManager.js.map +1 -1
  139. package/dist/helpers/CleanupHelper.js.map +1 -1
  140. package/dist/images/ImageOptimizer.js.map +1 -1
  141. package/dist/index.js.map +1 -1
  142. package/dist/managers/DrawingManager.d.ts.map +1 -1
  143. package/dist/managers/DrawingManager.js.map +1 -1
  144. package/dist/tracking/DocumentTrackingContext.js.map +1 -1
  145. package/dist/tracking/TrackingContext.js.map +1 -1
  146. package/dist/types/compatibility-types.js.map +1 -1
  147. package/dist/types/formatting.js.map +1 -1
  148. package/dist/types/list-types.d.ts +4 -4
  149. package/dist/types/list-types.d.ts.map +1 -1
  150. package/dist/types/list-types.js.map +1 -1
  151. package/dist/types/settings-types.js.map +1 -1
  152. package/dist/types/styleConfig.js.map +1 -1
  153. package/dist/utils/ChangelogGenerator.d.ts.map +1 -1
  154. package/dist/utils/ChangelogGenerator.js.map +1 -1
  155. package/dist/utils/CompatibilityUpgrader.d.ts.map +1 -1
  156. package/dist/utils/CompatibilityUpgrader.js +7 -7
  157. package/dist/utils/CompatibilityUpgrader.js.map +1 -1
  158. package/dist/utils/InMemoryRevisionAcceptor.d.ts.map +1 -1
  159. package/dist/utils/InMemoryRevisionAcceptor.js +23 -2
  160. package/dist/utils/InMemoryRevisionAcceptor.js.map +1 -1
  161. package/dist/utils/MoveOperationHelper.js.map +1 -1
  162. package/dist/utils/RevisionAwareProcessor.js.map +1 -1
  163. package/dist/utils/RevisionWalker.js.map +1 -1
  164. package/dist/utils/SelectiveRevisionAcceptor.d.ts +1 -0
  165. package/dist/utils/SelectiveRevisionAcceptor.d.ts.map +1 -1
  166. package/dist/utils/SelectiveRevisionAcceptor.js +46 -0
  167. package/dist/utils/SelectiveRevisionAcceptor.js.map +1 -1
  168. package/dist/utils/ShadingResolver.js +1 -1
  169. package/dist/utils/ShadingResolver.js.map +1 -1
  170. package/dist/utils/acceptRevisions.d.ts +0 -28
  171. package/dist/utils/acceptRevisions.d.ts.map +1 -1
  172. package/dist/utils/acceptRevisions.js +5 -7
  173. package/dist/utils/acceptRevisions.js.map +1 -1
  174. package/dist/utils/cnfStyleDecoder.js +1 -1
  175. package/dist/utils/cnfStyleDecoder.js.map +1 -1
  176. package/dist/utils/corruptionDetection.js.map +1 -1
  177. package/dist/utils/dateFormatting.js.map +1 -1
  178. package/dist/utils/deepClone.d.ts +0 -1
  179. package/dist/utils/deepClone.d.ts.map +1 -1
  180. package/dist/utils/deepClone.js +0 -7
  181. package/dist/utils/deepClone.js.map +1 -1
  182. package/dist/utils/diagnostics.d.ts +2 -2
  183. package/dist/utils/diagnostics.d.ts.map +1 -1
  184. package/dist/utils/diagnostics.js.map +1 -1
  185. package/dist/utils/errorHandling.js.map +1 -1
  186. package/dist/utils/formatting.js.map +1 -1
  187. package/dist/utils/list-detection.d.ts +2 -2
  188. package/dist/utils/list-detection.d.ts.map +1 -1
  189. package/dist/utils/list-detection.js +3 -3
  190. package/dist/utils/list-detection.js.map +1 -1
  191. package/dist/utils/logger.d.ts +2 -4
  192. package/dist/utils/logger.d.ts.map +1 -1
  193. package/dist/utils/logger.js +0 -2
  194. package/dist/utils/logger.js.map +1 -1
  195. package/dist/utils/parsingHelpers.js.map +1 -1
  196. package/dist/utils/stripTrackedChanges.d.ts +0 -19
  197. package/dist/utils/stripTrackedChanges.d.ts.map +1 -1
  198. package/dist/utils/stripTrackedChanges.js +0 -2
  199. package/dist/utils/stripTrackedChanges.js.map +1 -1
  200. package/dist/utils/textDiff.js.map +1 -1
  201. package/dist/utils/units.js.map +1 -1
  202. package/dist/utils/validation.d.ts.map +1 -1
  203. package/dist/utils/validation.js.map +1 -1
  204. package/dist/utils/xmlSanitization.js.map +1 -1
  205. package/dist/validation/RevisionAutoFixer.js.map +1 -1
  206. package/dist/validation/RevisionValidator.js.map +1 -1
  207. package/dist/validation/ValidationRules.js.map +1 -1
  208. package/dist/validation/index.js.map +1 -1
  209. package/dist/xml/XMLBuilder.d.ts.map +1 -1
  210. package/dist/xml/XMLBuilder.js +10 -0
  211. package/dist/xml/XMLBuilder.js.map +1 -1
  212. package/dist/xml/XMLParser.d.ts.map +1 -1
  213. package/dist/xml/XMLParser.js +4 -5
  214. package/dist/xml/XMLParser.js.map +1 -1
  215. package/dist/zip/ZipHandler.js.map +1 -1
  216. package/dist/zip/ZipReader.js.map +1 -1
  217. package/dist/zip/ZipWriter.js.map +1 -1
  218. package/dist/zip/errors.js.map +1 -1
  219. package/dist/zip/types.js.map +1 -1
  220. package/package.json +34 -4
  221. package/src/__tests__/helper-methods.test.ts +512 -0
  222. package/src/constants/legacyCompatFlags.ts +138 -0
  223. package/src/constants/limits.ts +50 -0
  224. package/src/core/CLAUDE.md +109 -0
  225. package/src/core/Document.ts +15569 -0
  226. package/src/core/DocumentContent.ts +467 -0
  227. package/src/core/DocumentGenerator.ts +1104 -0
  228. package/src/core/DocumentIdManager.ts +158 -0
  229. package/src/core/DocumentParser.ts +10140 -0
  230. package/src/core/DocumentValidator.ts +372 -0
  231. package/src/core/Relationship.ts +367 -0
  232. package/src/core/RelationshipManager.ts +428 -0
  233. package/src/elements/AlternateContent.ts +42 -0
  234. package/src/elements/Bookmark.ts +210 -0
  235. package/src/elements/BookmarkManager.ts +250 -0
  236. package/src/elements/CLAUDE.md +126 -0
  237. package/src/elements/Comment.ts +359 -0
  238. package/src/elements/CommentManager.ts +502 -0
  239. package/src/elements/CommonTypes.ts +549 -0
  240. package/src/elements/CustomXml.ts +36 -0
  241. package/src/elements/Endnote.ts +217 -0
  242. package/src/elements/EndnoteManager.ts +249 -0
  243. package/src/elements/Field.ts +1233 -0
  244. package/src/elements/FieldHelpers.ts +333 -0
  245. package/src/elements/FontManager.ts +339 -0
  246. package/src/elements/Footer.ts +269 -0
  247. package/src/elements/Footnote.ts +217 -0
  248. package/src/elements/FootnoteManager.ts +249 -0
  249. package/src/elements/Header.ts +269 -0
  250. package/src/elements/HeaderFooterManager.ts +219 -0
  251. package/src/elements/Hyperlink.ts +1146 -0
  252. package/src/elements/Image.ts +1756 -0
  253. package/src/elements/ImageManager.ts +432 -0
  254. package/src/elements/ImageRun.ts +59 -0
  255. package/src/elements/MathElement.ts +65 -0
  256. package/src/elements/Paragraph.ts +4287 -0
  257. package/src/elements/PreservedElement.ts +53 -0
  258. package/src/elements/PropertyChangeTypes.ts +442 -0
  259. package/src/elements/RangeMarker.ts +400 -0
  260. package/src/elements/Revision.ts +1217 -0
  261. package/src/elements/RevisionContent.ts +73 -0
  262. package/src/elements/RevisionManager.ts +1070 -0
  263. package/src/elements/Run.ts +3068 -0
  264. package/src/elements/Section.ts +1421 -0
  265. package/src/elements/Shape.ts +873 -0
  266. package/src/elements/StructuredDocumentTag.ts +978 -0
  267. package/src/elements/Table.ts +2524 -0
  268. package/src/elements/TableCell.ts +1586 -0
  269. package/src/elements/TableGridChange.ts +151 -0
  270. package/src/elements/TableOfContents.ts +691 -0
  271. package/src/elements/TableOfContentsElement.ts +89 -0
  272. package/src/elements/TableRow.ts +906 -0
  273. package/src/elements/TextBox.ts +768 -0
  274. package/src/formatting/AbstractNumbering.ts +548 -0
  275. package/src/formatting/CLAUDE.md +74 -0
  276. package/src/formatting/NumberingInstance.ts +212 -0
  277. package/src/formatting/NumberingLevel.ts +1006 -0
  278. package/src/formatting/NumberingManager.ts +827 -0
  279. package/src/formatting/Style.ts +1833 -0
  280. package/src/formatting/StylesManager.ts +1005 -0
  281. package/src/helpers/CleanupHelper.ts +524 -0
  282. package/src/images/ImageOptimizer.ts +274 -0
  283. package/src/index.ts +554 -0
  284. package/src/managers/CLAUDE.md +47 -0
  285. package/src/managers/DrawingManager.ts +319 -0
  286. package/src/tracking/DocumentTrackingContext.ts +643 -0
  287. package/src/tracking/TrackingContext.ts +173 -0
  288. package/src/types/compatibility-types.ts +49 -0
  289. package/src/types/formatting.ts +210 -0
  290. package/src/types/list-types.ts +152 -0
  291. package/src/types/settings-types.ts +59 -0
  292. package/src/types/styleConfig.ts +189 -0
  293. package/src/utils/CLAUDE.md +153 -0
  294. package/src/utils/ChangelogGenerator.ts +1581 -0
  295. package/src/utils/CompatibilityUpgrader.ts +237 -0
  296. package/src/utils/InMemoryRevisionAcceptor.ts +696 -0
  297. package/src/utils/MoveOperationHelper.ts +238 -0
  298. package/src/utils/RevisionAwareProcessor.ts +526 -0
  299. package/src/utils/RevisionWalker.ts +457 -0
  300. package/src/utils/SelectiveRevisionAcceptor.ts +683 -0
  301. package/src/utils/ShadingResolver.ts +107 -0
  302. package/src/utils/acceptRevisions.ts +714 -0
  303. package/src/utils/cnfStyleDecoder.ts +217 -0
  304. package/src/utils/corruptionDetection.ts +345 -0
  305. package/src/utils/dateFormatting.ts +20 -0
  306. package/src/utils/deepClone.ts +78 -0
  307. package/src/utils/diagnostics.ts +129 -0
  308. package/src/utils/errorHandling.ts +80 -0
  309. package/src/utils/formatting.ts +213 -0
  310. package/src/utils/list-detection.ts +274 -0
  311. package/src/utils/logger.ts +404 -0
  312. package/src/utils/parsingHelpers.ts +190 -0
  313. package/src/utils/stripTrackedChanges.ts +353 -0
  314. package/src/utils/textDiff.ts +100 -0
  315. package/src/utils/units.ts +421 -0
  316. package/src/utils/validation.ts +542 -0
  317. package/src/utils/xmlSanitization.ts +182 -0
  318. package/src/validation/RevisionAutoFixer.ts +542 -0
  319. package/src/validation/RevisionValidator.ts +460 -0
  320. package/src/validation/ValidationRules.ts +338 -0
  321. package/src/validation/index.ts +30 -0
  322. package/src/xml/CLAUDE.md +65 -0
  323. package/src/xml/XMLBuilder.ts +871 -0
  324. package/src/xml/XMLParser.ts +919 -0
  325. package/src/zip/CLAUDE.md +55 -0
  326. package/src/zip/ZipHandler.ts +637 -0
  327. package/src/zip/ZipReader.ts +299 -0
  328. package/src/zip/ZipWriter.ts +390 -0
  329. package/src/zip/errors.ts +69 -0
  330. package/src/zip/types.ts +116 -0
  331. package/dist/core/ListNormalizer.d.ts +0 -23
  332. package/dist/core/ListNormalizer.d.ts.map +0 -1
  333. package/dist/core/ListNormalizer.js +0 -624
  334. package/dist/core/ListNormalizer.js.map +0 -1
  335. package/dist/images/index.d.ts +0 -2
  336. package/dist/images/index.d.ts.map +0 -1
  337. package/dist/images/index.js +0 -8
  338. package/dist/images/index.js.map +0 -1
  339. package/dist/ms-doc/cfb/CFBReader.d.ts +0 -35
  340. package/dist/ms-doc/cfb/CFBReader.d.ts.map +0 -1
  341. package/dist/ms-doc/cfb/CFBReader.js +0 -360
  342. package/dist/ms-doc/cfb/CFBReader.js.map +0 -1
  343. package/dist/ms-doc/converter/DocToDocxConverter.d.ts +0 -55
  344. package/dist/ms-doc/converter/DocToDocxConverter.d.ts.map +0 -1
  345. package/dist/ms-doc/converter/DocToDocxConverter.js +0 -324
  346. package/dist/ms-doc/converter/DocToDocxConverter.js.map +0 -1
  347. package/dist/ms-doc/fib/FIB.d.ts +0 -18
  348. package/dist/ms-doc/fib/FIB.d.ts.map +0 -1
  349. package/dist/ms-doc/fib/FIB.js +0 -342
  350. package/dist/ms-doc/fib/FIB.js.map +0 -1
  351. package/dist/ms-doc/fields/FieldParser.d.ts +0 -31
  352. package/dist/ms-doc/fields/FieldParser.d.ts.map +0 -1
  353. package/dist/ms-doc/fields/FieldParser.js +0 -266
  354. package/dist/ms-doc/fields/FieldParser.js.map +0 -1
  355. package/dist/ms-doc/images/PictureExtractor.d.ts +0 -22
  356. package/dist/ms-doc/images/PictureExtractor.d.ts.map +0 -1
  357. package/dist/ms-doc/images/PictureExtractor.js +0 -233
  358. package/dist/ms-doc/images/PictureExtractor.js.map +0 -1
  359. package/dist/ms-doc/index.d.ts +0 -20
  360. package/dist/ms-doc/index.d.ts.map +0 -1
  361. package/dist/ms-doc/index.js +0 -59
  362. package/dist/ms-doc/index.js.map +0 -1
  363. package/dist/ms-doc/properties/SPRM.d.ts +0 -210
  364. package/dist/ms-doc/properties/SPRM.d.ts.map +0 -1
  365. package/dist/ms-doc/properties/SPRM.js +0 -633
  366. package/dist/ms-doc/properties/SPRM.js.map +0 -1
  367. package/dist/ms-doc/sections/SectionParser.d.ts +0 -25
  368. package/dist/ms-doc/sections/SectionParser.d.ts.map +0 -1
  369. package/dist/ms-doc/sections/SectionParser.js +0 -214
  370. package/dist/ms-doc/sections/SectionParser.js.map +0 -1
  371. package/dist/ms-doc/styles/StyleSheet.d.ts +0 -23
  372. package/dist/ms-doc/styles/StyleSheet.d.ts.map +0 -1
  373. package/dist/ms-doc/styles/StyleSheet.js +0 -268
  374. package/dist/ms-doc/styles/StyleSheet.js.map +0 -1
  375. package/dist/ms-doc/subdocuments/SubdocumentParser.d.ts +0 -61
  376. package/dist/ms-doc/subdocuments/SubdocumentParser.d.ts.map +0 -1
  377. package/dist/ms-doc/subdocuments/SubdocumentParser.js +0 -208
  378. package/dist/ms-doc/subdocuments/SubdocumentParser.js.map +0 -1
  379. package/dist/ms-doc/tables/TableParser.d.ts +0 -29
  380. package/dist/ms-doc/tables/TableParser.d.ts.map +0 -1
  381. package/dist/ms-doc/tables/TableParser.js +0 -176
  382. package/dist/ms-doc/tables/TableParser.js.map +0 -1
  383. package/dist/ms-doc/text/PieceTable.d.ts +0 -21
  384. package/dist/ms-doc/text/PieceTable.d.ts.map +0 -1
  385. package/dist/ms-doc/text/PieceTable.js +0 -171
  386. package/dist/ms-doc/text/PieceTable.js.map +0 -1
  387. package/dist/ms-doc/types/Constants.d.ts +0 -99
  388. package/dist/ms-doc/types/Constants.d.ts.map +0 -1
  389. package/dist/ms-doc/types/Constants.js +0 -102
  390. package/dist/ms-doc/types/Constants.js.map +0 -1
  391. package/dist/ms-doc/types/DocTypes.d.ts +0 -368
  392. package/dist/ms-doc/types/DocTypes.d.ts.map +0 -1
  393. package/dist/ms-doc/types/DocTypes.js +0 -3
  394. package/dist/ms-doc/types/DocTypes.js.map +0 -1
  395. package/dist/tracking/index.d.ts +0 -3
  396. package/dist/tracking/index.d.ts.map +0 -1
  397. package/dist/tracking/index.js +0 -6
  398. package/dist/tracking/index.js.map +0 -1
@@ -0,0 +1,1104 @@
1
+ /**
2
+ * DocumentGenerator - Handles XML generation for DOCX files
3
+ * Converts structured data to OpenXML format
4
+ */
5
+
6
+ import { CommentManager } from "../elements/CommentManager";
7
+ import { EndnoteManager } from "../elements/EndnoteManager";
8
+ import { FontManager } from "../elements/FontManager";
9
+ import { FootnoteManager } from "../elements/FootnoteManager";
10
+ import { HeaderFooterManager } from "../elements/HeaderFooterManager";
11
+ import { Hyperlink } from "../elements/Hyperlink";
12
+ import { ImageManager } from "../elements/ImageManager";
13
+ import { Paragraph } from "../elements/Paragraph";
14
+ import { Revision } from "../elements/Revision";
15
+ import { isHyperlinkContent } from "../elements/RevisionContent";
16
+ import { Section } from "../elements/Section";
17
+ import { StructuredDocumentTag } from "../elements/StructuredDocumentTag";
18
+ import { Table } from "../elements/Table";
19
+ import { TableOfContentsElement } from "../elements/TableOfContentsElement";
20
+ import { AlternateContent } from "../elements/AlternateContent";
21
+ import { MathParagraph } from "../elements/MathElement";
22
+ import { CustomXmlBlock } from "../elements/CustomXml";
23
+ import { PreservedElement } from "../elements/PreservedElement";
24
+ import { formatDateForXml } from "../utils/dateFormatting";
25
+ import { getGlobalLogger, createScopedLogger, ILogger } from "../utils/logger";
26
+ import { XMLBuilder, XMLElement } from "../xml/XMLBuilder";
27
+ import { DocumentProperties } from "./Document";
28
+ import { BodyElement } from "./DocumentContent";
29
+ import { RelationshipManager } from "./RelationshipManager";
30
+ import { TrackChangesSettings } from "../types/settings-types";
31
+
32
+ // Create scoped logger for DocumentGenerator operations
33
+ function getLogger(): ILogger {
34
+ return createScopedLogger(getGlobalLogger(), 'DocumentGenerator');
35
+ }
36
+
37
+ /**
38
+ * Interface for ZipHandler methods used in content type generation
39
+ * This provides type safety for the ZipHandler parameter without creating
40
+ * a circular dependency with the zip module.
41
+ */
42
+ export interface IZipHandlerReader {
43
+ /** Get list of file paths in the archive */
44
+ getFilePaths?(): string[];
45
+ /** Check if a file exists in the archive */
46
+ hasFile?(path: string): boolean;
47
+ }
48
+
49
+ /**
50
+ * Normalizes toXML() output to always return an array.
51
+ * Some elements (e.g., TableOfContentsElement) return XMLElement[], while others return XMLElement.
52
+ * This helper provides consistent array handling.
53
+ *
54
+ * @param xml - XMLElement or XMLElement[] from toXML()
55
+ * @returns XMLElement array
56
+ */
57
+ function normalizeXmlOutput(xml: XMLElement | XMLElement[]): XMLElement[] {
58
+ return Array.isArray(xml) ? xml : [xml];
59
+ }
60
+
61
+ /**
62
+ * DocumentGenerator handles all XML generation logic
63
+ */
64
+ export class DocumentGenerator {
65
+ /**
66
+ * Generates [Content_Types].xml
67
+ */
68
+ generateContentTypes(): string {
69
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
70
+ <Types xmlns="http://schemas.openxmlformats.org/package/2006/content-types">
71
+ <Default Extension="rels" ContentType="application/vnd.openxmlformats-package.relationships+xml"/>
72
+ <Default Extension="xml" ContentType="application/xml"/>
73
+ <Override PartName="/word/document.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.document.main+xml"/>
74
+ <Override PartName="/word/styles.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.styles+xml"/>
75
+ <Override PartName="/word/numbering.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.numbering+xml"/>
76
+ <Override PartName="/word/fontTable.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.fontTable+xml"/>
77
+ <Override PartName="/word/settings.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.settings+xml"/>
78
+ <Override PartName="/word/webSettings.xml" ContentType="application/vnd.openxmlformats-officedocument.wordprocessingml.webSettings+xml"/>
79
+ <Override PartName="/word/theme/theme1.xml" ContentType="application/vnd.openxmlformats-officedocument.theme+xml"/>
80
+ <Override PartName="/docProps/core.xml" ContentType="application/vnd.openxmlformats-package.core-properties+xml"/>
81
+ <Override PartName="/docProps/app.xml" ContentType="application/vnd.openxmlformats-officedocument.extended-properties+xml"/>
82
+ </Types>`;
83
+ }
84
+
85
+ /**
86
+ * Generates _rels/.rels
87
+ */
88
+ generateRels(): string {
89
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
90
+ <Relationships xmlns="http://schemas.openxmlformats.org/package/2006/relationships">
91
+ <Relationship Id="rId1" Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/officeDocument" Target="word/document.xml"/>
92
+ <Relationship Id="rId2" Type="http://schemas.openxmlformats.org/package/2006/relationships/metadata/core-properties" Target="docProps/core.xml"/>
93
+ <Relationship Id="rId3" Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/extended-properties" Target="docProps/app.xml"/>
94
+ </Relationships>`;
95
+ }
96
+
97
+ /**
98
+ * Generates word/document.xml with current body elements
99
+ */
100
+ generateDocumentXml(
101
+ bodyElements: BodyElement[],
102
+ section: Section,
103
+ namespaces: Record<string, string>,
104
+ documentBackground?: { color?: string; themeColor?: string; themeTint?: string; themeShade?: string }
105
+ ): string {
106
+ const logger = getLogger();
107
+ logger.info('Generating document.xml', { elementCount: bodyElements.length });
108
+
109
+ const bodyXmls: XMLElement[] = [];
110
+
111
+ // Generate XML for each body element
112
+ // Uses normalizeXmlOutput() to handle both single XMLElement and XMLElement[] returns
113
+ for (const element of bodyElements) {
114
+ const xmlElements = normalizeXmlOutput(element.toXML());
115
+ bodyXmls.push(...xmlElements);
116
+ }
117
+
118
+ // Add section properties at the end
119
+ bodyXmls.push(section.toXML());
120
+
121
+ // Build pre-body content (w:background) per ECMA-376 Part 1 §17.2.1
122
+ let preBodyContent: XMLElement[] | undefined;
123
+ if (documentBackground) {
124
+ const bgAttrs: Record<string, string> = {};
125
+ if (documentBackground.color) bgAttrs["w:color"] = documentBackground.color;
126
+ if (documentBackground.themeColor) bgAttrs["w:themeColor"] = documentBackground.themeColor;
127
+ if (documentBackground.themeTint) bgAttrs["w:themeTint"] = documentBackground.themeTint;
128
+ if (documentBackground.themeShade) bgAttrs["w:themeShade"] = documentBackground.themeShade;
129
+ preBodyContent = [XMLBuilder.wSelf("background", bgAttrs)];
130
+ }
131
+
132
+ const result = XMLBuilder.createDocument(bodyXmls, namespaces, preBodyContent);
133
+ logger.info('Document.xml generated', { xmlSize: result.length });
134
+ return result;
135
+ }
136
+
137
+ /**
138
+ * Generates docProps/core.xml with extended properties
139
+ */
140
+ generateCoreProps(properties: DocumentProperties): string {
141
+ const now = new Date();
142
+ const created = properties.created || now;
143
+ const modified = properties.modified || now;
144
+
145
+ const formatDate = (date: Date): string => {
146
+ return formatDateForXml(date);
147
+ };
148
+
149
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
150
+ <cp:coreProperties xmlns:cp="http://schemas.openxmlformats.org/package/2006/metadata/core-properties"
151
+ xmlns:dc="http://purl.org/dc/elements/1.1/"
152
+ xmlns:dcterms="http://purl.org/dc/terms/"
153
+ xmlns:dcmitype="http://purl.org/dc/dcmitype/"
154
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
155
+ <dc:title>${XMLBuilder.sanitizeXmlContent(properties.title || "")}</dc:title>
156
+ <dc:subject>${XMLBuilder.sanitizeXmlContent(
157
+ properties.subject || ""
158
+ )}</dc:subject>
159
+ <dc:creator>${XMLBuilder.sanitizeXmlContent(
160
+ properties.creator || "DocXML"
161
+ )}</dc:creator>
162
+ <cp:keywords>${XMLBuilder.sanitizeXmlContent(
163
+ properties.keywords || ""
164
+ )}</cp:keywords>
165
+ <dc:description>${XMLBuilder.sanitizeXmlContent(
166
+ properties.description || ""
167
+ )}</dc:description>
168
+ <cp:lastModifiedBy>${XMLBuilder.sanitizeXmlContent(
169
+ properties.lastModifiedBy || properties.creator || "DocXML"
170
+ )}</cp:lastModifiedBy>
171
+ <cp:revision>${properties.revision || 1}</cp:revision>${
172
+ properties.category
173
+ ? `\n <cp:category>${XMLBuilder.sanitizeXmlContent(
174
+ properties.category
175
+ )}</cp:category>`
176
+ : ""
177
+ }${
178
+ properties.contentStatus
179
+ ? `\n <cp:contentStatus>${XMLBuilder.sanitizeXmlContent(
180
+ properties.contentStatus
181
+ )}</cp:contentStatus>`
182
+ : ""
183
+ }${
184
+ properties.language
185
+ ? `\n <dc:language>${XMLBuilder.sanitizeXmlContent(
186
+ properties.language
187
+ )}</dc:language>`
188
+ : ""
189
+ }
190
+ <dcterms:created xsi:type="dcterms:W3CDTF">${formatDate(
191
+ created
192
+ )}</dcterms:created>
193
+ <dcterms:modified xsi:type="dcterms:W3CDTF">${formatDate(
194
+ modified
195
+ )}</dcterms:modified>
196
+ </cp:coreProperties>`;
197
+ }
198
+
199
+ /**
200
+ * Generates docProps/app.xml with extended properties
201
+ */
202
+ generateAppProps(properties: DocumentProperties = {}): string {
203
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
204
+ <Properties xmlns="http://schemas.openxmlformats.org/officeDocument/2006/extended-properties"
205
+ xmlns:vt="http://schemas.openxmlformats.org/officeDocument/2006/docPropsVTypes">
206
+ <Application>${XMLBuilder.sanitizeXmlContent(
207
+ properties.application || "docxmlater"
208
+ )}</Application>
209
+ <DocSecurity>0</DocSecurity>
210
+ <ScaleCrop>false</ScaleCrop>
211
+ <Company>${XMLBuilder.sanitizeXmlContent(properties.company || "")}</Company>${
212
+ properties.manager
213
+ ? `\n <Manager>${XMLBuilder.sanitizeXmlContent(
214
+ properties.manager
215
+ )}</Manager>`
216
+ : ""
217
+ }
218
+ <LinksUpToDate>false</LinksUpToDate>
219
+ <SharedDoc>false</SharedDoc>
220
+ <HyperlinksChanged>false</HyperlinksChanged>
221
+ <AppVersion>${XMLBuilder.sanitizeXmlContent(
222
+ properties.appVersion || properties.version || "1.0.0"
223
+ )}</AppVersion>
224
+ </Properties>`;
225
+ }
226
+
227
+ /**
228
+ * Generates docProps/custom.xml with custom properties
229
+ */
230
+ generateCustomProps(
231
+ customProps: Record<string, string | number | boolean | Date>
232
+ ): string {
233
+ if (!customProps || Object.keys(customProps).length === 0) {
234
+ return "";
235
+ }
236
+
237
+ const formatCustomValue = (
238
+ key: string,
239
+ value: string | number | boolean | Date,
240
+ pid: number
241
+ ): string => {
242
+ if (typeof value === "string") {
243
+ return ` <property fmtid="{D5CDD505-2E9C-101B-9397-08002B2CF9AE}" pid="${pid}" name="${XMLBuilder.sanitizeXmlContent(
244
+ key
245
+ )}">
246
+ <vt:lpwstr>${XMLBuilder.sanitizeXmlContent(value)}</vt:lpwstr>
247
+ </property>`;
248
+ } else if (typeof value === "number") {
249
+ return ` <property fmtid="{D5CDD505-2E9C-101B-9397-08002B2CF9AE}" pid="${pid}" name="${XMLBuilder.sanitizeXmlContent(
250
+ key
251
+ )}">
252
+ <vt:r8>${value}</vt:r8>
253
+ </property>`;
254
+ } else if (typeof value === "boolean") {
255
+ return ` <property fmtid="{D5CDD505-2E9C-101B-9397-08002B2CF9AE}" pid="${pid}" name="${XMLBuilder.sanitizeXmlContent(
256
+ key
257
+ )}">
258
+ <vt:bool>${value ? "true" : "false"}</vt:bool>
259
+ </property>`;
260
+ } else if (value instanceof Date) {
261
+ return ` <property fmtid="{D5CDD505-2E9C-101B-9397-08002B2CF9AE}" pid="${pid}" name="${XMLBuilder.sanitizeXmlContent(
262
+ key
263
+ )}">
264
+ <vt:filetime>${value.toISOString()}</vt:filetime>
265
+ </property>`;
266
+ }
267
+ return "";
268
+ };
269
+
270
+ const properties = Object.entries(customProps)
271
+ .map(([key, value], index) => formatCustomValue(key, value, index + 2))
272
+ .filter((prop) => prop !== "")
273
+ .join("\n");
274
+
275
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
276
+ <Properties xmlns="http://schemas.openxmlformats.org/officeDocument/2006/custom-properties"
277
+ xmlns:vt="http://schemas.openxmlformats.org/officeDocument/2006/docPropsVTypes">
278
+ ${properties}
279
+ </Properties>`;
280
+ }
281
+
282
+ /**
283
+ * Generates [Content_Types].xml with image extensions, headers/footers, comments, and fonts
284
+ * Preserves entries for files that exist in the loaded document (customXML, etc.)
285
+ * Merges framework-generated entries with original entries for round-trip fidelity
286
+ */
287
+ generateContentTypesWithImagesHeadersFootersAndComments(
288
+ imageManager: ImageManager,
289
+ headerFooterManager: HeaderFooterManager,
290
+ commentManager: CommentManager,
291
+ zipHandler: IZipHandlerReader,
292
+ fontManager?: FontManager,
293
+ hasCustomProperties = false,
294
+ originalContentTypes?: { defaults: Set<string>; overrides: Set<string> },
295
+ footnoteManager?: FootnoteManager,
296
+ endnoteManager?: EndnoteManager
297
+ ): string {
298
+ const images = imageManager.getAllImages();
299
+ const headers = headerFooterManager.getAllHeaders();
300
+ const footers = headerFooterManager.getAllFooters();
301
+ const hasComments = commentManager.getCount() > 0;
302
+
303
+ // Build sets for framework-generated entries
304
+ const generatedDefaults = new Set<string>();
305
+ const generatedOverrides = new Set<string>();
306
+
307
+ // Default types - always needed
308
+ generatedDefaults.add('rels|application/vnd.openxmlformats-package.relationships+xml');
309
+ generatedDefaults.add('xml|application/xml');
310
+
311
+ // Image extensions from ImageManager
312
+ for (const entry of images) {
313
+ const ext = entry.image.getExtension();
314
+ const mimeType = ImageManager.getMimeType(ext);
315
+ generatedDefaults.add(`${ext}|${mimeType}`);
316
+ }
317
+
318
+ // Also detect image files in the archive not tracked by ImageManager
319
+ // (e.g., numPicBullet images referenced by numbering.xml.rels)
320
+ const mediaExtensions = new Map<string, string>([
321
+ ['png', 'image/png'], ['jpeg', 'image/jpeg'], ['jpg', 'image/jpeg'],
322
+ ['gif', 'image/gif'], ['bmp', 'image/bmp'], ['tiff', 'image/tiff'],
323
+ ['emf', 'image/x-emf'], ['wmf', 'image/x-wmf'],
324
+ ]);
325
+ for (const file of (zipHandler.getFilePaths?.() || [])) {
326
+ if (file.startsWith('word/media/')) {
327
+ const ext = file.split('.').pop()?.toLowerCase();
328
+ if (ext && mediaExtensions.has(ext)) {
329
+ generatedDefaults.add(`${ext}|${mediaExtensions.get(ext)}`);
330
+ }
331
+ }
332
+ }
333
+
334
+ // Font extensions (if FontManager provided)
335
+ if (fontManager && fontManager.getCount() > 0) {
336
+ const fontEntries = fontManager.generateContentTypeEntries();
337
+ for (const entry of fontEntries) {
338
+ // Parse each entry and add to set (entries are XML strings)
339
+ const extMatch = /Extension="([^"]+)"/.exec(entry);
340
+ const typeMatch = /ContentType="([^"]+)"/.exec(entry);
341
+ if (extMatch && typeMatch) {
342
+ generatedDefaults.add(`${extMatch[1]}|${typeMatch[1]}`);
343
+ }
344
+ }
345
+ }
346
+
347
+ // Check for embedded .ttf fonts from original document
348
+ // Also create a Set for efficient file existence checks
349
+ const files = zipHandler.getFilePaths?.() || [];
350
+ const filesInArchive = new Set(files);
351
+ const hasTtfFonts = files.some((f: string) => f.endsWith(".ttf"));
352
+ if (hasTtfFonts) {
353
+ generatedDefaults.add('ttf|application/x-font-ttf');
354
+ }
355
+
356
+ // Override types - only add if file exists in archive
357
+ generatedOverrides.add('/word/document.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.document.main+xml');
358
+ if (filesInArchive.has('word/styles.xml')) {
359
+ generatedOverrides.add('/word/styles.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.styles+xml');
360
+ }
361
+ if (filesInArchive.has('word/numbering.xml')) {
362
+ generatedOverrides.add('/word/numbering.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.numbering+xml');
363
+ }
364
+ if (filesInArchive.has('word/fontTable.xml')) {
365
+ generatedOverrides.add('/word/fontTable.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.fontTable+xml');
366
+ }
367
+ if (filesInArchive.has('word/settings.xml')) {
368
+ generatedOverrides.add('/word/settings.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.settings+xml');
369
+ }
370
+ if (filesInArchive.has('word/webSettings.xml')) {
371
+ generatedOverrides.add('/word/webSettings.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.webSettings+xml');
372
+ }
373
+ if (filesInArchive.has('word/theme/theme1.xml')) {
374
+ generatedOverrides.add('/word/theme/theme1.xml|application/vnd.openxmlformats-officedocument.theme+xml');
375
+ }
376
+
377
+ // Headers - only add if file actually exists in archive
378
+ // This prevents corruption when HeaderFooterManager has stale entries for removed headers
379
+ for (const entry of headers) {
380
+ const filePath = `word/${entry.filename}`;
381
+ if (filesInArchive.has(filePath)) {
382
+ generatedOverrides.add(`/word/${entry.filename}|application/vnd.openxmlformats-officedocument.wordprocessingml.header+xml`);
383
+ }
384
+ }
385
+
386
+ // Footers - only add if file actually exists in archive
387
+ // This prevents corruption when HeaderFooterManager has stale entries for removed footers
388
+ for (const entry of footers) {
389
+ const filePath = `word/${entry.filename}`;
390
+ if (filesInArchive.has(filePath)) {
391
+ generatedOverrides.add(`/word/${entry.filename}|application/vnd.openxmlformats-officedocument.wordprocessingml.footer+xml`);
392
+ }
393
+ }
394
+
395
+ // Comments
396
+ if (hasComments) {
397
+ generatedOverrides.add('/word/comments.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.comments+xml');
398
+ }
399
+
400
+ // Footnotes
401
+ const hasFootnotes = (footnoteManager && footnoteManager.getCount() > 0) || filesInArchive.has('word/footnotes.xml');
402
+ if (hasFootnotes) {
403
+ generatedOverrides.add('/word/footnotes.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.footnotes+xml');
404
+ }
405
+
406
+ // Endnotes
407
+ const hasEndnotes = (endnoteManager && endnoteManager.getCount() > 0) || filesInArchive.has('word/endnotes.xml');
408
+ if (hasEndnotes) {
409
+ generatedOverrides.add('/word/endnotes.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.endnotes+xml');
410
+ }
411
+
412
+ // People (track changes authors)
413
+ if (filesInArchive.has('word/people.xml')) {
414
+ generatedOverrides.add('/word/people.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.people+xml');
415
+ }
416
+
417
+ // Comment companion files (passthrough)
418
+ if (filesInArchive.has('word/commentsExtended.xml')) {
419
+ generatedOverrides.add('/word/commentsExtended.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.commentsExtended+xml');
420
+ }
421
+ if (filesInArchive.has('word/commentsIds.xml')) {
422
+ generatedOverrides.add('/word/commentsIds.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.commentsIds+xml');
423
+ }
424
+ if (filesInArchive.has('word/commentsExtensible.xml')) {
425
+ generatedOverrides.add('/word/commentsExtensible.xml|application/vnd.openxmlformats-officedocument.wordprocessingml.commentsExtensible+xml');
426
+ }
427
+
428
+ // Core properties (only add if file exists)
429
+ if (zipHandler.hasFile?.("docProps/core.xml") || filesInArchive.has('docProps/core.xml')) {
430
+ generatedOverrides.add('/docProps/core.xml|application/vnd.openxmlformats-package.core-properties+xml');
431
+ }
432
+
433
+ // App.xml if it exists
434
+ if (zipHandler.hasFile?.("docProps/app.xml")) {
435
+ generatedOverrides.add('/docProps/app.xml|application/vnd.openxmlformats-officedocument.extended-properties+xml');
436
+ }
437
+
438
+ // Custom properties if exists or will be created
439
+ if (zipHandler.hasFile?.("docProps/custom.xml") || hasCustomProperties) {
440
+ generatedOverrides.add('/docProps/custom.xml|application/vnd.openxmlformats-officedocument.custom-properties+xml');
441
+ }
442
+
443
+ // CustomXML entries if they exist
444
+ if (zipHandler.hasFile?.("customXML/item1.xml")) {
445
+ generatedOverrides.add('/customXML/item1.xml|application/xml');
446
+ }
447
+ if (zipHandler.hasFile?.("customXML/itemProps1.xml")) {
448
+ generatedOverrides.add('/customXML/itemProps1.xml|application/vnd.openxmlformats-officedocument.customXmlProperties+xml');
449
+ }
450
+
451
+ // Merge with original entries, but ONLY keep overrides for files that actually exist
452
+ // This prevents corruption when headers/footers are removed but their Content_Types entries
453
+ // from the original document would otherwise be preserved
454
+ const allDefaults = new Set([
455
+ ...generatedDefaults,
456
+ ...(originalContentTypes?.defaults || [])
457
+ ]);
458
+
459
+ // filesInArchive was created earlier (line 318) for header/footer validation
460
+ // Reuse it here to filter original overrides as well
461
+
462
+ // Filter original overrides to only include files that exist in the archive
463
+ const filteredOriginalOverrides: string[] = [];
464
+ for (const entry of (originalContentTypes?.overrides || [])) {
465
+ const parts = entry.split('|');
466
+ const partName = parts[0] || '';
467
+ // Convert /word/footer1.xml to word/footer1.xml for comparison
468
+ const normalizedPath = partName.startsWith('/') ? partName.slice(1) : partName;
469
+ if (filesInArchive.has(normalizedPath)) {
470
+ filteredOriginalOverrides.push(entry);
471
+ }
472
+ }
473
+
474
+ const allOverrides = new Set([
475
+ ...generatedOverrides,
476
+ ...filteredOriginalOverrides
477
+ ]);
478
+
479
+ // Build XML from merged sets
480
+ let xml = '<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n';
481
+ xml += '<Types xmlns="http://schemas.openxmlformats.org/package/2006/content-types">\n';
482
+
483
+ // Add all default entries (escape attribute values for security)
484
+ for (const entry of allDefaults) {
485
+ const parts = entry.split('|');
486
+ const ext = parts[0] || '';
487
+ const contentType = parts[1] || '';
488
+ const escapedExt = XMLBuilder.escapeXmlAttribute(ext);
489
+ const escapedContentType = XMLBuilder.escapeXmlAttribute(contentType);
490
+ xml += ` <Default Extension="${escapedExt}" ContentType="${escapedContentType}"/>\n`;
491
+ }
492
+
493
+ // Add all override entries (escape attribute values for security)
494
+ for (const entry of allOverrides) {
495
+ const parts = entry.split('|');
496
+ const partName = parts[0] || '';
497
+ const contentType = parts[1] || '';
498
+ const escapedPartName = XMLBuilder.escapeXmlAttribute(partName);
499
+ const escapedContentType = XMLBuilder.escapeXmlAttribute(contentType);
500
+ xml += ` <Override PartName="${escapedPartName}" ContentType="${escapedContentType}"/>\n`;
501
+ }
502
+
503
+ xml += '</Types>';
504
+
505
+ return xml;
506
+ }
507
+
508
+ /**
509
+ * Clears ORPHANED hyperlink relationships from the RelationshipManager
510
+ * Only removes relationships that don't have corresponding hyperlinks in the document
511
+ *
512
+ * This prevents corruption when paragraphs with hyperlinks are removed but
513
+ * their relationships remain, causing Word's "unreadable content" error.
514
+ * Preserves relationships for existing hyperlinks to maintain round-trip integrity.
515
+ */
516
+ private clearOrphanedHyperlinkRelationships(
517
+ bodyElements: BodyElement[],
518
+ headerFooterManager: HeaderFooterManager,
519
+ relationshipManager: RelationshipManager,
520
+ footnoteManager?: FootnoteManager,
521
+ endnoteManager?: EndnoteManager
522
+ ): void {
523
+ // Step 1: Collect all relationship IDs currently used by hyperlinks
524
+ const usedRelIds = new Set<string>();
525
+
526
+ // Helper to scan paragraphs for hyperlink relationship IDs
527
+ const scanParagraph = (para: Paragraph) => {
528
+ for (const item of para.getContent()) {
529
+ // Direct hyperlinks in paragraph
530
+ if (item instanceof Hyperlink && item.isExternal()) {
531
+ const relId = item.getRelationshipId();
532
+ if (relId) {
533
+ usedRelIds.add(relId);
534
+ }
535
+ }
536
+ // Hyperlinks inside Revision objects (tracked changes)
537
+ if (item instanceof Revision) {
538
+ for (const revContent of item.getContent()) {
539
+ if (isHyperlinkContent(revContent)) {
540
+ const hyperlink = revContent;
541
+ if (hyperlink.isExternal()) {
542
+ const relId = hyperlink.getRelationshipId();
543
+ if (relId) {
544
+ usedRelIds.add(relId);
545
+ }
546
+ }
547
+ }
548
+ }
549
+ }
550
+ }
551
+ };
552
+
553
+ // Helper to recursively scan any element type for hyperlinks
554
+ const scanElement = (element: BodyElement | Paragraph | Table | StructuredDocumentTag): void => {
555
+ if (element instanceof Paragraph) {
556
+ // Scan paragraph content for hyperlinks
557
+ scanParagraph(element);
558
+ }
559
+ else if (element instanceof Table) {
560
+ // Scan all cells in the table
561
+ for (let row = 0; row < element.getRowCount(); row++) {
562
+ for (let col = 0; col < element.getColumnCount(); col++) {
563
+ const cell = element.getCell(row, col);
564
+ if (cell) {
565
+ // Scan each paragraph in the cell
566
+ const paragraphs = cell.getParagraphs();
567
+ for (const para of paragraphs) {
568
+ scanParagraph(para);
569
+ }
570
+ // Scan raw nested content (nested tables, SDTs stored as raw XML)
571
+ // Extract any relationship IDs referenced in the raw XML to prevent orphan removal
572
+ for (const nested of cell.getRawNestedContent()) {
573
+ const rIdPattern = /r:id="(rId\d+)"/g;
574
+ let rIdMatch: RegExpExecArray | null;
575
+ while ((rIdMatch = rIdPattern.exec(nested.xml)) !== null) {
576
+ usedRelIds.add(rIdMatch[1]!);
577
+ }
578
+ }
579
+ }
580
+ }
581
+ }
582
+ }
583
+ else if (element instanceof StructuredDocumentTag) {
584
+ // Recursively scan SDT content (can contain Paragraphs, Tables, or nested SDTs)
585
+ const content = element.getContent();
586
+ for (const item of content) {
587
+ scanElement(item); // Recursive call handles nested structures
588
+ }
589
+ }
590
+ // TableOfContentsElement is for programmatic TOCs - real TOCs come as SDTs
591
+ };
592
+
593
+ // Scan body elements (handles all nested structures)
594
+ for (const element of bodyElements) {
595
+ scanElement(element);
596
+ }
597
+
598
+ // Scan headers (including tables and SDTs in headers)
599
+ const headers = headerFooterManager.getAllHeaders();
600
+ for (const header of headers) {
601
+ for (const element of header.header.getElements()) {
602
+ scanElement(element);
603
+ }
604
+ }
605
+
606
+ // Scan footers (including tables and SDTs in footers)
607
+ const footers = headerFooterManager.getAllFooters();
608
+ for (const footer of footers) {
609
+ for (const element of footer.footer.getElements()) {
610
+ scanElement(element);
611
+ }
612
+ }
613
+
614
+ // Scan footnotes for hyperlink relationship IDs
615
+ if (footnoteManager) {
616
+ for (const footnote of footnoteManager.getAllFootnotes()) {
617
+ for (const para of footnote.getParagraphs()) {
618
+ scanParagraph(para);
619
+ }
620
+ }
621
+ }
622
+
623
+ // Scan endnotes for hyperlink relationship IDs
624
+ if (endnoteManager) {
625
+ for (const endnote of endnoteManager.getAllEndnotes()) {
626
+ for (const para of endnote.getParagraphs()) {
627
+ scanParagraph(para);
628
+ }
629
+ }
630
+ }
631
+
632
+ // Step 2: Remove ONLY orphaned relationships (not used by any hyperlink)
633
+ const allHyperlinkRels = relationshipManager.getRelationshipsByType(
634
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/hyperlink"
635
+ );
636
+
637
+ for (const rel of allHyperlinkRels) {
638
+ if (!usedRelIds.has(rel.getId())) {
639
+ // This relationship is orphaned - remove it
640
+ relationshipManager.removeRelationship(rel.getId());
641
+ }
642
+ }
643
+ }
644
+
645
+ /**
646
+ * Processes all hyperlinks in paragraphs and registers them with RelationshipManager
647
+ * Clears orphaned hyperlink relationships to prevent corruption while preserving valid ones
648
+ *
649
+ * **IMPORTANT:** This method recursively processes ALL element types including:
650
+ * - Top-level paragraphs
651
+ * - Tables (all cells)
652
+ * - StructuredDocumentTags (SDTs / content controls)
653
+ * - Headers and footers
654
+ *
655
+ * This ensures hyperlinks with URL+anchor combinations (like theSource links)
656
+ * that have their relationshipId cleared during parsing get new relationships
657
+ * registered before XML generation.
658
+ */
659
+ processHyperlinks(
660
+ bodyElements: BodyElement[],
661
+ headerFooterManager: HeaderFooterManager,
662
+ relationshipManager: RelationshipManager,
663
+ footnoteManager?: FootnoteManager,
664
+ endnoteManager?: EndnoteManager
665
+ ): void {
666
+ const logger = getLogger();
667
+ logger.info('Processing hyperlinks');
668
+
669
+ // Clear ORPHANED hyperlink relationships to prevent corruption
670
+ // This is critical when paragraphs are removed (e.g., via clearParagraphs())
671
+ // but preserves relationships for existing hyperlinks (round-trip integrity)
672
+ this.clearOrphanedHyperlinkRelationships(
673
+ bodyElements,
674
+ headerFooterManager,
675
+ relationshipManager,
676
+ footnoteManager,
677
+ endnoteManager
678
+ );
679
+
680
+ // Helper to recursively process any element type for hyperlinks
681
+ // Mirrors the pattern in clearOrphanedHyperlinkRelationships() for consistency
682
+ const processElement = (element: BodyElement | Paragraph | Table | StructuredDocumentTag): void => {
683
+ if (element instanceof Paragraph) {
684
+ this.processHyperlinksInParagraph(element, relationshipManager);
685
+ }
686
+ else if (element instanceof Table) {
687
+ // Process all cells in the table
688
+ for (let row = 0; row < element.getRowCount(); row++) {
689
+ for (let col = 0; col < element.getColumnCount(); col++) {
690
+ const cell = element.getCell(row, col);
691
+ if (cell) {
692
+ // Scan each paragraph in the cell
693
+ const paragraphs = cell.getParagraphs();
694
+ for (const para of paragraphs) {
695
+ this.processHyperlinksInParagraph(para, relationshipManager);
696
+ }
697
+ }
698
+ }
699
+ }
700
+ }
701
+ else if (element instanceof StructuredDocumentTag) {
702
+ // Recursively process SDT content (can contain Paragraphs, Tables, or nested SDTs)
703
+ const content = element.getContent();
704
+ for (const item of content) {
705
+ processElement(item); // Recursive call handles nested structures
706
+ }
707
+ }
708
+ // TableOfContentsElement is for programmatic TOCs - real TOCs come as SDTs
709
+ };
710
+
711
+ // Process body elements (handles all nested structures)
712
+ for (const element of bodyElements) {
713
+ processElement(element);
714
+ }
715
+
716
+ // Process headers (including tables and SDTs in headers)
717
+ const headers = headerFooterManager.getAllHeaders();
718
+ for (const header of headers) {
719
+ for (const element of header.header.getElements()) {
720
+ processElement(element);
721
+ }
722
+ }
723
+
724
+ // Process footers (including tables and SDTs in footers)
725
+ const footers = headerFooterManager.getAllFooters();
726
+ for (const footer of footers) {
727
+ for (const element of footer.footer.getElements()) {
728
+ processElement(element);
729
+ }
730
+ }
731
+
732
+ // Process footnotes for hyperlinks that need relationship registration
733
+ if (footnoteManager) {
734
+ for (const footnote of footnoteManager.getAllFootnotes()) {
735
+ for (const para of footnote.getParagraphs()) {
736
+ this.processHyperlinksInParagraph(para, relationshipManager);
737
+ }
738
+ }
739
+ }
740
+
741
+ // Process endnotes for hyperlinks that need relationship registration
742
+ if (endnoteManager) {
743
+ for (const endnote of endnoteManager.getAllEndnotes()) {
744
+ for (const para of endnote.getParagraphs()) {
745
+ this.processHyperlinksInParagraph(para, relationshipManager);
746
+ }
747
+ }
748
+ }
749
+
750
+ logger.info('Hyperlinks processed');
751
+ }
752
+
753
+ /**
754
+ * Processes hyperlinks in a single paragraph
755
+ *
756
+ * **Validation:** Throws error if external hyperlink has no URL to prevent
757
+ * document corruption per ECMA-376 §17.16.22.
758
+ *
759
+ * Also processes hyperlinks inside Revision objects (tracked changes).
760
+ *
761
+ * @throws {Error} If external hyperlink has undefined/empty URL
762
+ */
763
+ private processHyperlinksInParagraph(
764
+ paragraph: Paragraph,
765
+ relationshipManager: RelationshipManager
766
+ ): void {
767
+ const content = paragraph.getContent();
768
+
769
+ for (const item of content) {
770
+ // Direct hyperlink in paragraph
771
+ if (
772
+ item instanceof Hyperlink &&
773
+ item.isExternal() &&
774
+ !item.getRelationshipId()
775
+ ) {
776
+ this.registerHyperlinkRelationship(item, relationshipManager);
777
+ }
778
+
779
+ // Hyperlinks inside Revision objects (tracked changes)
780
+ if (item instanceof Revision) {
781
+ for (const revContent of item.getContent()) {
782
+ if (isHyperlinkContent(revContent)) {
783
+ const hyperlink = revContent;
784
+ if (hyperlink.isExternal() && !hyperlink.getRelationshipId()) {
785
+ this.registerHyperlinkRelationship(hyperlink, relationshipManager);
786
+ }
787
+ }
788
+ }
789
+ }
790
+ }
791
+ }
792
+
793
+ /**
794
+ * Registers a hyperlink with the relationship manager
795
+ *
796
+ * @throws {Error} If hyperlink has no URL
797
+ */
798
+ private registerHyperlinkRelationship(
799
+ hyperlink: Hyperlink,
800
+ relationshipManager: RelationshipManager
801
+ ): void {
802
+ const url = hyperlink.getUrl();
803
+
804
+ // Validate that external hyperlink has a URL
805
+ // This prevents invalid document generation and fails early with clear error
806
+ if (!url) {
807
+ throw new Error(
808
+ `Invalid hyperlink in paragraph: External hyperlink "${hyperlink.getText()}" has no URL. ` +
809
+ `This would create a corrupted document per ECMA-376 §17.16.22. ` +
810
+ `Fix the hyperlink by providing a valid URL before saving.`
811
+ );
812
+ }
813
+
814
+ const relationship = relationshipManager.addHyperlink(url);
815
+ hyperlink.setRelationshipId(relationship.getId());
816
+ }
817
+
818
+ /**
819
+ * Generates word/fontTable.xml
820
+ * Required for DOCX compliance - defines fonts used in the document
821
+ */
822
+ generateFontTable(): string {
823
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
824
+ <w:fonts xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"
825
+ xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships">
826
+ <w:font w:name="Calibri">
827
+ <w:panose1 w:val="020F0502020204030204"/>
828
+ <w:charset w:val="00"/>
829
+ <w:family w:val="swiss"/>
830
+ <w:pitch w:val="variable"/>
831
+ <w:sig w:usb0="E10002FF" w:usb1="4000ACFF" w:usb2="00000009" w:usb3="00000000" w:csb0="0000019F" w:csb1="00000000"/>
832
+ </w:font>
833
+ <w:font w:name="Times New Roman">
834
+ <w:panose1 w:val="02020603050405020304"/>
835
+ <w:charset w:val="00"/>
836
+ <w:family w:val="roman"/>
837
+ <w:pitch w:val="variable"/>
838
+ <w:sig w:usb0="E0002AFF" w:usb1="C000785B" w:usb2="00000009" w:usb3="00000000" w:csb0="000001FF" w:csb1="00000000"/>
839
+ </w:font>
840
+ <w:font w:name="Arial">
841
+ <w:panose1 w:val="020B0604020202020204"/>
842
+ <w:charset w:val="00"/>
843
+ <w:family w:val="swiss"/>
844
+ <w:pitch w:val="variable"/>
845
+ <w:sig w:usb0="E0002AFF" w:usb1="C000247B" w:usb2="00000009" w:usb3="00000000" w:csb0="000001FF" w:csb1="00000000"/>
846
+ </w:font>
847
+ <w:font w:name="Courier New">
848
+ <w:panose1 w:val="02070309020205020404"/>
849
+ <w:charset w:val="00"/>
850
+ <w:family w:val="modern"/>
851
+ <w:pitch w:val="fixed"/>
852
+ <w:sig w:usb0="E0002AFF" w:usb1="C0007843" w:usb2="00000009" w:usb3="00000000" w:csb0="000001FF" w:csb1="00000000"/>
853
+ </w:font>
854
+ <w:font w:name="Calibri Light">
855
+ <w:panose1 w:val="020F0302020204030204"/>
856
+ <w:charset w:val="00"/>
857
+ <w:family w:val="swiss"/>
858
+ <w:pitch w:val="variable"/>
859
+ <w:sig w:usb0="E10002FF" w:usb1="4000ACFF" w:usb2="00000009" w:usb3="00000000" w:csb0="0000019F" w:csb1="00000000"/>
860
+ </w:font>
861
+ <w:font w:name="Georgia">
862
+ <w:panose1 w:val="02040502050204030303"/>
863
+ <w:charset w:val="00"/>
864
+ <w:family w:val="roman"/>
865
+ <w:pitch w:val="variable"/>
866
+ <w:sig w:usb0="E0002AFF" w:usb1="00000000" w:usb2="00000000" w:usb3="00000000" w:csb0="000001FF" w:csb1="00000000"/>
867
+ </w:font>
868
+ </w:fonts>`;
869
+ }
870
+
871
+ /**
872
+ * Generates word/webSettings.xml
873
+ * When called with no argument, produces the minimal static template.
874
+ * When called with settings, generates XML reflecting in-memory state.
875
+ */
876
+ generateWebSettings(settings?: {
877
+ optimizeForBrowser?: boolean;
878
+ allowPNG?: boolean;
879
+ relyOnVML?: boolean;
880
+ doNotRelyOnCSS?: boolean;
881
+ doNotSaveAsSingleFile?: boolean;
882
+ doNotOrganizeInFolder?: boolean;
883
+ doNotUseLongFileNames?: boolean;
884
+ pixelsPerInch?: number;
885
+ targetScreenSz?: string;
886
+ encoding?: string;
887
+ }): string {
888
+ if (!settings) {
889
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
890
+ <w:webSettings xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"
891
+ xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships">
892
+ <w:optimizeForBrowser/>
893
+ <w:allowPNG/>
894
+ </w:webSettings>`;
895
+ }
896
+
897
+ let children = '';
898
+ if (settings.optimizeForBrowser) children += '\n <w:optimizeForBrowser/>';
899
+ if (settings.allowPNG) children += '\n <w:allowPNG/>';
900
+ if (settings.relyOnVML) children += '\n <w:relyOnVML/>';
901
+ if (settings.doNotRelyOnCSS) children += '\n <w:doNotRelyOnCSS/>';
902
+ if (settings.doNotSaveAsSingleFile) children += '\n <w:doNotSaveAsSingleFile/>';
903
+ if (settings.doNotOrganizeInFolder) children += '\n <w:doNotOrganizeInFolder/>';
904
+ if (settings.doNotUseLongFileNames) children += '\n <w:doNotUseLongFileNames/>';
905
+ if (settings.pixelsPerInch !== undefined) children += `\n <w:pixelsPerInch w:val="${settings.pixelsPerInch}"/>`;
906
+ if (settings.targetScreenSz) children += `\n <w:targetScreenSz w:val="${settings.targetScreenSz}"/>`;
907
+ if (settings.encoding) children += `\n <w:encoding w:val="${settings.encoding}"/>`;
908
+
909
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
910
+ <w:webSettings xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"
911
+ xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships">${children}
912
+ </w:webSettings>`;
913
+ }
914
+
915
+ /**
916
+ * Generates word/settings.xml
917
+ * Required for DOCX compliance - defines document settings
918
+ * @param trackChangesSettings - Optional track changes settings
919
+ */
920
+ generateSettings(trackChangesSettings?: TrackChangesSettings): string {
921
+ let xml = `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
922
+ <w:settings xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"
923
+ xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships">
924
+ <w:zoom w:percent="100"/>`;
925
+
926
+ // Track changes settings — ordered per CT_Settings:
927
+ // revisionView (#30) → trackRevisions (#31) → doNotTrackFormatting (#33)
928
+ if (trackChangesSettings?.revisionView) {
929
+ const view = trackChangesSettings.revisionView;
930
+ // Only emit revisionView if it differs from defaults (all true)
931
+ if (!view.showInsertionsAndDeletions || !view.showFormatting || view.showInkAnnotations === false) {
932
+ xml += `\n <w:revisionView w:insDel="${view.showInsertionsAndDeletions ? '1' : '0'}" w:formatting="${view.showFormatting ? '1' : '0'}"`;
933
+ if (view.showInkAnnotations !== undefined) {
934
+ xml += ` w:inkAnnotations="${view.showInkAnnotations ? '1' : '0'}"`;
935
+ }
936
+ xml += '/>';
937
+ }
938
+ }
939
+
940
+ if (trackChangesSettings?.trackChangesEnabled) {
941
+ xml += '\n <w:trackRevisions/>';
942
+ }
943
+
944
+ if (trackChangesSettings?.trackFormatting !== undefined) {
945
+ if (!trackChangesSettings.trackFormatting) {
946
+ xml += '\n <w:doNotTrackFormatting/>';
947
+ }
948
+ // trackFormatting=true: emit nothing (default when w:trackRevisions is present)
949
+ }
950
+
951
+ // Document protection
952
+ if (trackChangesSettings?.documentProtection) {
953
+ const prot = trackChangesSettings.documentProtection;
954
+ xml += `\n <w:documentProtection w:edit="${prot.edit}" w:enforcement="${prot.enforcement ? '1' : '0'}"`;
955
+ if (prot.cryptProviderType) {
956
+ xml += ` w:cryptProviderType="${prot.cryptProviderType}"`;
957
+ }
958
+ if (prot.cryptAlgorithmClass) {
959
+ xml += ` w:cryptAlgorithmClass="${prot.cryptAlgorithmClass}"`;
960
+ }
961
+ if (prot.cryptAlgorithmType) {
962
+ xml += ` w:cryptAlgorithmType="${prot.cryptAlgorithmType}"`;
963
+ }
964
+ if (prot.cryptAlgorithmSid) {
965
+ xml += ` w:cryptAlgorithmSid="${prot.cryptAlgorithmSid}"`;
966
+ }
967
+ if (prot.cryptSpinCount) {
968
+ xml += ` w:cryptSpinCount="${prot.cryptSpinCount}"`;
969
+ }
970
+ if (prot.hash) {
971
+ xml += ` w:hash="${prot.hash}"`;
972
+ }
973
+ if (prot.salt) {
974
+ xml += ` w:salt="${prot.salt}"`;
975
+ }
976
+ xml += '/>';
977
+ }
978
+
979
+ xml += `
980
+ <w:defaultTabStop w:val="720"/>
981
+ <w:characterSpacingControl w:val="doNotCompress"/>
982
+ <w:updateFields w:val="true"/>`;
983
+
984
+ // RSIDs (Revision Save IDs)
985
+ if (trackChangesSettings?.rsids && trackChangesSettings.rsids.length > 0) {
986
+ xml += '\n <w:rsids>';
987
+ if (trackChangesSettings.rsidRoot) {
988
+ xml += `\n <w:rsidRoot w:val="${trackChangesSettings.rsidRoot}"/>`;
989
+ }
990
+ for (const rsid of trackChangesSettings.rsids) {
991
+ xml += `\n <w:rsid w:val="${rsid}"/>`;
992
+ }
993
+ xml += '\n </w:rsids>';
994
+ }
995
+
996
+ xml += `
997
+ <w:compat>
998
+ <w:compatSetting w:name="compatibilityMode" w:uri="http://schemas.microsoft.com/office/word" w:val="15"/>
999
+ <w:compatSetting w:name="overrideTableStyleFontSizeAndJustification" w:uri="http://schemas.microsoft.com/office/word" w:val="1"/>
1000
+ <w:compatSetting w:name="enableOpenTypeFeatures" w:uri="http://schemas.microsoft.com/office/word" w:val="1"/>
1001
+ <w:compatSetting w:name="doNotFlipMirrorIndents" w:uri="http://schemas.microsoft.com/office/word" w:val="1"/>
1002
+ <w:compatSetting w:name="differentiateMultirowTableHeaders" w:uri="http://schemas.microsoft.com/office/word" w:val="1"/>
1003
+ </w:compat>
1004
+ <w:themeFontLang w:val="en-US"/>
1005
+ <w:clrSchemeMapping w:bg1="light1" w:t1="dark1" w:bg2="light2" w:t2="dark2" w:accent1="accent1" w:accent2="accent2" w:accent3="accent3" w:accent4="accent4" w:accent5="accent5" w:accent6="accent6" w:hyperlink="hyperlink" w:followedHyperlink="followedHyperlink"/>
1006
+ </w:settings>`;
1007
+
1008
+ return xml;
1009
+ }
1010
+
1011
+ /**
1012
+ * Generates word/theme/theme1.xml
1013
+ * Required for DOCX compliance - defines color and font theme
1014
+ */
1015
+ generateTheme(): string {
1016
+ return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
1017
+ <a:theme xmlns:a="http://schemas.openxmlformats.org/drawingml/2006/main" name="Office Theme">
1018
+ <a:themeElements>
1019
+ <a:clrScheme name="Office">
1020
+ <a:dk1><a:sysClr val="windowText" lastClr="000000"/></a:dk1>
1021
+ <a:lt1><a:sysClr val="window" lastClr="FFFFFF"/></a:lt1>
1022
+ <a:dk2><a:srgbClr val="44546A"/></a:dk2>
1023
+ <a:lt2><a:srgbClr val="E7E6E6"/></a:lt2>
1024
+ <a:accent1><a:srgbClr val="5B9BD5"/></a:accent1>
1025
+ <a:accent2><a:srgbClr val="ED7D31"/></a:accent2>
1026
+ <a:accent3><a:srgbClr val="A5A5A5"/></a:accent3>
1027
+ <a:accent4><a:srgbClr val="FFC000"/></a:accent4>
1028
+ <a:accent5><a:srgbClr val="4472C4"/></a:accent5>
1029
+ <a:accent6><a:srgbClr val="70AD47"/></a:accent6>
1030
+ <a:hlink><a:srgbClr val="0563C1"/></a:hlink>
1031
+ <a:folHlink><a:srgbClr val="954F72"/></a:folHlink>
1032
+ </a:clrScheme>
1033
+ <a:fontScheme name="Office">
1034
+ <a:majorFont>
1035
+ <a:latin typeface="Calibri Light" panose="020F0302020204030204"/>
1036
+ <a:ea typeface=""/>
1037
+ <a:cs typeface=""/>
1038
+ </a:majorFont>
1039
+ <a:minorFont>
1040
+ <a:latin typeface="Calibri" panose="020F0502020204030204"/>
1041
+ <a:ea typeface=""/>
1042
+ <a:cs typeface=""/>
1043
+ </a:minorFont>
1044
+ </a:fontScheme>
1045
+ <a:fmtScheme name="Office">
1046
+ <a:fillStyleLst>
1047
+ <a:solidFill><a:schemeClr val="phClr"/></a:solidFill>
1048
+ <a:gradFill rotWithShape="1">
1049
+ <a:gsLst>
1050
+ <a:gs pos="0"><a:schemeClr val="phClr"><a:lumMod val="110000"/><a:satMod val="105000"/><a:tint val="67000"/></a:schemeClr></a:gs>
1051
+ <a:gs pos="50000"><a:schemeClr val="phClr"><a:lumMod val="105000"/><a:satMod val="103000"/><a:tint val="73000"/></a:schemeClr></a:gs>
1052
+ <a:gs pos="100000"><a:schemeClr val="phClr"><a:lumMod val="105000"/><a:satMod val="109000"/><a:tint val="81000"/></a:schemeClr></a:gs>
1053
+ </a:gsLst>
1054
+ <a:lin ang="5400000" scaled="0"/>
1055
+ </a:gradFill>
1056
+ <a:gradFill rotWithShape="1">
1057
+ <a:gsLst>
1058
+ <a:gs pos="0"><a:schemeClr val="phClr"><a:satMod val="103000"/><a:lumMod val="102000"/><a:tint val="94000"/></a:schemeClr></a:gs>
1059
+ <a:gs pos="50000"><a:schemeClr val="phClr"><a:satMod val="110000"/><a:lumMod val="100000"/><a:shade val="100000"/></a:schemeClr></a:gs>
1060
+ <a:gs pos="100000"><a:schemeClr val="phClr"><a:lumMod val="99000"/><a:satMod val="120000"/><a:shade val="78000"/></a:schemeClr></a:gs>
1061
+ </a:gsLst>
1062
+ <a:lin ang="5400000" scaled="0"/>
1063
+ </a:gradFill>
1064
+ </a:fillStyleLst>
1065
+ <a:lnStyleLst>
1066
+ <a:ln w="6350" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"/></a:solidFill><a:prstDash val="solid"/><a:miter lim="800000"/></a:ln>
1067
+ <a:ln w="12700" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"/></a:solidFill><a:prstDash val="solid"/><a:miter lim="800000"/></a:ln>
1068
+ <a:ln w="19050" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"/></a:solidFill><a:prstDash val="solid"/><a:miter lim="800000"/></a:ln>
1069
+ </a:lnStyleLst>
1070
+ <a:effectStyleLst>
1071
+ <a:effectStyle><a:effectLst/></a:effectStyle>
1072
+ <a:effectStyle><a:effectLst/></a:effectStyle>
1073
+ <a:effectStyle>
1074
+ <a:effectLst>
1075
+ <a:outerShdw blurRad="57150" dist="19050" dir="5400000" algn="ctr" rotWithShape="0">
1076
+ <a:srgbClr val="000000"><a:alpha val="63000"/></a:srgbClr>
1077
+ </a:outerShdw>
1078
+ </a:effectLst>
1079
+ </a:effectStyle>
1080
+ </a:effectStyleLst>
1081
+ <a:bgFillStyleLst>
1082
+ <a:solidFill><a:schemeClr val="phClr"/></a:solidFill>
1083
+ <a:solidFill><a:schemeClr val="phClr"><a:tint val="95000"/><a:satMod val="170000"/></a:schemeClr></a:solidFill>
1084
+ <a:gradFill rotWithShape="1">
1085
+ <a:gsLst>
1086
+ <a:gs pos="0"><a:schemeClr val="phClr"><a:tint val="93000"/><a:satMod val="150000"/><a:shade val="98000"/><a:lumMod val="102000"/></a:schemeClr></a:gs>
1087
+ <a:gs pos="50000"><a:schemeClr val="phClr"><a:tint val="98000"/><a:satMod val="130000"/><a:shade val="90000"/><a:lumMod val="103000"/></a:schemeClr></a:gs>
1088
+ <a:gs pos="100000"><a:schemeClr val="phClr"><a:shade val="63000"/><a:satMod val="120000"/></a:schemeClr></a:gs>
1089
+ </a:gsLst>
1090
+ <a:lin ang="5400000" scaled="0"/>
1091
+ </a:gradFill>
1092
+ </a:bgFillStyleLst>
1093
+ </a:fmtScheme>
1094
+ </a:themeElements>
1095
+ <a:objectDefaults/>
1096
+ <a:extraClrSchemeLst/>
1097
+ <a:extLst>
1098
+ <a:ext uri="{05A4C25C-085E-4340-85A3-A5531E510DB2}">
1099
+ <thm15:themeFamily xmlns:thm15="http://schemas.microsoft.com/office/thememl/2012/main" name="Office Theme" id="{62F939B6-93AF-4DB8-9C6B-D6C7DFDC589F}" vid="{4A3C46E8-61CC-4603-A589-7422A47A8E4A}"/>
1100
+ </a:ext>
1101
+ </a:extLst>
1102
+ </a:theme>`;
1103
+ }
1104
+ }