docrev 0.9.18 → 0.10.1

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 (134) hide show
  1. package/.gitattributes +1 -1
  2. package/CHANGELOG.md +173 -149
  3. package/PLAN-tables-and-postprocess.md +850 -850
  4. package/README.md +431 -406
  5. package/bin/rev.js +11 -11
  6. package/bin/rev.ts +145 -145
  7. package/completions/rev.bash +127 -127
  8. package/completions/rev.ps1 +210 -210
  9. package/completions/rev.zsh +207 -207
  10. package/dist/lib/build.d.ts +8 -0
  11. package/dist/lib/build.d.ts.map +1 -1
  12. package/dist/lib/build.js +62 -6
  13. package/dist/lib/build.js.map +1 -1
  14. package/dist/lib/commands/context.d.ts +1 -1
  15. package/dist/lib/commands/context.d.ts.map +1 -1
  16. package/dist/lib/commands/context.js +1 -1
  17. package/dist/lib/commands/context.js.map +1 -1
  18. package/dist/lib/commands/sections.js +7 -7
  19. package/dist/lib/commands/sections.js.map +1 -1
  20. package/dist/lib/commands/sync.d.ts.map +1 -1
  21. package/dist/lib/commands/sync.js +15 -14
  22. package/dist/lib/commands/sync.js.map +1 -1
  23. package/dist/lib/commands/utilities.js +164 -164
  24. package/dist/lib/commands/verify-anchors.js +6 -6
  25. package/dist/lib/commands/verify-anchors.js.map +1 -1
  26. package/dist/lib/commands/word-tools.js +8 -8
  27. package/dist/lib/grammar.js +3 -3
  28. package/dist/lib/macro-filter.lua +201 -0
  29. package/dist/lib/macros.d.ts +102 -0
  30. package/dist/lib/macros.d.ts.map +1 -0
  31. package/dist/lib/macros.js +218 -0
  32. package/dist/lib/macros.js.map +1 -0
  33. package/dist/lib/pdf-comments.js +44 -44
  34. package/dist/lib/plugins.js +57 -57
  35. package/dist/lib/pptx-color-filter.lua +37 -0
  36. package/dist/lib/pptx-themes.js +115 -115
  37. package/dist/lib/schema.d.ts.map +1 -1
  38. package/dist/lib/schema.js +34 -0
  39. package/dist/lib/schema.js.map +1 -1
  40. package/dist/lib/sections.d.ts +35 -0
  41. package/dist/lib/sections.d.ts.map +1 -1
  42. package/dist/lib/sections.js +81 -0
  43. package/dist/lib/sections.js.map +1 -1
  44. package/dist/lib/spelling.js +2 -2
  45. package/dist/lib/templates.js +387 -387
  46. package/dist/lib/themes.js +51 -51
  47. package/eslint.config.js +27 -27
  48. package/lib/anchor-match.ts +276 -276
  49. package/lib/annotations.ts +644 -644
  50. package/lib/build.ts +1766 -1694
  51. package/lib/citations.ts +160 -160
  52. package/lib/commands/build.ts +855 -855
  53. package/lib/commands/citations.ts +515 -515
  54. package/lib/commands/comments.ts +1050 -1050
  55. package/lib/commands/context.ts +176 -174
  56. package/lib/commands/core.ts +309 -309
  57. package/lib/commands/doi.ts +435 -435
  58. package/lib/commands/file-ops.ts +372 -372
  59. package/lib/commands/history.ts +320 -320
  60. package/lib/commands/index.ts +87 -87
  61. package/lib/commands/init.ts +259 -259
  62. package/lib/commands/merge-resolve.ts +378 -378
  63. package/lib/commands/preview.ts +178 -178
  64. package/lib/commands/project-info.ts +244 -244
  65. package/lib/commands/quality.ts +517 -517
  66. package/lib/commands/response.ts +454 -454
  67. package/lib/commands/section-boundaries.ts +82 -82
  68. package/lib/commands/sections.ts +451 -451
  69. package/lib/commands/sync.ts +709 -706
  70. package/lib/commands/text-ops.ts +449 -449
  71. package/lib/commands/utilities.ts +448 -448
  72. package/lib/commands/verify-anchors.ts +272 -272
  73. package/lib/commands/word-tools.ts +340 -340
  74. package/lib/comment-realign.ts +517 -517
  75. package/lib/config.ts +84 -84
  76. package/lib/crossref.ts +781 -781
  77. package/lib/csl.ts +191 -191
  78. package/lib/dependencies.ts +98 -98
  79. package/lib/diff-engine.ts +465 -465
  80. package/lib/doi-cache.ts +115 -115
  81. package/lib/doi.ts +897 -897
  82. package/lib/equations.ts +506 -506
  83. package/lib/errors.ts +346 -346
  84. package/lib/format.ts +541 -541
  85. package/lib/git.ts +326 -326
  86. package/lib/grammar.ts +303 -303
  87. package/lib/image-registry.ts +180 -180
  88. package/lib/import.ts +911 -911
  89. package/lib/journals.ts +543 -543
  90. package/lib/macro-filter.lua +201 -0
  91. package/lib/macros.ts +273 -0
  92. package/lib/merge.ts +633 -633
  93. package/lib/orcid.ts +144 -144
  94. package/lib/pdf-comments.ts +263 -263
  95. package/lib/pdf-import.ts +524 -524
  96. package/lib/plugins.ts +362 -362
  97. package/lib/postprocess.ts +188 -188
  98. package/lib/pptx-color-filter.lua +37 -37
  99. package/lib/pptx-template.ts +469 -469
  100. package/lib/pptx-themes.ts +483 -483
  101. package/lib/protect-restore.ts +520 -520
  102. package/lib/rate-limiter.ts +94 -94
  103. package/lib/response.ts +197 -197
  104. package/lib/restore-references.ts +240 -240
  105. package/lib/review.ts +327 -327
  106. package/lib/schema.ts +488 -454
  107. package/lib/scientific-words.ts +73 -73
  108. package/lib/sections.ts +425 -335
  109. package/lib/slides.ts +756 -756
  110. package/lib/spelling.ts +334 -334
  111. package/lib/templates.ts +526 -526
  112. package/lib/themes.ts +742 -742
  113. package/lib/trackchanges.ts +247 -247
  114. package/lib/tui.ts +450 -450
  115. package/lib/types.ts +550 -550
  116. package/lib/undo.ts +250 -250
  117. package/lib/utils.ts +69 -69
  118. package/lib/variables.ts +179 -179
  119. package/lib/word-extraction.ts +806 -806
  120. package/lib/word.ts +643 -643
  121. package/lib/wordcomments.ts +840 -840
  122. package/package.json +137 -137
  123. package/scripts/postbuild.js +47 -28
  124. package/skill/REFERENCE.md +539 -539
  125. package/skill/SKILL.md +295 -295
  126. package/tsconfig.json +26 -26
  127. package/types/index.d.ts +525 -525
  128. package/issues.md +0 -180
  129. package/site/assets/extra.css +0 -208
  130. package/site/commands.html +0 -926
  131. package/site/configuration.html +0 -469
  132. package/site/index.html +0 -288
  133. package/site/troubleshooting.html +0 -461
  134. package/site/workflow.html +0 -518
package/.gitattributes CHANGED
@@ -1 +1 @@
1
- dist/** linguist-generated
1
+ dist/** linguist-generated
package/CHANGELOG.md CHANGED
@@ -1,149 +1,173 @@
1
- # Changelog
2
-
3
- All notable changes to docrev will be documented in this file.
4
-
5
- The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
6
- and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
-
8
- ## [0.9.11] - 2026-04-30
9
-
10
- ### Fixed
11
- - **Single-section comment placement.** `computeSectionBoundaries` left the last section's `end` at `Number.MAX_SAFE_INTEGER`, which collapsed the proportional-position math in `insertCommentsIntoMarkdown` to ~0. Every comment whose anchor wasn't in the first 200 chars of the markdown stacked at position 0. Now caps the last boundary's `end` at `fullDocText.length`, passed in from sync and verify-anchors.
12
- - **Re-sync duplicated comments.** `sync --comments-only` re-inserted every comment on each invocation, producing `{>>R1<<}{>>R1<<}{>>R1<<}…` over time. `insertCommentsIntoMarkdown` now scans ±200 chars around the target for an identical `{>>author: text<<}` block and skips insertion when found.
13
- - **Threading content destruction.** `prepareMarkdownWithMarkers`'s whitespace-consumption loops captured `charBefore` once outside the loop, so a single leading space caused `removeStart` to walk to position 0 and `slice()` to delete every preceding paragraph. Replaced with a one-char check.
14
- - **Multi-run anchor injection.** Pandoc splits a single anchor across multiple `<w:r>` blocks whenever it applies styling mid-anchor — smart-quote substitution, `*italic*`, `` `code` ``, `**bold**` all trigger this. The single-run scan in `injectCommentsAtMarkers` grabbed the start marker's `<w:t>`, looked for the end marker inside it, found nothing, and silently skipped the comment. New multi-run path splits the start run at the start marker, keeps middle runs verbatim, splits the end run at the end marker, and rebuilds with `commentRangeStart`/`commentRangeEnd` around the styled anchor portions.
15
- - **Nested-bracket anchors.** `prepareMarkdownWithMarkers` used `\[([^\]]+)\]\{\.mark\}` for the trailing anchor group, so any inner `]` (e.g. `[[0..9]]{.mark}`, `[*italic*]{.mark}`) terminated the match prematurely. Replaced with a manual balanced-bracket walker that handles arbitrary nesting depth and verifies a `{.mark}` suffix.
16
- - **Orphan-`[` over-stripping.** `stripAnnotations`'s orphan cleanup used `\[(?![^\[\]]*\])`, treating any inner `[` as a barrier and stripping the outer `[` of nested forms. Loosened to `\[(?![^\]\n]*\])`: an `[` is orphan only when no `]` follows before end of line.
17
-
18
- ### Changed
19
- - `sync --comments-only` summary distinguishes `placed` / `already present` / `unmatched` instead of subtracting before/after counts. Re-syncs now report "6 already present (skipped to avoid duplication)" instead of misreporting them as fully placed or fully unmatched. New `outStats` channel from `insertCommentsIntoMarkdown`.
20
-
21
- ## [0.9.10] - 2026-04-30
22
-
23
- ### Fixed
24
- - `stripAnnotations` stripped `[anchor]{.mark}` spans even when `keepComments=true`, leaving the dual-build marker generator with no anchor text and collapsing every multi-word anchor to a single fallback word in the rebuilt docx. Now preserves anchor spans that belong to retained `{>>...<<}` comments.
25
- - Comments authored at the very start of a Word section landed before the markdown file's `# Heading` line and rendered in the previous section. Added `pushPastSectionHeading` so position-0 comments advance to the first body paragraph of the section they were authored in.
26
- - Empty-anchor comments fell through to proportional placement even when before/after context uniquely identified the position, landing mid-word or splitting unrelated phrases. Context match now runs first; proportional placement is the fallback.
27
- - When an anchor appeared multiple times in the search window (repeated phrasing, formulaic prose), `insertCommentsIntoMarkdown` always picked the first occurrence. Now picks the occurrence closest to the docx-derived insert position.
28
-
29
- ## [0.9.7] - 2026-04-29
30
-
31
- ### Added
32
- - `rev sync --comments-only` — import only Word comments at fuzzy-matched anchors, leaving prose byte-identical. Use when the markdown was revised between sending the docx out for review and receiving it back; applying track changes from a stale draft would clobber newer edits.
33
- - `rev verify-anchors <docx>` — drift report classifying every comment as `clean` / `drift` / `context-only` / `ambiguous` / `unmatched` against the current section markdown. Pair with `--comments-only` to plan placement before sync. Supports `--json` for scripting.
34
- - `extractHeadings()` in `word-extraction.ts` — read heading paragraphs directly from `<w:pStyle>` styles, returning text + level + position in the same coordinate system as comment anchors.
35
- - Shared `lib/commands/section-boundaries.ts` single source of truth that maps `sections.yaml` to docx text positions, used by both sync and verify-anchors.
36
- - Shared `lib/anchor-match.ts` pure anchor-matching primitives (`findAnchorInText`, `stripCriticMarkup`, `classifyStrategy`) so sync (insertion) and verify-anchors (drift reporting) use the same fallback strategies.
37
- - New tests: `test/anchor-match.test.js` (11 cases covering each fallback strategy and the quality classifier).
38
-
39
- ### Fixed
40
- - **Section detection mistook prose for headings.** The old keyword finder scanned the concatenated body text and would match "results across countries" as the Results heading or skip the real Methods heading because the structured-abstract label `Methods:` lost its colon during text-run concatenation. Replaced with paragraph-style-based heading extraction, so boundaries now reflect actual heading paragraphs. Affects the new commands; the existing sync flow already used pandoc-derived headings and was unaffected.
41
- - `stripCriticMarkup` regex used `[^<]*` and silently failed on comments containing `<` characters (e.g. `pre-industrial trade (<1825)`). Switched to non-greedy `[\s\S]*?`.
42
- - `insertCommentsIntoMarkdown` always prepended a leading space when there was no anchor, accumulating multiple spaces when several comments shared a position 0 anchor. Removed the heuristic; comments insert at exact position so prose stays byte-identical except for the inserted blocks.
43
- - `verify-anchors` crashed with a stack trace when given a non-docx file (e.g. an `.md` path). Now reports a friendly error.
44
-
45
- ### Changed
46
- - New flag is `--comments-only` (positive form). The originally proposed `--no-overwrite` was dropped because Commander assigns `--no-X` to `options.x === false` rather than `options.noX === true`, which made the flag silently ignored.
47
- - `insertCommentsIntoMarkdown` now accepts `wrapAnchor?: boolean` (default `true`). When `false`, comment blocks are inserted next to the anchor without `[anchor]{.mark}` wrapping. `--comments-only` uses this so multiple comments sharing an anchor (e.g. 6 reviewers commenting on the same word) no longer produce nested broken CriticMarkup.
48
-
49
- ## [0.7.1] - 2025-01-02
50
-
51
- ### Added
52
- - Writing Markdown guide in docs (tables, equations, citations, cross-refs)
53
- - Grid table syntax documentation for merged cells
54
-
55
- ### Changed
56
- - README restructured for better scannability (490 290 lines)
57
- - Install section moved up for faster onboarding
58
- - Added Highlights section with quick feature overview
59
- - Condensed overlapping sections
60
-
61
- ## [0.7.0] - 2025-01-02
62
-
63
- ### Added
64
- - API rate limiting with exponential backoff for Crossref/DataCite/doi.org APIs
65
- - Windows support in CI matrix
66
- - Architecture documentation for contributors (`ARCHITECTURE.md`)
67
- - Exclusion patterns for cross-reference false positives (e.g., "Table of Contents")
68
- - Timeout support for PDF extraction (30s default)
69
-
70
- ### Changed
71
- - Consolidated YAML dependencies (removed `js-yaml`, using `yaml` package only)
72
- - Improved annotation false positive detection (code blocks, URLs, LaTeX patterns)
73
- - Enhanced error messages for Word import and PDF extraction
74
- - Updated User-Agent strings for API requests
75
- - Improved README with problem statement and quick example
76
-
77
- ### Fixed
78
- - CI lint step now checks all command files separately
79
- - Windows temp directory paths in tests
80
-
81
- ## [0.3.2] - 2024-12-29
82
-
83
- ### Added
84
- - Full TypeScript type definitions (`types/index.d.ts`)
85
- - GitHub Actions CI workflow (Node 18/20/22)
86
- - ESM subpath exports for all library modules
87
- - CLI integration tests (26 tests)
88
- - Comprehensive test coverage: 419 tests across 18 modules
89
-
90
- ### Fixed
91
- - DOI skip detection: `% no-doi` comments now correctly apply only to the next entry
92
-
93
- ### Changed
94
- - Added `engines` field requiring Node.js >=18.0.0
95
- - Updated README with badges (npm, CI, Node.js, License)
96
-
97
- ## [0.3.1] - 2024-12-28
98
-
99
- ### Fixed
100
- - Equation extraction test assertions
101
- - Minor bug fixes
102
-
103
- ## [0.3.0] - 2024-12-27
104
-
105
- ### Added
106
- - DOI validation via Crossref and DataCite APIs (`rev doi check`)
107
- - DOI lookup for missing entries (`rev doi lookup`)
108
- - DOI fetch and add commands (`rev doi fetch`, `rev doi add`)
109
- - Citation validation against bibliography (`rev citations`)
110
- - LaTeX equation extraction (`rev equations list`)
111
- - Word equation import OMML → LaTeX (`rev equations from-word`)
112
- - Response letter generation (`rev response`)
113
- - Journal validation profiles (`rev validate --journal`)
114
- - Advanced figure/table reference patterns (Figs. 1-3, Fig. 1a-c)
115
-
116
- ### Changed
117
- - Improved cross-reference pattern detection
118
- - Enhanced Word import with better section splitting
119
-
120
- ## [0.2.1] - 2024-12-26
121
-
122
- ### Added
123
- - Table of contents option (`rev build --toc`)
124
- - CSV export for comments (`rev comments --export`)
125
- - Anonymize command for blind review (`rev anonymize`)
126
- - Formatting utilities (tables, boxes, spinners)
127
-
128
- ## [0.2.0] - 2024-12-25
129
-
130
- ### Added
131
- - Integrated build system (`rev build pdf/docx/tex`)
132
- - Comment reply functionality (`rev reply`)
133
- - Word document bootstrap (`rev import` creates project from .docx)
134
- - Section-aware import (`rev sections`)
135
- - Cross-reference migration (`rev migrate`)
136
-
137
- ### Changed
138
- - Renamed project to docrev
139
- - Published to npm
140
-
141
- ## [0.1.0] - 2024-12-24
142
-
143
- ### Added
144
- - Initial release
145
- - CriticMarkup annotation parsing
146
- - Word ↔ Markdown round-trips
147
- - Interactive review TUI (`rev review`)
148
- - Comment management (`rev comments`, `rev resolve`)
149
- - Project templates (`rev new`)
1
+ # Changelog
2
+
3
+ All notable changes to docrev will be documented in this file.
4
+
5
+ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
6
+ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
+
8
+ ## [0.10.1] - 2026-06-05
9
+
10
+ ### Fixed
11
+ - **`sync`/`verify-anchors`/`split` no longer require a separate `sections.yaml`.** `build` and `status` read the `sections:` list from `rev.yaml`, but the sync-side commands demanded a `sections.yaml` and aborted with `Config not found` on a perfectly valid `rev.yaml`-only project. They now resolve the section config from a single source of truth: an explicit `sections.yaml` is used when present (to override headers/aliases/order), otherwise the config is derived from `rev.yaml`'s `sections:` list. Running `rev init` first is no longer necessary.
12
+ - **Derived section headers honor the file's first heading at any level.** A section file that leads with a subsection (e.g. `02_objectives.md` starting with `## 1.2 Objectives`) now derives the header `1.2 Objectives` (via the new `extractFirstHeading`), so it matches the corresponding docx heading and routes to its own file instead of being folded into the preceding section. `extractHeader` keeps its H1-only contract for existing callers.
13
+
14
+ ### Added
15
+ - `resolveSectionsConfig(dir, configFile?)` and `deriveSectionsFromRev(dir)` in `lib/sections.ts`, with tests.
16
+
17
+ ## [0.10.0] - 2026-05-20
18
+
19
+ ### Added
20
+ - **Placeholder macros.** Built-in `\tofill{X}` renders as bold orange `[X]` across DOCX, PDF/TeX/Beamer, and HTML — drop the LaTeX-style command anywhere in a section file and the build expands it per format. No more copying a `tofill_filter.lua` + `\providecommand` boilerplate into each project. Users can add their own one-argument macros under `macros:` in `rev.yaml` (color, bold/italic, bracket wrap, prefix/suffix, per-format overrides). See `docs/configuration.md` for the schema.
21
+ - **`lib/macros.ts` + `lib/macro-filter.lua`** — registry-style implementation. Built-ins and user macros merge by name; user entries override built-ins. The lua filter loads its macro list from a JSON sidecar via the `DOCREV_MACROS_FILE` env var.
22
+
23
+ ### Fixed
24
+ - **Silent DOCX color drop.** Pandoc 3.x's docx writer does NOT honor `Span` with `style="color: #..."`, so a lua filter returning `pandoc.Span(pandoc.Strong(...), pandoc.Attr("", {}, {style="color: #C2410C"}))` produced zero `<w:color>` runs and the highlight vanished. The new filter emits raw OpenXML `<w:r>` nodes directly (the same pattern as the existing `pptx-color-filter.lua`), which produces real colored runs in Word.
25
+ - **Silent DOCX paragraph drop.** Wrapping a `\tofill{X}` line in raw LaTeX like `\noindent \textit{\small ...}` made pandoc drop the entire paragraph from DOCX output, including the `\tofill` inside. The built-in `\providecommand` + lua-filter mechanism removes the need for any wrapper at all placeholders survive every format.
26
+ - **Filter path resolution on Windows paths with spaces.** `lib/build.ts` resolved `pptx-color-filter.lua` via `new URL(import.meta.url).pathname`, which returns URL-encoded `%20` for spaces, so `fs.existsSync` silently returned false on paths like `C:\Users\Gilles Colling\...`. Switched both pptx and macro filter resolution to `fileURLToPath`. PPTX color highlighting now actually works under default installs on user-named directories.
27
+ - **Lua filter not shipped to runtime.** `scripts/postbuild.js` left `.lua` files in `lib/` only; the compiled `dist/lib/build.js` looked for them next to itself. PPTX colour filter (and now the macro filter) are now copied to `dist/lib/` during build.
28
+
29
+ ### Notes for `\tofill` migration
30
+ Projects that ship their own `tofill_filter.lua` and `\providecommand{\tofill}{...}` in a custom header keep working — docrev's preamble uses `\providecommand`, so user `\renewcommand` (or pre-existing `\providecommand`) takes priority. To migrate, delete the local lua filter and any markdown wrappers like `\noindent \textit{\small ...}`; docrev's built-in covers all three formats. Override the color or style by adding the macro under `macros:` in `rev.yaml`.
31
+
32
+ ## [0.9.11] - 2026-04-30
33
+
34
+ ### Fixed
35
+ - **Single-section comment placement.** `computeSectionBoundaries` left the last section's `end` at `Number.MAX_SAFE_INTEGER`, which collapsed the proportional-position math in `insertCommentsIntoMarkdown` to ~0. Every comment whose anchor wasn't in the first 200 chars of the markdown stacked at position 0. Now caps the last boundary's `end` at `fullDocText.length`, passed in from sync and verify-anchors.
36
+ - **Re-sync duplicated comments.** `sync --comments-only` re-inserted every comment on each invocation, producing `{>>R1<<}{>>R1<<}{>>R1<<}…` over time. `insertCommentsIntoMarkdown` now scans ±200 chars around the target for an identical `{>>author: text<<}` block and skips insertion when found.
37
+ - **Threading content destruction.** `prepareMarkdownWithMarkers`'s whitespace-consumption loops captured `charBefore` once outside the loop, so a single leading space caused `removeStart` to walk to position 0 and `slice()` to delete every preceding paragraph. Replaced with a one-char check.
38
+ - **Multi-run anchor injection.** Pandoc splits a single anchor across multiple `<w:r>` blocks whenever it applies styling mid-anchor — smart-quote substitution, `*italic*`, `` `code` ``, `**bold**` all trigger this. The single-run scan in `injectCommentsAtMarkers` grabbed the start marker's `<w:t>`, looked for the end marker inside it, found nothing, and silently skipped the comment. New multi-run path splits the start run at the start marker, keeps middle runs verbatim, splits the end run at the end marker, and rebuilds with `commentRangeStart`/`commentRangeEnd` around the styled anchor portions.
39
+ - **Nested-bracket anchors.** `prepareMarkdownWithMarkers` used `\[([^\]]+)\]\{\.mark\}` for the trailing anchor group, so any inner `]` (e.g. `[[0..9]]{.mark}`, `[*italic*]{.mark}`) terminated the match prematurely. Replaced with a manual balanced-bracket walker that handles arbitrary nesting depth and verifies a `{.mark}` suffix.
40
+ - **Orphan-`[` over-stripping.** `stripAnnotations`'s orphan cleanup used `\[(?![^\[\]]*\])`, treating any inner `[` as a barrier and stripping the outer `[` of nested forms. Loosened to `\[(?![^\]\n]*\])`: an `[` is orphan only when no `]` follows before end of line.
41
+
42
+ ### Changed
43
+ - `sync --comments-only` summary distinguishes `placed` / `already present` / `unmatched` instead of subtracting before/after counts. Re-syncs now report "6 already present (skipped to avoid duplication)" instead of misreporting them as fully placed or fully unmatched. New `outStats` channel from `insertCommentsIntoMarkdown`.
44
+
45
+ ## [0.9.10] - 2026-04-30
46
+
47
+ ### Fixed
48
+ - `stripAnnotations` stripped `[anchor]{.mark}` spans even when `keepComments=true`, leaving the dual-build marker generator with no anchor text and collapsing every multi-word anchor to a single fallback word in the rebuilt docx. Now preserves anchor spans that belong to retained `{>>...<<}` comments.
49
+ - Comments authored at the very start of a Word section landed before the markdown file's `# Heading` line and rendered in the previous section. Added `pushPastSectionHeading` so position-0 comments advance to the first body paragraph of the section they were authored in.
50
+ - Empty-anchor comments fell through to proportional placement even when before/after context uniquely identified the position, landing mid-word or splitting unrelated phrases. Context match now runs first; proportional placement is the fallback.
51
+ - When an anchor appeared multiple times in the search window (repeated phrasing, formulaic prose), `insertCommentsIntoMarkdown` always picked the first occurrence. Now picks the occurrence closest to the docx-derived insert position.
52
+
53
+ ## [0.9.7] - 2026-04-29
54
+
55
+ ### Added
56
+ - `rev sync --comments-only` — import only Word comments at fuzzy-matched anchors, leaving prose byte-identical. Use when the markdown was revised between sending the docx out for review and receiving it back; applying track changes from a stale draft would clobber newer edits.
57
+ - `rev verify-anchors <docx>` — drift report classifying every comment as `clean` / `drift` / `context-only` / `ambiguous` / `unmatched` against the current section markdown. Pair with `--comments-only` to plan placement before sync. Supports `--json` for scripting.
58
+ - `extractHeadings()` in `word-extraction.ts` read heading paragraphs directly from `<w:pStyle>` styles, returning text + level + position in the same coordinate system as comment anchors.
59
+ - Shared `lib/commands/section-boundaries.ts` — single source of truth that maps `sections.yaml` to docx text positions, used by both sync and verify-anchors.
60
+ - Shared `lib/anchor-match.ts` — pure anchor-matching primitives (`findAnchorInText`, `stripCriticMarkup`, `classifyStrategy`) so sync (insertion) and verify-anchors (drift reporting) use the same fallback strategies.
61
+ - New tests: `test/anchor-match.test.js` (11 cases covering each fallback strategy and the quality classifier).
62
+
63
+ ### Fixed
64
+ - **Section detection mistook prose for headings.** The old keyword finder scanned the concatenated body text and would match "results across countries" as the Results heading or skip the real Methods heading because the structured-abstract label `Methods:` lost its colon during text-run concatenation. Replaced with paragraph-style-based heading extraction, so boundaries now reflect actual heading paragraphs. Affects the new commands; the existing sync flow already used pandoc-derived headings and was unaffected.
65
+ - `stripCriticMarkup` regex used `[^<]*` and silently failed on comments containing `<` characters (e.g. `pre-industrial trade (<1825)`). Switched to non-greedy `[\s\S]*?`.
66
+ - `insertCommentsIntoMarkdown` always prepended a leading space when there was no anchor, accumulating multiple spaces when several comments shared a position 0 anchor. Removed the heuristic; comments insert at exact position so prose stays byte-identical except for the inserted blocks.
67
+ - `verify-anchors` crashed with a stack trace when given a non-docx file (e.g. an `.md` path). Now reports a friendly error.
68
+
69
+ ### Changed
70
+ - New flag is `--comments-only` (positive form). The originally proposed `--no-overwrite` was dropped because Commander assigns `--no-X` to `options.x === false` rather than `options.noX === true`, which made the flag silently ignored.
71
+ - `insertCommentsIntoMarkdown` now accepts `wrapAnchor?: boolean` (default `true`). When `false`, comment blocks are inserted next to the anchor without `[anchor]{.mark}` wrapping. `--comments-only` uses this so multiple comments sharing an anchor (e.g. 6 reviewers commenting on the same word) no longer produce nested broken CriticMarkup.
72
+
73
+ ## [0.7.1] - 2025-01-02
74
+
75
+ ### Added
76
+ - Writing Markdown guide in docs (tables, equations, citations, cross-refs)
77
+ - Grid table syntax documentation for merged cells
78
+
79
+ ### Changed
80
+ - README restructured for better scannability (490 → 290 lines)
81
+ - Install section moved up for faster onboarding
82
+ - Added Highlights section with quick feature overview
83
+ - Condensed overlapping sections
84
+
85
+ ## [0.7.0] - 2025-01-02
86
+
87
+ ### Added
88
+ - API rate limiting with exponential backoff for Crossref/DataCite/doi.org APIs
89
+ - Windows support in CI matrix
90
+ - Architecture documentation for contributors (`ARCHITECTURE.md`)
91
+ - Exclusion patterns for cross-reference false positives (e.g., "Table of Contents")
92
+ - Timeout support for PDF extraction (30s default)
93
+
94
+ ### Changed
95
+ - Consolidated YAML dependencies (removed `js-yaml`, using `yaml` package only)
96
+ - Improved annotation false positive detection (code blocks, URLs, LaTeX patterns)
97
+ - Enhanced error messages for Word import and PDF extraction
98
+ - Updated User-Agent strings for API requests
99
+ - Improved README with problem statement and quick example
100
+
101
+ ### Fixed
102
+ - CI lint step now checks all command files separately
103
+ - Windows temp directory paths in tests
104
+
105
+ ## [0.3.2] - 2024-12-29
106
+
107
+ ### Added
108
+ - Full TypeScript type definitions (`types/index.d.ts`)
109
+ - GitHub Actions CI workflow (Node 18/20/22)
110
+ - ESM subpath exports for all library modules
111
+ - CLI integration tests (26 tests)
112
+ - Comprehensive test coverage: 419 tests across 18 modules
113
+
114
+ ### Fixed
115
+ - DOI skip detection: `% no-doi` comments now correctly apply only to the next entry
116
+
117
+ ### Changed
118
+ - Added `engines` field requiring Node.js >=18.0.0
119
+ - Updated README with badges (npm, CI, Node.js, License)
120
+
121
+ ## [0.3.1] - 2024-12-28
122
+
123
+ ### Fixed
124
+ - Equation extraction test assertions
125
+ - Minor bug fixes
126
+
127
+ ## [0.3.0] - 2024-12-27
128
+
129
+ ### Added
130
+ - DOI validation via Crossref and DataCite APIs (`rev doi check`)
131
+ - DOI lookup for missing entries (`rev doi lookup`)
132
+ - DOI fetch and add commands (`rev doi fetch`, `rev doi add`)
133
+ - Citation validation against bibliography (`rev citations`)
134
+ - LaTeX equation extraction (`rev equations list`)
135
+ - Word equation import OMML → LaTeX (`rev equations from-word`)
136
+ - Response letter generation (`rev response`)
137
+ - Journal validation profiles (`rev validate --journal`)
138
+ - Advanced figure/table reference patterns (Figs. 1-3, Fig. 1a-c)
139
+
140
+ ### Changed
141
+ - Improved cross-reference pattern detection
142
+ - Enhanced Word import with better section splitting
143
+
144
+ ## [0.2.1] - 2024-12-26
145
+
146
+ ### Added
147
+ - Table of contents option (`rev build --toc`)
148
+ - CSV export for comments (`rev comments --export`)
149
+ - Anonymize command for blind review (`rev anonymize`)
150
+ - Formatting utilities (tables, boxes, spinners)
151
+
152
+ ## [0.2.0] - 2024-12-25
153
+
154
+ ### Added
155
+ - Integrated build system (`rev build pdf/docx/tex`)
156
+ - Comment reply functionality (`rev reply`)
157
+ - Word document bootstrap (`rev import` creates project from .docx)
158
+ - Section-aware import (`rev sections`)
159
+ - Cross-reference migration (`rev migrate`)
160
+
161
+ ### Changed
162
+ - Renamed project to docrev
163
+ - Published to npm
164
+
165
+ ## [0.1.0] - 2024-12-24
166
+
167
+ ### Added
168
+ - Initial release
169
+ - CriticMarkup annotation parsing
170
+ - Word ↔ Markdown round-trips
171
+ - Interactive review TUI (`rev review`)
172
+ - Comment management (`rev comments`, `rev resolve`)
173
+ - Project templates (`rev new`)