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.
- package/README.md +3 -2
- package/dist/constants/legacyCompatFlags.d.ts.map +1 -1
- package/dist/constants/legacyCompatFlags.js.map +1 -1
- package/dist/constants/limits.d.ts +0 -27
- package/dist/constants/limits.d.ts.map +1 -1
- package/dist/constants/limits.js +13 -13
- package/dist/constants/limits.js.map +1 -1
- package/dist/core/Document.d.ts +23 -19
- package/dist/core/Document.d.ts.map +1 -1
- package/dist/core/Document.js +197 -63
- package/dist/core/Document.js.map +1 -1
- package/dist/core/DocumentContent.d.ts.map +1 -1
- package/dist/core/DocumentContent.js.map +1 -1
- package/dist/core/DocumentGenerator.d.ts.map +1 -1
- package/dist/core/DocumentGenerator.js +59 -24
- package/dist/core/DocumentGenerator.js.map +1 -1
- package/dist/core/DocumentIdManager.d.ts.map +1 -1
- package/dist/core/DocumentIdManager.js.map +1 -1
- package/dist/core/DocumentParser.d.ts +6 -6
- package/dist/core/DocumentParser.d.ts.map +1 -1
- package/dist/core/DocumentParser.js +86 -55
- package/dist/core/DocumentParser.js.map +1 -1
- package/dist/core/DocumentValidator.d.ts.map +1 -1
- package/dist/core/DocumentValidator.js.map +1 -1
- package/dist/core/Relationship.d.ts.map +1 -1
- package/dist/core/Relationship.js +1 -1
- package/dist/core/Relationship.js.map +1 -1
- package/dist/core/RelationshipManager.js +3 -3
- package/dist/core/RelationshipManager.js.map +1 -1
- package/dist/elements/AlternateContent.js.map +1 -1
- package/dist/elements/Bookmark.d.ts.map +1 -1
- package/dist/elements/Bookmark.js.map +1 -1
- package/dist/elements/BookmarkManager.d.ts.map +1 -1
- package/dist/elements/BookmarkManager.js.map +1 -1
- package/dist/elements/Comment.js +1 -1
- package/dist/elements/Comment.js.map +1 -1
- package/dist/elements/CommentManager.d.ts.map +1 -1
- package/dist/elements/CommentManager.js +8 -2
- package/dist/elements/CommentManager.js.map +1 -1
- package/dist/elements/CommonTypes.d.ts.map +1 -1
- package/dist/elements/CommonTypes.js +1 -2
- package/dist/elements/CommonTypes.js.map +1 -1
- package/dist/elements/CustomXml.js.map +1 -1
- package/dist/elements/Endnote.d.ts.map +1 -1
- package/dist/elements/Endnote.js.map +1 -1
- package/dist/elements/EndnoteManager.d.ts.map +1 -1
- package/dist/elements/EndnoteManager.js.map +1 -1
- package/dist/elements/Field.d.ts.map +1 -1
- package/dist/elements/Field.js +31 -28
- package/dist/elements/Field.js.map +1 -1
- package/dist/elements/FieldHelpers.d.ts.map +1 -1
- package/dist/elements/FieldHelpers.js +6 -6
- package/dist/elements/FieldHelpers.js.map +1 -1
- package/dist/elements/FontManager.d.ts.map +1 -1
- package/dist/elements/FontManager.js.map +1 -1
- package/dist/elements/Footer.js.map +1 -1
- package/dist/elements/Footnote.d.ts.map +1 -1
- package/dist/elements/Footnote.js.map +1 -1
- package/dist/elements/FootnoteManager.d.ts.map +1 -1
- package/dist/elements/FootnoteManager.js.map +1 -1
- package/dist/elements/Header.js.map +1 -1
- package/dist/elements/HeaderFooterManager.js.map +1 -1
- package/dist/elements/Hyperlink.d.ts.map +1 -1
- package/dist/elements/Hyperlink.js +5 -5
- package/dist/elements/Hyperlink.js.map +1 -1
- package/dist/elements/Image.d.ts +2 -2
- package/dist/elements/Image.d.ts.map +1 -1
- package/dist/elements/Image.js +21 -5
- package/dist/elements/Image.js.map +1 -1
- package/dist/elements/ImageManager.d.ts.map +1 -1
- package/dist/elements/ImageManager.js +2 -2
- package/dist/elements/ImageManager.js.map +1 -1
- package/dist/elements/ImageRun.js.map +1 -1
- package/dist/elements/MathElement.js.map +1 -1
- package/dist/elements/Paragraph.d.ts +8 -0
- package/dist/elements/Paragraph.d.ts.map +1 -1
- package/dist/elements/Paragraph.js +153 -118
- package/dist/elements/Paragraph.js.map +1 -1
- package/dist/elements/PreservedElement.js.map +1 -1
- package/dist/elements/PropertyChangeTypes.js.map +1 -1
- package/dist/elements/RangeMarker.js.map +1 -1
- package/dist/elements/Revision.d.ts +1 -0
- package/dist/elements/Revision.d.ts.map +1 -1
- package/dist/elements/Revision.js +44 -5
- package/dist/elements/Revision.js.map +1 -1
- package/dist/elements/RevisionContent.js.map +1 -1
- package/dist/elements/RevisionManager.d.ts.map +1 -1
- package/dist/elements/RevisionManager.js.map +1 -1
- package/dist/elements/Run.d.ts.map +1 -1
- package/dist/elements/Run.js +1 -3
- package/dist/elements/Run.js.map +1 -1
- package/dist/elements/Section.d.ts.map +1 -1
- package/dist/elements/Section.js +127 -118
- package/dist/elements/Section.js.map +1 -1
- package/dist/elements/Shape.d.ts.map +1 -1
- package/dist/elements/Shape.js +21 -0
- package/dist/elements/Shape.js.map +1 -1
- package/dist/elements/StructuredDocumentTag.d.ts.map +1 -1
- package/dist/elements/StructuredDocumentTag.js +20 -8
- package/dist/elements/StructuredDocumentTag.js.map +1 -1
- package/dist/elements/Table.d.ts +2 -2
- package/dist/elements/Table.d.ts.map +1 -1
- package/dist/elements/Table.js +29 -35
- package/dist/elements/Table.js.map +1 -1
- package/dist/elements/TableCell.d.ts +2 -2
- package/dist/elements/TableCell.d.ts.map +1 -1
- package/dist/elements/TableCell.js +63 -67
- package/dist/elements/TableCell.js.map +1 -1
- package/dist/elements/TableGridChange.js.map +1 -1
- package/dist/elements/TableOfContents.d.ts +6 -6
- package/dist/elements/TableOfContents.d.ts.map +1 -1
- package/dist/elements/TableOfContents.js.map +1 -1
- package/dist/elements/TableOfContentsElement.js.map +1 -1
- package/dist/elements/TableRow.d.ts.map +1 -1
- package/dist/elements/TableRow.js +65 -47
- package/dist/elements/TableRow.js.map +1 -1
- package/dist/elements/TextBox.d.ts.map +1 -1
- package/dist/elements/TextBox.js +1 -1
- package/dist/elements/TextBox.js.map +1 -1
- package/dist/formatting/AbstractNumbering.d.ts +1 -1
- package/dist/formatting/AbstractNumbering.d.ts.map +1 -1
- package/dist/formatting/AbstractNumbering.js +11 -11
- package/dist/formatting/AbstractNumbering.js.map +1 -1
- package/dist/formatting/NumberingInstance.d.ts.map +1 -1
- package/dist/formatting/NumberingInstance.js +4 -4
- package/dist/formatting/NumberingInstance.js.map +1 -1
- package/dist/formatting/NumberingLevel.d.ts.map +1 -1
- package/dist/formatting/NumberingLevel.js +26 -26
- package/dist/formatting/NumberingLevel.js.map +1 -1
- package/dist/formatting/NumberingManager.d.ts +1 -1
- package/dist/formatting/NumberingManager.d.ts.map +1 -1
- package/dist/formatting/NumberingManager.js.map +1 -1
- package/dist/formatting/Style.d.ts.map +1 -1
- package/dist/formatting/Style.js +87 -95
- package/dist/formatting/Style.js.map +1 -1
- package/dist/formatting/StylesManager.d.ts +3 -3
- package/dist/formatting/StylesManager.d.ts.map +1 -1
- package/dist/formatting/StylesManager.js.map +1 -1
- package/dist/helpers/CleanupHelper.js.map +1 -1
- package/dist/images/ImageOptimizer.js.map +1 -1
- package/dist/index.js.map +1 -1
- package/dist/managers/DrawingManager.d.ts.map +1 -1
- package/dist/managers/DrawingManager.js.map +1 -1
- package/dist/tracking/DocumentTrackingContext.js.map +1 -1
- package/dist/tracking/TrackingContext.js.map +1 -1
- package/dist/types/compatibility-types.js.map +1 -1
- package/dist/types/formatting.js.map +1 -1
- package/dist/types/list-types.d.ts +4 -4
- package/dist/types/list-types.d.ts.map +1 -1
- package/dist/types/list-types.js.map +1 -1
- package/dist/types/settings-types.js.map +1 -1
- package/dist/types/styleConfig.js.map +1 -1
- package/dist/utils/ChangelogGenerator.d.ts.map +1 -1
- package/dist/utils/ChangelogGenerator.js.map +1 -1
- package/dist/utils/CompatibilityUpgrader.d.ts.map +1 -1
- package/dist/utils/CompatibilityUpgrader.js +7 -7
- package/dist/utils/CompatibilityUpgrader.js.map +1 -1
- package/dist/utils/InMemoryRevisionAcceptor.d.ts.map +1 -1
- package/dist/utils/InMemoryRevisionAcceptor.js +23 -2
- package/dist/utils/InMemoryRevisionAcceptor.js.map +1 -1
- package/dist/utils/MoveOperationHelper.js.map +1 -1
- package/dist/utils/RevisionAwareProcessor.js.map +1 -1
- package/dist/utils/RevisionWalker.js.map +1 -1
- package/dist/utils/SelectiveRevisionAcceptor.d.ts +1 -0
- package/dist/utils/SelectiveRevisionAcceptor.d.ts.map +1 -1
- package/dist/utils/SelectiveRevisionAcceptor.js +46 -0
- package/dist/utils/SelectiveRevisionAcceptor.js.map +1 -1
- package/dist/utils/ShadingResolver.js +1 -1
- package/dist/utils/ShadingResolver.js.map +1 -1
- package/dist/utils/acceptRevisions.d.ts +0 -28
- package/dist/utils/acceptRevisions.d.ts.map +1 -1
- package/dist/utils/acceptRevisions.js +5 -7
- package/dist/utils/acceptRevisions.js.map +1 -1
- package/dist/utils/cnfStyleDecoder.js +1 -1
- package/dist/utils/cnfStyleDecoder.js.map +1 -1
- package/dist/utils/corruptionDetection.js.map +1 -1
- package/dist/utils/dateFormatting.js.map +1 -1
- package/dist/utils/deepClone.d.ts +0 -1
- package/dist/utils/deepClone.d.ts.map +1 -1
- package/dist/utils/deepClone.js +0 -7
- package/dist/utils/deepClone.js.map +1 -1
- package/dist/utils/diagnostics.d.ts +2 -2
- package/dist/utils/diagnostics.d.ts.map +1 -1
- package/dist/utils/diagnostics.js.map +1 -1
- package/dist/utils/errorHandling.js.map +1 -1
- package/dist/utils/formatting.js.map +1 -1
- package/dist/utils/list-detection.d.ts +2 -2
- package/dist/utils/list-detection.d.ts.map +1 -1
- package/dist/utils/list-detection.js +3 -3
- package/dist/utils/list-detection.js.map +1 -1
- package/dist/utils/logger.d.ts +2 -4
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +0 -2
- package/dist/utils/logger.js.map +1 -1
- package/dist/utils/parsingHelpers.js.map +1 -1
- package/dist/utils/stripTrackedChanges.d.ts +0 -19
- package/dist/utils/stripTrackedChanges.d.ts.map +1 -1
- package/dist/utils/stripTrackedChanges.js +0 -2
- package/dist/utils/stripTrackedChanges.js.map +1 -1
- package/dist/utils/textDiff.js.map +1 -1
- package/dist/utils/units.js.map +1 -1
- package/dist/utils/validation.d.ts.map +1 -1
- package/dist/utils/validation.js.map +1 -1
- package/dist/utils/xmlSanitization.js.map +1 -1
- package/dist/validation/RevisionAutoFixer.js.map +1 -1
- package/dist/validation/RevisionValidator.js.map +1 -1
- package/dist/validation/ValidationRules.js.map +1 -1
- package/dist/validation/index.js.map +1 -1
- package/dist/xml/XMLBuilder.d.ts.map +1 -1
- package/dist/xml/XMLBuilder.js +10 -0
- package/dist/xml/XMLBuilder.js.map +1 -1
- package/dist/xml/XMLParser.d.ts.map +1 -1
- package/dist/xml/XMLParser.js +4 -5
- package/dist/xml/XMLParser.js.map +1 -1
- package/dist/zip/ZipHandler.js.map +1 -1
- package/dist/zip/ZipReader.js.map +1 -1
- package/dist/zip/ZipWriter.js.map +1 -1
- package/dist/zip/errors.js.map +1 -1
- package/dist/zip/types.js.map +1 -1
- package/package.json +34 -4
- package/src/__tests__/helper-methods.test.ts +512 -0
- package/src/constants/legacyCompatFlags.ts +138 -0
- package/src/constants/limits.ts +50 -0
- package/src/core/CLAUDE.md +109 -0
- package/src/core/Document.ts +15569 -0
- package/src/core/DocumentContent.ts +467 -0
- package/src/core/DocumentGenerator.ts +1104 -0
- package/src/core/DocumentIdManager.ts +158 -0
- package/src/core/DocumentParser.ts +10140 -0
- package/src/core/DocumentValidator.ts +372 -0
- package/src/core/Relationship.ts +367 -0
- package/src/core/RelationshipManager.ts +428 -0
- package/src/elements/AlternateContent.ts +42 -0
- package/src/elements/Bookmark.ts +210 -0
- package/src/elements/BookmarkManager.ts +250 -0
- package/src/elements/CLAUDE.md +126 -0
- package/src/elements/Comment.ts +359 -0
- package/src/elements/CommentManager.ts +502 -0
- package/src/elements/CommonTypes.ts +549 -0
- package/src/elements/CustomXml.ts +36 -0
- package/src/elements/Endnote.ts +217 -0
- package/src/elements/EndnoteManager.ts +249 -0
- package/src/elements/Field.ts +1233 -0
- package/src/elements/FieldHelpers.ts +333 -0
- package/src/elements/FontManager.ts +339 -0
- package/src/elements/Footer.ts +269 -0
- package/src/elements/Footnote.ts +217 -0
- package/src/elements/FootnoteManager.ts +249 -0
- package/src/elements/Header.ts +269 -0
- package/src/elements/HeaderFooterManager.ts +219 -0
- package/src/elements/Hyperlink.ts +1146 -0
- package/src/elements/Image.ts +1756 -0
- package/src/elements/ImageManager.ts +432 -0
- package/src/elements/ImageRun.ts +59 -0
- package/src/elements/MathElement.ts +65 -0
- package/src/elements/Paragraph.ts +4287 -0
- package/src/elements/PreservedElement.ts +53 -0
- package/src/elements/PropertyChangeTypes.ts +442 -0
- package/src/elements/RangeMarker.ts +400 -0
- package/src/elements/Revision.ts +1217 -0
- package/src/elements/RevisionContent.ts +73 -0
- package/src/elements/RevisionManager.ts +1070 -0
- package/src/elements/Run.ts +3068 -0
- package/src/elements/Section.ts +1421 -0
- package/src/elements/Shape.ts +873 -0
- package/src/elements/StructuredDocumentTag.ts +978 -0
- package/src/elements/Table.ts +2524 -0
- package/src/elements/TableCell.ts +1586 -0
- package/src/elements/TableGridChange.ts +151 -0
- package/src/elements/TableOfContents.ts +691 -0
- package/src/elements/TableOfContentsElement.ts +89 -0
- package/src/elements/TableRow.ts +906 -0
- package/src/elements/TextBox.ts +768 -0
- package/src/formatting/AbstractNumbering.ts +548 -0
- package/src/formatting/CLAUDE.md +74 -0
- package/src/formatting/NumberingInstance.ts +212 -0
- package/src/formatting/NumberingLevel.ts +1006 -0
- package/src/formatting/NumberingManager.ts +827 -0
- package/src/formatting/Style.ts +1833 -0
- package/src/formatting/StylesManager.ts +1005 -0
- package/src/helpers/CleanupHelper.ts +524 -0
- package/src/images/ImageOptimizer.ts +274 -0
- package/src/index.ts +554 -0
- package/src/managers/CLAUDE.md +47 -0
- package/src/managers/DrawingManager.ts +319 -0
- package/src/tracking/DocumentTrackingContext.ts +643 -0
- package/src/tracking/TrackingContext.ts +173 -0
- package/src/types/compatibility-types.ts +49 -0
- package/src/types/formatting.ts +210 -0
- package/src/types/list-types.ts +152 -0
- package/src/types/settings-types.ts +59 -0
- package/src/types/styleConfig.ts +189 -0
- package/src/utils/CLAUDE.md +153 -0
- package/src/utils/ChangelogGenerator.ts +1581 -0
- package/src/utils/CompatibilityUpgrader.ts +237 -0
- package/src/utils/InMemoryRevisionAcceptor.ts +696 -0
- package/src/utils/MoveOperationHelper.ts +238 -0
- package/src/utils/RevisionAwareProcessor.ts +526 -0
- package/src/utils/RevisionWalker.ts +457 -0
- package/src/utils/SelectiveRevisionAcceptor.ts +683 -0
- package/src/utils/ShadingResolver.ts +107 -0
- package/src/utils/acceptRevisions.ts +714 -0
- package/src/utils/cnfStyleDecoder.ts +217 -0
- package/src/utils/corruptionDetection.ts +345 -0
- package/src/utils/dateFormatting.ts +20 -0
- package/src/utils/deepClone.ts +78 -0
- package/src/utils/diagnostics.ts +129 -0
- package/src/utils/errorHandling.ts +80 -0
- package/src/utils/formatting.ts +213 -0
- package/src/utils/list-detection.ts +274 -0
- package/src/utils/logger.ts +404 -0
- package/src/utils/parsingHelpers.ts +190 -0
- package/src/utils/stripTrackedChanges.ts +353 -0
- package/src/utils/textDiff.ts +100 -0
- package/src/utils/units.ts +421 -0
- package/src/utils/validation.ts +542 -0
- package/src/utils/xmlSanitization.ts +182 -0
- package/src/validation/RevisionAutoFixer.ts +542 -0
- package/src/validation/RevisionValidator.ts +460 -0
- package/src/validation/ValidationRules.ts +338 -0
- package/src/validation/index.ts +30 -0
- package/src/xml/CLAUDE.md +65 -0
- package/src/xml/XMLBuilder.ts +871 -0
- package/src/xml/XMLParser.ts +919 -0
- package/src/zip/CLAUDE.md +55 -0
- package/src/zip/ZipHandler.ts +637 -0
- package/src/zip/ZipReader.ts +299 -0
- package/src/zip/ZipWriter.ts +390 -0
- package/src/zip/errors.ts +69 -0
- package/src/zip/types.ts +116 -0
- package/dist/core/ListNormalizer.d.ts +0 -23
- package/dist/core/ListNormalizer.d.ts.map +0 -1
- package/dist/core/ListNormalizer.js +0 -624
- package/dist/core/ListNormalizer.js.map +0 -1
- package/dist/images/index.d.ts +0 -2
- package/dist/images/index.d.ts.map +0 -1
- package/dist/images/index.js +0 -8
- package/dist/images/index.js.map +0 -1
- package/dist/ms-doc/cfb/CFBReader.d.ts +0 -35
- package/dist/ms-doc/cfb/CFBReader.d.ts.map +0 -1
- package/dist/ms-doc/cfb/CFBReader.js +0 -360
- package/dist/ms-doc/cfb/CFBReader.js.map +0 -1
- package/dist/ms-doc/converter/DocToDocxConverter.d.ts +0 -55
- package/dist/ms-doc/converter/DocToDocxConverter.d.ts.map +0 -1
- package/dist/ms-doc/converter/DocToDocxConverter.js +0 -324
- package/dist/ms-doc/converter/DocToDocxConverter.js.map +0 -1
- package/dist/ms-doc/fib/FIB.d.ts +0 -18
- package/dist/ms-doc/fib/FIB.d.ts.map +0 -1
- package/dist/ms-doc/fib/FIB.js +0 -342
- package/dist/ms-doc/fib/FIB.js.map +0 -1
- package/dist/ms-doc/fields/FieldParser.d.ts +0 -31
- package/dist/ms-doc/fields/FieldParser.d.ts.map +0 -1
- package/dist/ms-doc/fields/FieldParser.js +0 -266
- package/dist/ms-doc/fields/FieldParser.js.map +0 -1
- package/dist/ms-doc/images/PictureExtractor.d.ts +0 -22
- package/dist/ms-doc/images/PictureExtractor.d.ts.map +0 -1
- package/dist/ms-doc/images/PictureExtractor.js +0 -233
- package/dist/ms-doc/images/PictureExtractor.js.map +0 -1
- package/dist/ms-doc/index.d.ts +0 -20
- package/dist/ms-doc/index.d.ts.map +0 -1
- package/dist/ms-doc/index.js +0 -59
- package/dist/ms-doc/index.js.map +0 -1
- package/dist/ms-doc/properties/SPRM.d.ts +0 -210
- package/dist/ms-doc/properties/SPRM.d.ts.map +0 -1
- package/dist/ms-doc/properties/SPRM.js +0 -633
- package/dist/ms-doc/properties/SPRM.js.map +0 -1
- package/dist/ms-doc/sections/SectionParser.d.ts +0 -25
- package/dist/ms-doc/sections/SectionParser.d.ts.map +0 -1
- package/dist/ms-doc/sections/SectionParser.js +0 -214
- package/dist/ms-doc/sections/SectionParser.js.map +0 -1
- package/dist/ms-doc/styles/StyleSheet.d.ts +0 -23
- package/dist/ms-doc/styles/StyleSheet.d.ts.map +0 -1
- package/dist/ms-doc/styles/StyleSheet.js +0 -268
- package/dist/ms-doc/styles/StyleSheet.js.map +0 -1
- package/dist/ms-doc/subdocuments/SubdocumentParser.d.ts +0 -61
- package/dist/ms-doc/subdocuments/SubdocumentParser.d.ts.map +0 -1
- package/dist/ms-doc/subdocuments/SubdocumentParser.js +0 -208
- package/dist/ms-doc/subdocuments/SubdocumentParser.js.map +0 -1
- package/dist/ms-doc/tables/TableParser.d.ts +0 -29
- package/dist/ms-doc/tables/TableParser.d.ts.map +0 -1
- package/dist/ms-doc/tables/TableParser.js +0 -176
- package/dist/ms-doc/tables/TableParser.js.map +0 -1
- package/dist/ms-doc/text/PieceTable.d.ts +0 -21
- package/dist/ms-doc/text/PieceTable.d.ts.map +0 -1
- package/dist/ms-doc/text/PieceTable.js +0 -171
- package/dist/ms-doc/text/PieceTable.js.map +0 -1
- package/dist/ms-doc/types/Constants.d.ts +0 -99
- package/dist/ms-doc/types/Constants.d.ts.map +0 -1
- package/dist/ms-doc/types/Constants.js +0 -102
- package/dist/ms-doc/types/Constants.js.map +0 -1
- package/dist/ms-doc/types/DocTypes.d.ts +0 -368
- package/dist/ms-doc/types/DocTypes.d.ts.map +0 -1
- package/dist/ms-doc/types/DocTypes.js +0 -3
- package/dist/ms-doc/types/DocTypes.js.map +0 -1
- package/dist/tracking/index.d.ts +0 -3
- package/dist/tracking/index.d.ts.map +0 -1
- package/dist/tracking/index.js +0 -6
- package/dist/tracking/index.js.map +0 -1
|
@@ -0,0 +1,502 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CommentManager - Manages comments in a document
|
|
3
|
+
*
|
|
4
|
+
* Tracks all comments, assigns unique IDs, handles replies,
|
|
5
|
+
* and generates the comments.xml file.
|
|
6
|
+
*
|
|
7
|
+
* Per ECMA-376, comment IDs must be unique across ALL annotation types
|
|
8
|
+
* in a document. Use setIdProvider() to connect to a centralized ID allocator.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { Comment } from './Comment';
|
|
12
|
+
import { Run } from './Run';
|
|
13
|
+
import { XMLBuilder } from '../xml/XMLBuilder';
|
|
14
|
+
import { formatDateForXml } from '../utils/dateFormatting';
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Type for the centralized ID provider callback.
|
|
18
|
+
* Returns the next available annotation ID from a shared counter.
|
|
19
|
+
*/
|
|
20
|
+
export type IdProviderCallback = () => number;
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Type for callback to notify of existing IDs (for synchronization).
|
|
24
|
+
* Called when registering existing comments to keep the central counter in sync.
|
|
25
|
+
*/
|
|
26
|
+
export type IdExistsCallback = (existingId: number) => void;
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Comment entry stored by the manager
|
|
30
|
+
*/
|
|
31
|
+
interface CommentEntry {
|
|
32
|
+
comment: Comment;
|
|
33
|
+
/** Comments that are replies to this comment */
|
|
34
|
+
replies: Comment[];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Manages document comments
|
|
39
|
+
*/
|
|
40
|
+
export class CommentManager {
|
|
41
|
+
private comments = new Map<number, CommentEntry>();
|
|
42
|
+
private nextId = 0;
|
|
43
|
+
private idProvider: IdProviderCallback | null = null;
|
|
44
|
+
private idExistsNotifier: IdExistsCallback | null = null;
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Sets the centralized ID provider callback.
|
|
48
|
+
* When set, IDs will be allocated from the centralized DocumentIdManager
|
|
49
|
+
* instead of the local nextId counter.
|
|
50
|
+
*
|
|
51
|
+
* @param provider - Callback that returns the next available ID
|
|
52
|
+
* @param existsNotifier - Optional callback to notify when existing IDs are found
|
|
53
|
+
*/
|
|
54
|
+
setIdProvider(provider: IdProviderCallback, existsNotifier?: IdExistsCallback): void {
|
|
55
|
+
this.idProvider = provider;
|
|
56
|
+
this.idExistsNotifier = existsNotifier || null;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Registers a comment with the manager
|
|
61
|
+
* Assigns a unique ID
|
|
62
|
+
* @param comment - Comment to register
|
|
63
|
+
* @returns The registered comment (same instance)
|
|
64
|
+
*/
|
|
65
|
+
register(comment: Comment): Comment {
|
|
66
|
+
// Assign unique ID - use centralized provider if available
|
|
67
|
+
const id = this.idProvider ? this.idProvider() : this.nextId++;
|
|
68
|
+
comment.setId(id);
|
|
69
|
+
|
|
70
|
+
// Store comment
|
|
71
|
+
const entry: CommentEntry = {
|
|
72
|
+
comment,
|
|
73
|
+
replies: [],
|
|
74
|
+
};
|
|
75
|
+
this.comments.set(comment.getId(), entry);
|
|
76
|
+
|
|
77
|
+
// If this is a reply, add it to the parent's replies array
|
|
78
|
+
if (comment.isReply() && comment.getParentId() !== undefined) {
|
|
79
|
+
const parentEntry = this.comments.get(comment.getParentId()!);
|
|
80
|
+
if (parentEntry) {
|
|
81
|
+
parentEntry.replies.push(comment);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
return comment;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Registers an existing comment (from parsing) with its pre-assigned ID.
|
|
90
|
+
* Unlike register(), this does NOT assign a new ID - the comment must already have one.
|
|
91
|
+
* Used when loading comments from an existing document.
|
|
92
|
+
* @param comment - Comment with ID already set
|
|
93
|
+
*/
|
|
94
|
+
registerExisting(comment: Comment): void {
|
|
95
|
+
const id = comment.getId();
|
|
96
|
+
|
|
97
|
+
// Notify centralized ID manager if connected
|
|
98
|
+
if (this.idExistsNotifier) {
|
|
99
|
+
this.idExistsNotifier(id);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// Store comment
|
|
103
|
+
const entry: CommentEntry = {
|
|
104
|
+
comment,
|
|
105
|
+
replies: [],
|
|
106
|
+
};
|
|
107
|
+
this.comments.set(id, entry);
|
|
108
|
+
|
|
109
|
+
// Update local nextId if needed
|
|
110
|
+
if (id >= this.nextId) {
|
|
111
|
+
this.nextId = id + 1;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Links reply comments to their parents after all comments are parsed.
|
|
117
|
+
* Must be called after all comments are registered via registerExisting().
|
|
118
|
+
* This builds the reply arrays for each parent comment.
|
|
119
|
+
*/
|
|
120
|
+
linkReplies(): void {
|
|
121
|
+
// Clear existing replies first for idempotency (safe to call multiple times)
|
|
122
|
+
for (const entry of this.comments.values()) {
|
|
123
|
+
entry.replies = [];
|
|
124
|
+
}
|
|
125
|
+
for (const entry of this.comments.values()) {
|
|
126
|
+
const comment = entry.comment;
|
|
127
|
+
if (comment.isReply() && comment.getParentId() !== undefined) {
|
|
128
|
+
const parentEntry = this.comments.get(comment.getParentId()!);
|
|
129
|
+
if (parentEntry) {
|
|
130
|
+
parentEntry.replies.push(comment);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/**
|
|
137
|
+
* Gets a comment by ID
|
|
138
|
+
* @param id - Comment ID
|
|
139
|
+
* @returns The comment, or undefined if not found
|
|
140
|
+
*/
|
|
141
|
+
getComment(id: number): Comment | undefined {
|
|
142
|
+
return this.comments.get(id)?.comment;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* Gets all comments (top-level only, not replies)
|
|
147
|
+
* @returns Array of all top-level comments
|
|
148
|
+
*/
|
|
149
|
+
getAllComments(): Comment[] {
|
|
150
|
+
return Array.from(this.comments.values())
|
|
151
|
+
.filter(entry => !entry.comment.isReply())
|
|
152
|
+
.map(entry => entry.comment);
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Gets all comments including replies
|
|
157
|
+
* @returns Array of all comments
|
|
158
|
+
*/
|
|
159
|
+
getAllCommentsWithReplies(): Comment[] {
|
|
160
|
+
return Array.from(this.comments.values()).map(entry => entry.comment);
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Gets replies to a comment
|
|
165
|
+
* @param commentId - ID of the parent comment
|
|
166
|
+
* @returns Array of reply comments
|
|
167
|
+
*/
|
|
168
|
+
getReplies(commentId: number): Comment[] {
|
|
169
|
+
const entry = this.comments.get(commentId);
|
|
170
|
+
return entry ? [...entry.replies] : [];
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
* Checks if a comment has replies
|
|
175
|
+
* @param commentId - ID of the comment
|
|
176
|
+
* @returns True if the comment has replies
|
|
177
|
+
*/
|
|
178
|
+
hasReplies(commentId: number): boolean {
|
|
179
|
+
const entry = this.comments.get(commentId);
|
|
180
|
+
return entry ? entry.replies.length > 0 : false;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* Gets the number of comments (including replies)
|
|
185
|
+
* @returns Number of comments
|
|
186
|
+
*/
|
|
187
|
+
getCount(): number {
|
|
188
|
+
return this.comments.size;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/**
|
|
192
|
+
* Gets the number of top-level comments (excluding replies)
|
|
193
|
+
* @returns Number of top-level comments
|
|
194
|
+
*/
|
|
195
|
+
getTopLevelCount(): number {
|
|
196
|
+
return this.getAllComments().length;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
/**
|
|
200
|
+
* Gets all unique authors who have made comments
|
|
201
|
+
* @returns Array of unique author names
|
|
202
|
+
*/
|
|
203
|
+
getAuthors(): string[] {
|
|
204
|
+
const authorsSet = new Set<string>();
|
|
205
|
+
for (const entry of this.comments.values()) {
|
|
206
|
+
authorsSet.add(entry.comment.getAuthor());
|
|
207
|
+
}
|
|
208
|
+
return Array.from(authorsSet);
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
/**
|
|
212
|
+
* Gets comments by author
|
|
213
|
+
* @param author - Author name to filter by
|
|
214
|
+
* @returns Array of comments by the specified author
|
|
215
|
+
*/
|
|
216
|
+
getCommentsByAuthor(author: string): Comment[] {
|
|
217
|
+
return Array.from(this.comments.values())
|
|
218
|
+
.map(entry => entry.comment)
|
|
219
|
+
.filter(comment => comment.getAuthor() === author);
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
/**
|
|
223
|
+
* Gets comments within a date range
|
|
224
|
+
* @param startDate - Start of date range
|
|
225
|
+
* @param endDate - End of date range
|
|
226
|
+
* @returns Array of comments within the date range
|
|
227
|
+
*/
|
|
228
|
+
getCommentsByDateRange(startDate: Date, endDate: Date): Comment[] {
|
|
229
|
+
return Array.from(this.comments.values())
|
|
230
|
+
.map(entry => entry.comment)
|
|
231
|
+
.filter(comment => {
|
|
232
|
+
const commentDate = comment.getDate();
|
|
233
|
+
return commentDate >= startDate && commentDate <= endDate;
|
|
234
|
+
});
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
/**
|
|
238
|
+
* Removes a comment
|
|
239
|
+
* Also removes all replies to that comment
|
|
240
|
+
* @param id - Comment ID
|
|
241
|
+
* @returns True if the comment was removed
|
|
242
|
+
*/
|
|
243
|
+
removeComment(id: number): boolean {
|
|
244
|
+
const entry = this.comments.get(id);
|
|
245
|
+
if (!entry) {
|
|
246
|
+
return false;
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
// Remove all replies first
|
|
250
|
+
for (const reply of entry.replies) {
|
|
251
|
+
this.comments.delete(reply.getId());
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Remove the comment itself
|
|
255
|
+
return this.comments.delete(id);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
/**
|
|
259
|
+
* Clears all comments
|
|
260
|
+
*/
|
|
261
|
+
clear(): void {
|
|
262
|
+
this.comments.clear();
|
|
263
|
+
this.nextId = 0;
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
/**
|
|
267
|
+
* Sets the next ID to be assigned.
|
|
268
|
+
* Used when loading documents to avoid ID collisions with existing comments.
|
|
269
|
+
* @param id - The next ID value to use
|
|
270
|
+
*/
|
|
271
|
+
setNextId(id: number): void {
|
|
272
|
+
this.nextId = id;
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Creates and registers a new comment
|
|
277
|
+
* @param author - Comment author
|
|
278
|
+
* @param content - Comment content (text or runs)
|
|
279
|
+
* @param initials - Optional author initials
|
|
280
|
+
* @returns The created and registered comment
|
|
281
|
+
*/
|
|
282
|
+
createComment(
|
|
283
|
+
author: string,
|
|
284
|
+
content: string | Run | Run[],
|
|
285
|
+
initials?: string
|
|
286
|
+
): Comment {
|
|
287
|
+
const comment = Comment.create(author, content, initials);
|
|
288
|
+
return this.register(comment);
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
/**
|
|
292
|
+
* Creates and registers a reply to an existing comment
|
|
293
|
+
* @param parentCommentId - ID of the parent comment
|
|
294
|
+
* @param author - Reply author
|
|
295
|
+
* @param content - Reply content (text or runs)
|
|
296
|
+
* @param initials - Optional author initials
|
|
297
|
+
* @returns The created and registered reply
|
|
298
|
+
* @throws Error if parent comment doesn't exist
|
|
299
|
+
*/
|
|
300
|
+
createReply(
|
|
301
|
+
parentCommentId: number,
|
|
302
|
+
author: string,
|
|
303
|
+
content: string | Run | Run[],
|
|
304
|
+
initials?: string
|
|
305
|
+
): Comment {
|
|
306
|
+
// Verify parent exists
|
|
307
|
+
if (!this.comments.has(parentCommentId)) {
|
|
308
|
+
throw new Error(
|
|
309
|
+
`Cannot create reply: parent comment with ID ${parentCommentId} does not exist`
|
|
310
|
+
);
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
const reply = Comment.createReply(parentCommentId, author, content, initials);
|
|
314
|
+
return this.register(reply);
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
/**
|
|
318
|
+
* Checks if there are any comments
|
|
319
|
+
* @returns True if there are no comments
|
|
320
|
+
*/
|
|
321
|
+
isEmpty(): boolean {
|
|
322
|
+
return this.comments.size === 0;
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
/**
|
|
326
|
+
* Gets a comment thread (comment and all its replies)
|
|
327
|
+
* @param commentId - ID of the top-level comment
|
|
328
|
+
* @returns Object with the comment and its replies
|
|
329
|
+
*/
|
|
330
|
+
getCommentThread(commentId: number): { comment: Comment; replies: Comment[] } | undefined {
|
|
331
|
+
const entry = this.comments.get(commentId);
|
|
332
|
+
if (!entry || entry.comment.isReply()) {
|
|
333
|
+
return undefined;
|
|
334
|
+
}
|
|
335
|
+
return {
|
|
336
|
+
comment: entry.comment,
|
|
337
|
+
replies: [...entry.replies],
|
|
338
|
+
};
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
/**
|
|
342
|
+
* Searches comments by text content
|
|
343
|
+
* @param searchText - Text to search for (case-insensitive)
|
|
344
|
+
* @returns Array of comments containing the search text
|
|
345
|
+
*/
|
|
346
|
+
findCommentsByText(searchText: string): Comment[] {
|
|
347
|
+
const lowerSearch = searchText.toLowerCase();
|
|
348
|
+
return Array.from(this.comments.values())
|
|
349
|
+
.map(entry => entry.comment)
|
|
350
|
+
.filter(comment => comment.getText().toLowerCase().includes(lowerSearch));
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
/**
|
|
354
|
+
* Gets all resolved/done comments
|
|
355
|
+
* @returns Array of resolved comments
|
|
356
|
+
*/
|
|
357
|
+
getResolvedComments(): Comment[] {
|
|
358
|
+
return Array.from(this.comments.values())
|
|
359
|
+
.map(entry => entry.comment)
|
|
360
|
+
.filter(comment => comment.isResolved());
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
/**
|
|
364
|
+
* Gets all unresolved comments
|
|
365
|
+
* @returns Array of unresolved comments
|
|
366
|
+
*/
|
|
367
|
+
getUnresolvedComments(): Comment[] {
|
|
368
|
+
return Array.from(this.comments.values())
|
|
369
|
+
.map(entry => entry.comment)
|
|
370
|
+
.filter(comment => !comment.isResolved());
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
/**
|
|
374
|
+
* Gets the most recent comments
|
|
375
|
+
* @param count - Number of recent comments to return
|
|
376
|
+
* @returns Array of most recent comments
|
|
377
|
+
*/
|
|
378
|
+
getRecentComments(count: number): Comment[] {
|
|
379
|
+
const allComments = this.getAllCommentsWithReplies();
|
|
380
|
+
return allComments
|
|
381
|
+
.sort((a, b) => b.getDate().getTime() - a.getDate().getTime())
|
|
382
|
+
.slice(0, count);
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
/**
|
|
386
|
+
* Gets statistics about comments
|
|
387
|
+
* @returns Object with comment statistics
|
|
388
|
+
*/
|
|
389
|
+
getStats(): {
|
|
390
|
+
total: number;
|
|
391
|
+
topLevel: number;
|
|
392
|
+
replies: number;
|
|
393
|
+
resolved: number;
|
|
394
|
+
unresolved: number;
|
|
395
|
+
authors: string[];
|
|
396
|
+
nextId: number;
|
|
397
|
+
} {
|
|
398
|
+
const topLevel = this.getTopLevelCount();
|
|
399
|
+
const resolved = this.getResolvedComments().length;
|
|
400
|
+
return {
|
|
401
|
+
total: this.comments.size,
|
|
402
|
+
topLevel,
|
|
403
|
+
replies: this.comments.size - topLevel,
|
|
404
|
+
resolved,
|
|
405
|
+
unresolved: this.comments.size - resolved,
|
|
406
|
+
authors: this.getAuthors(),
|
|
407
|
+
nextId: this.nextId,
|
|
408
|
+
};
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
/**
|
|
412
|
+
* Generates the word/comments.xml file content
|
|
413
|
+
* @returns XML string for comments.xml
|
|
414
|
+
*/
|
|
415
|
+
generateCommentsXml(): string {
|
|
416
|
+
const comments = this.getAllCommentsWithReplies();
|
|
417
|
+
|
|
418
|
+
if (comments.length === 0) {
|
|
419
|
+
// Return minimal comments.xml
|
|
420
|
+
return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
|
|
421
|
+
<w:comments xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main">
|
|
422
|
+
</w:comments>`;
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
// Build XML manually for comments
|
|
426
|
+
const hasReplies = comments.some(c => c.isReply());
|
|
427
|
+
let xml = '<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n';
|
|
428
|
+
xml += '<w:comments xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"';
|
|
429
|
+
xml += ' xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships"';
|
|
430
|
+
if (hasReplies) {
|
|
431
|
+
xml += ' xmlns:w15="http://schemas.microsoft.com/office/word/2012/wordml"';
|
|
432
|
+
xml += ' xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="w15"';
|
|
433
|
+
}
|
|
434
|
+
xml += '>\n';
|
|
435
|
+
|
|
436
|
+
// Add each comment
|
|
437
|
+
for (const comment of comments) {
|
|
438
|
+
xml += this.commentToXmlString(comment);
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
xml += '</w:comments>';
|
|
442
|
+
return xml;
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
/**
|
|
446
|
+
* Converts a comment to XML string
|
|
447
|
+
* @param comment - Comment to convert
|
|
448
|
+
* @returns XML string for the comment
|
|
449
|
+
*/
|
|
450
|
+
private commentToXmlString(comment: Comment): string {
|
|
451
|
+
let xml = ` <w:comment w:id="${comment.getId()}"`;
|
|
452
|
+
xml += ` w:author="${XMLBuilder.escapeXmlAttribute(comment.getAuthor())}"`;
|
|
453
|
+
xml += ` w:date="${formatDateForXml(comment.getDate())}"`;
|
|
454
|
+
xml += ` w:initials="${XMLBuilder.escapeXmlAttribute(comment.getInitials())}"`;
|
|
455
|
+
|
|
456
|
+
if (comment.isReply() && comment.getParentId() !== undefined) {
|
|
457
|
+
xml += ` w15:parentId="${comment.getParentId()}"`;
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// Add done attribute for resolved comments (per ECMA-376)
|
|
461
|
+
if (comment.isResolved()) {
|
|
462
|
+
xml += ` w:done="1"`;
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
xml += '>\n';
|
|
466
|
+
|
|
467
|
+
// Add paragraph with runs
|
|
468
|
+
xml += ' <w:p>\n';
|
|
469
|
+
for (const run of comment.getRuns()) {
|
|
470
|
+
xml += this.runToXmlString(run, 6);
|
|
471
|
+
}
|
|
472
|
+
xml += ' </w:p>\n';
|
|
473
|
+
|
|
474
|
+
xml += ' </w:comment>\n';
|
|
475
|
+
return xml;
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
/**
|
|
479
|
+
* Converts a run to XML string
|
|
480
|
+
* @param run - Run to convert
|
|
481
|
+
* @param indent - Number of spaces for indentation
|
|
482
|
+
* @returns XML string for the run
|
|
483
|
+
*/
|
|
484
|
+
private runToXmlString(run: Run, indent: number): string {
|
|
485
|
+
const spaces = ' '.repeat(indent);
|
|
486
|
+
const text = XMLBuilder.escapeXmlText(run.getText());
|
|
487
|
+
|
|
488
|
+
let xml = `${spaces}<w:r>\n`;
|
|
489
|
+
xml += `${spaces} <w:t xml:space="preserve">${text}</w:t>\n`;
|
|
490
|
+
xml += `${spaces}</w:r>\n`;
|
|
491
|
+
|
|
492
|
+
return xml;
|
|
493
|
+
}
|
|
494
|
+
|
|
495
|
+
/**
|
|
496
|
+
* Creates a new CommentManager
|
|
497
|
+
* @returns New CommentManager instance
|
|
498
|
+
*/
|
|
499
|
+
static create(): CommentManager {
|
|
500
|
+
return new CommentManager();
|
|
501
|
+
}
|
|
502
|
+
}
|