@dev-pi2pie/word-counter 0.1.4 → 0.1.5-canary.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.
- package/README.md +27 -0
- package/dist/cjs/index.cjs +7 -30
- package/dist/cjs/index.cjs.map +1 -1
- package/dist/esm/bin.mjs +487 -226
- package/dist/esm/bin.mjs.map +1 -1
- package/dist/esm/index.mjs +7 -29
- package/dist/esm/index.mjs.map +1 -1
- package/dist/esm/worker/count-worker.mjs +8 -32
- package/dist/esm/worker/count-worker.mjs.map +1 -1
- package/dist/esm/worker-pool.mjs +5 -3
- package/dist/esm/worker-pool.mjs.map +1 -1
- package/package.json +5 -5
package/README.md
CHANGED
|
@@ -164,6 +164,33 @@ word-counter --print-jobs-limit
|
|
|
164
164
|
|
|
165
165
|
`--print-jobs-limit` must be used alone (no other inputs or runtime flags).
|
|
166
166
|
|
|
167
|
+
### Doctor (`doctor`)
|
|
168
|
+
|
|
169
|
+
Use `doctor` to verify whether the current host can run `word-counter` reliably:
|
|
170
|
+
|
|
171
|
+
```bash
|
|
172
|
+
word-counter doctor
|
|
173
|
+
word-counter doctor --format json
|
|
174
|
+
word-counter doctor --format json --pretty
|
|
175
|
+
```
|
|
176
|
+
|
|
177
|
+
Doctor scope in v1:
|
|
178
|
+
|
|
179
|
+
- checks runtime support policy against Node.js `>=20`
|
|
180
|
+
- verifies `Intl.Segmenter` availability plus word/grapheme constructor health
|
|
181
|
+
- reports batch jobs host limits using the same heuristics as `--print-jobs-limit`
|
|
182
|
+
- reports worker-route preflight signals and the worker-disable env toggle that affects worker availability
|
|
183
|
+
|
|
184
|
+
Doctor output contract:
|
|
185
|
+
|
|
186
|
+
- default output is human-readable text
|
|
187
|
+
- `--format json` prints compact machine-readable JSON
|
|
188
|
+
- `--format json --pretty` prints indented JSON
|
|
189
|
+
- doctor exits with code `0` for `ok` / `warn`, `1` for invalid doctor usage, and `2` for runtime `fail`
|
|
190
|
+
- doctor does not accept counting inputs, `--path`, `--jobs`, or other counting/debug flags
|
|
191
|
+
|
|
192
|
+
For a field-by-field explanation of doctor text and JSON output, see [`docs/doctor-usage-guide.md`](docs/doctor-usage-guide.md).
|
|
193
|
+
|
|
167
194
|
For full policy details, JSON parity expectations (`--misc`, `--total-of whitespace,words`), and benchmark standards, see [`docs/batch-jobs-usage-guide.md`](docs/batch-jobs-usage-guide.md).
|
|
168
195
|
|
|
169
196
|
### Stable Path Resolution Contract
|
package/dist/cjs/index.cjs
CHANGED
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
let yaml = require("yaml");
|
|
2
|
-
|
|
3
2
|
//#region src/wc/segmenter.ts
|
|
4
3
|
const segmenterCache = /* @__PURE__ */ new Map();
|
|
5
4
|
const graphemeSegmenterCache = /* @__PURE__ */ new Map();
|
|
@@ -33,13 +32,11 @@ function countCharsForLocale(text, locale) {
|
|
|
33
32
|
for (const _segment of segmenter.segment(text)) count++;
|
|
34
33
|
return count;
|
|
35
34
|
}
|
|
36
|
-
|
|
37
35
|
//#endregion
|
|
38
36
|
//#region src/utils/append-all.ts
|
|
39
37
|
function appendAll(target, source) {
|
|
40
38
|
for (const item of source) target.push(item);
|
|
41
39
|
}
|
|
42
|
-
|
|
43
40
|
//#endregion
|
|
44
41
|
//#region src/wc/non-words.ts
|
|
45
42
|
const emojiRegex = /(?:\p{Extended_Pictographic}|\p{Emoji_Presentation})/u;
|
|
@@ -153,7 +150,6 @@ function createWhitespaceCounts() {
|
|
|
153
150
|
other: 0
|
|
154
151
|
};
|
|
155
152
|
}
|
|
156
|
-
|
|
157
153
|
//#endregion
|
|
158
154
|
//#region src/wc/analyze.ts
|
|
159
155
|
function analyzeChunk(chunk, collectNonWords, includeWhitespace) {
|
|
@@ -253,7 +249,6 @@ function aggregateByLocale(chunks) {
|
|
|
253
249
|
}
|
|
254
250
|
return order.map((locale) => map.get(locale));
|
|
255
251
|
}
|
|
256
|
-
|
|
257
252
|
//#endregion
|
|
258
253
|
//#region src/wc/mode.ts
|
|
259
254
|
const MODE_ALIASES = {
|
|
@@ -321,10 +316,7 @@ function normalizeMode(input) {
|
|
|
321
316
|
function resolveMode(input, fallback = "chunk") {
|
|
322
317
|
return normalizeMode(input) ?? fallback;
|
|
323
318
|
}
|
|
324
|
-
|
|
325
|
-
//#endregion
|
|
326
|
-
//#region src/wc/latin-hints.ts
|
|
327
|
-
const DEFAULT_LATIN_HINT_RULES_SOURCE = [
|
|
319
|
+
const DEFAULT_LATIN_HINT_RULES = Object.freeze([
|
|
328
320
|
{
|
|
329
321
|
tag: "de",
|
|
330
322
|
pattern: "[äöüÄÖÜß]"
|
|
@@ -361,13 +353,10 @@ const DEFAULT_LATIN_HINT_RULES_SOURCE = [
|
|
|
361
353
|
tag: "is",
|
|
362
354
|
pattern: "[ðÐþÞ]"
|
|
363
355
|
}
|
|
364
|
-
];
|
|
365
|
-
const DEFAULT_LATIN_HINT_RULES = Object.freeze(DEFAULT_LATIN_HINT_RULES_SOURCE.map((rule) => Object.freeze({ ...rule })));
|
|
366
|
-
|
|
356
|
+
].map((rule) => Object.freeze({ ...rule })));
|
|
367
357
|
//#endregion
|
|
368
358
|
//#region src/wc/locale-detect.ts
|
|
369
359
|
const DEFAULT_LOCALE = "und-Latn";
|
|
370
|
-
const DEFAULT_HAN_TAG = "und-Hani";
|
|
371
360
|
const MAX_LATIN_HINT_PATTERN_LENGTH = 256;
|
|
372
361
|
const regex = {
|
|
373
362
|
hiragana: /\p{Script=Hiragana}/u,
|
|
@@ -483,18 +472,17 @@ function detectLocaleForChar(char, previousLocale, options = {}, context = resol
|
|
|
483
472
|
if (regex.thai.test(char)) return "th";
|
|
484
473
|
if (regex.han.test(char)) {
|
|
485
474
|
if (allowJapaneseHanCarry && previousLocale && previousLocale.startsWith("ja")) return previousLocale;
|
|
486
|
-
return context.hanHint ??
|
|
475
|
+
return context.hanHint ?? "und-Hani";
|
|
487
476
|
}
|
|
488
477
|
if (regex.latin.test(char)) {
|
|
489
478
|
const hintedLocale = detectLatinLocale(char, context);
|
|
490
|
-
if (hintedLocale !==
|
|
491
|
-
if (allowLatinLocaleCarry && previousLocale && isLatinLocale(previousLocale, context) && previousLocale !==
|
|
479
|
+
if (hintedLocale !== "und-Latn") return hintedLocale;
|
|
480
|
+
if (allowLatinLocaleCarry && previousLocale && isLatinLocale(previousLocale, context) && previousLocale !== "und-Latn") return previousLocale;
|
|
492
481
|
if (context.latinHint) return context.latinHint;
|
|
493
482
|
return DEFAULT_LOCALE;
|
|
494
483
|
}
|
|
495
484
|
return null;
|
|
496
485
|
}
|
|
497
|
-
|
|
498
486
|
//#endregion
|
|
499
487
|
//#region src/wc/segment.ts
|
|
500
488
|
const HARD_BOUNDARY_REGEX = /[\r\n,.!?;:,、。!?;:.。、]/u;
|
|
@@ -531,7 +519,7 @@ function segmentTextByLocale(text, options = {}) {
|
|
|
531
519
|
continue;
|
|
532
520
|
}
|
|
533
521
|
if (targetLocale !== currentLocale && detected !== null) {
|
|
534
|
-
if (currentLocale ===
|
|
522
|
+
if (currentLocale === "und-Latn" && isLatinLocale(targetLocale, context)) {
|
|
535
523
|
const promotionBreakIndex = findLastLatinPromotionBreakIndex(buffer);
|
|
536
524
|
if (promotionBreakIndex === -1) {
|
|
537
525
|
currentLocale = targetLocale;
|
|
@@ -598,7 +586,6 @@ function mergeAdjacentChunks(chunks) {
|
|
|
598
586
|
merged.push(last);
|
|
599
587
|
return merged;
|
|
600
588
|
}
|
|
601
|
-
|
|
602
589
|
//#endregion
|
|
603
590
|
//#region src/wc/wc.ts
|
|
604
591
|
function wordCounter(text, options = {}) {
|
|
@@ -713,11 +700,9 @@ function collectNonWordsAggregate(analyzed, enabled) {
|
|
|
713
700
|
}
|
|
714
701
|
return collection;
|
|
715
702
|
}
|
|
716
|
-
|
|
717
703
|
//#endregion
|
|
718
704
|
//#region src/wc/index.ts
|
|
719
705
|
var wc_default = wordCounter;
|
|
720
|
-
|
|
721
706
|
//#endregion
|
|
722
707
|
//#region src/markdown/toml/arrays.ts
|
|
723
708
|
function ensureArrayContainer(result, key) {
|
|
@@ -733,7 +718,6 @@ function flattenArrayTables(result) {
|
|
|
733
718
|
result[key] = value.map((entry) => Object.entries(entry).map(([entryKey, entryValue]) => `${entryKey}=${entryValue}`).join(", ")).join(" | ");
|
|
734
719
|
}
|
|
735
720
|
}
|
|
736
|
-
|
|
737
721
|
//#endregion
|
|
738
722
|
//#region src/markdown/toml/keys.ts
|
|
739
723
|
function stripKeyQuotes(key) {
|
|
@@ -752,7 +736,6 @@ function normalizeKeyPath(key) {
|
|
|
752
736
|
if (segments.some((segment) => !segment)) return null;
|
|
753
737
|
return segments.join(".");
|
|
754
738
|
}
|
|
755
|
-
|
|
756
739
|
//#endregion
|
|
757
740
|
//#region src/markdown/toml/strings.ts
|
|
758
741
|
function stripInlineComment(line) {
|
|
@@ -801,7 +784,6 @@ function parseStringLiteral(value) {
|
|
|
801
784
|
if (value.startsWith("'") && value.endsWith("'")) return value.slice(1, -1);
|
|
802
785
|
return null;
|
|
803
786
|
}
|
|
804
|
-
|
|
805
787
|
//#endregion
|
|
806
788
|
//#region src/markdown/toml/values.ts
|
|
807
789
|
function parsePrimitive(raw) {
|
|
@@ -959,7 +941,6 @@ function toPlainText(value) {
|
|
|
959
941
|
if (Array.isArray(value)) return value.map((item) => String(item)).join(", ");
|
|
960
942
|
return String(value);
|
|
961
943
|
}
|
|
962
|
-
|
|
963
944
|
//#endregion
|
|
964
945
|
//#region src/markdown/toml/parse-frontmatter.ts
|
|
965
946
|
function parseTomlFrontmatter(frontmatter) {
|
|
@@ -1043,7 +1024,6 @@ function parseTomlFrontmatter(frontmatter) {
|
|
|
1043
1024
|
flattenArrayTables(result);
|
|
1044
1025
|
return result;
|
|
1045
1026
|
}
|
|
1046
|
-
|
|
1047
1027
|
//#endregion
|
|
1048
1028
|
//#region src/markdown/parse-markdown.ts
|
|
1049
1029
|
const FENCE_TO_TYPE = {
|
|
@@ -1178,7 +1158,6 @@ function parseMarkdown(input) {
|
|
|
1178
1158
|
frontmatterType: openingType
|
|
1179
1159
|
};
|
|
1180
1160
|
}
|
|
1181
|
-
|
|
1182
1161
|
//#endregion
|
|
1183
1162
|
//#region src/markdown/section-count.ts
|
|
1184
1163
|
function normalizeText(value) {
|
|
@@ -1243,13 +1222,11 @@ function countSections(input, section, options = {}) {
|
|
|
1243
1222
|
items
|
|
1244
1223
|
};
|
|
1245
1224
|
}
|
|
1246
|
-
|
|
1247
1225
|
//#endregion
|
|
1248
1226
|
//#region src/utils/show-singular-or-plural-word.ts
|
|
1249
1227
|
function showSingularOrPluralWord(count, word) {
|
|
1250
1228
|
return `${count} ${word}${count === 1 ? "" : "s"}`;
|
|
1251
1229
|
}
|
|
1252
|
-
|
|
1253
1230
|
//#endregion
|
|
1254
1231
|
//#region src/index.cjs.ts
|
|
1255
1232
|
const cjsExports = Object.assign(wc_default, {
|
|
@@ -1263,6 +1240,6 @@ const cjsExports = Object.assign(wc_default, {
|
|
|
1263
1240
|
showSingularOrPluralWord
|
|
1264
1241
|
});
|
|
1265
1242
|
module.exports = cjsExports;
|
|
1266
|
-
|
|
1267
1243
|
//#endregion
|
|
1244
|
+
|
|
1268
1245
|
//# sourceMappingURL=index.cjs.map
|
package/dist/cjs/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.cjs","names":["wordCounter","wordCounter"],"sources":["../../src/wc/segmenter.ts","../../src/utils/append-all.ts","../../src/wc/non-words.ts","../../src/wc/analyze.ts","../../src/wc/mode.ts","../../src/wc/latin-hints.ts","../../src/wc/locale-detect.ts","../../src/wc/segment.ts","../../src/wc/wc.ts","../../src/wc/index.ts","../../src/markdown/toml/arrays.ts","../../src/markdown/toml/keys.ts","../../src/markdown/toml/strings.ts","../../src/markdown/toml/values.ts","../../src/markdown/toml/parse-frontmatter.ts","../../src/markdown/parse-markdown.ts","../../src/markdown/section-count.ts","../../src/utils/show-singular-or-plural-word.ts","../../src/index.cjs.ts"],"sourcesContent":["const segmenterCache = new Map<string, Intl.Segmenter>();\nconst graphemeSegmenterCache = new Map<string, Intl.Segmenter>();\n\nexport function getSegmenter(locale: string): Intl.Segmenter {\n const cached = segmenterCache.get(locale);\n if (cached) {\n return cached;\n }\n const segmenter = new Intl.Segmenter(locale, { granularity: \"word\" });\n segmenterCache.set(locale, segmenter);\n return segmenter;\n}\n\nexport function getGraphemeSegmenter(locale: string): Intl.Segmenter {\n const cached = graphemeSegmenterCache.get(locale);\n if (cached) {\n return cached;\n }\n const segmenter = new Intl.Segmenter(locale, { granularity: \"grapheme\" });\n graphemeSegmenterCache.set(locale, segmenter);\n return segmenter;\n}\n\nfunction supportsSegmenter(): boolean {\n return typeof Intl !== \"undefined\" && typeof Intl.Segmenter === \"function\";\n}\n\nexport function countWordsForLocale(text: string, locale: string): number {\n const segmenter = getSegmenter(locale);\n let count = 0;\n for (const segment of segmenter.segment(text)) {\n if (segment.isWordLike) {\n count++;\n }\n }\n return count;\n}\n\nexport function countCharsForLocale(text: string, locale: string): number {\n if (!supportsSegmenter()) {\n return Array.from(text).length;\n }\n const segmenter = getGraphemeSegmenter(locale);\n let count = 0;\n for (const _segment of segmenter.segment(text)) {\n count++;\n }\n return count;\n}\n","export function appendAll<T>(target: T[], source: readonly T[]): void {\n for (const item of source) {\n target.push(item);\n }\n}\n","import type { NonWordCollection, WhitespaceCounts } from \"./types\";\nimport { appendAll } from \"../utils/append-all\";\n\nconst emojiRegex = /(?:\\p{Extended_Pictographic}|\\p{Emoji_Presentation})/u;\nconst emojiPresentationRegex = /\\p{Emoji_Presentation}/u;\nconst keycapEmojiRegex = /[0-9#*]\\uFE0F?\\u20E3/u;\nconst symbolRegex = /\\p{S}/u;\nconst punctuationRegex = /\\p{P}/u;\nconst whitespaceRegex = /\\s/u;\nconst newlineChars = new Set([\"\\n\", \"\\r\", \"\\u2028\", \"\\u2029\"]);\n\nexport function createNonWordCollection(): NonWordCollection {\n return {\n emoji: [],\n symbols: [],\n punctuation: [],\n counts: {\n emoji: 0,\n symbols: 0,\n punctuation: 0,\n },\n };\n}\n\nexport function addNonWord(\n collection: NonWordCollection,\n category: \"emoji\" | \"symbol\" | \"punctuation\",\n segment: string,\n): void {\n if (category === \"emoji\") {\n collection.emoji.push(segment);\n collection.counts.emoji += 1;\n return;\n }\n if (category === \"symbol\") {\n collection.symbols.push(segment);\n collection.counts.symbols += 1;\n return;\n }\n collection.punctuation.push(segment);\n collection.counts.punctuation += 1;\n}\n\nexport function addWhitespace(\n collection: NonWordCollection,\n segment: string,\n): number {\n let whitespace = collection.whitespace;\n let count = 0;\n for (const char of segment) {\n if (char === \" \") {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.spaces += 1;\n count += 1;\n continue;\n }\n if (char === \"\\t\") {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.tabs += 1;\n count += 1;\n continue;\n }\n if (newlineChars.has(char)) {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.newlines += 1;\n count += 1;\n continue;\n }\n if (whitespaceRegex.test(char)) {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.other += 1;\n count += 1;\n }\n }\n\n if (count > 0) {\n collection.whitespace = whitespace ?? createWhitespaceCounts();\n collection.counts.whitespace = (collection.counts.whitespace ?? 0) + count;\n }\n\n return count;\n}\n\nexport function classifyNonWordSegment(\n segment: string,\n): \"emoji\" | \"symbol\" | \"punctuation\" | null {\n const hasEmojiVariationSelector = segment.includes(\"\\uFE0F\");\n if (\n keycapEmojiRegex.test(segment) ||\n emojiPresentationRegex.test(segment) ||\n (hasEmojiVariationSelector && emojiRegex.test(segment))\n ) {\n return \"emoji\";\n }\n if (symbolRegex.test(segment)) {\n return \"symbol\";\n }\n if (punctuationRegex.test(segment)) {\n return \"punctuation\";\n }\n return null;\n}\n\nexport function mergeNonWordCollections(\n target: NonWordCollection,\n source: NonWordCollection,\n): NonWordCollection {\n if (source.counts.emoji > 0) {\n appendAll(target.emoji, source.emoji);\n target.counts.emoji += source.counts.emoji;\n }\n if (source.counts.symbols > 0) {\n appendAll(target.symbols, source.symbols);\n target.counts.symbols += source.counts.symbols;\n }\n if (source.counts.punctuation > 0) {\n appendAll(target.punctuation, source.punctuation);\n target.counts.punctuation += source.counts.punctuation;\n }\n if (source.counts.whitespace && source.counts.whitespace > 0 && source.whitespace) {\n const whitespace = target.whitespace ?? createWhitespaceCounts();\n whitespace.spaces += source.whitespace.spaces;\n whitespace.tabs += source.whitespace.tabs;\n whitespace.newlines += source.whitespace.newlines;\n whitespace.other += source.whitespace.other;\n target.whitespace = whitespace;\n target.counts.whitespace = (target.counts.whitespace ?? 0) + source.counts.whitespace;\n }\n return target;\n}\n\nfunction createWhitespaceCounts(): WhitespaceCounts {\n return { spaces: 0, tabs: 0, newlines: 0, other: 0 };\n}\n","import { countCharsForLocale, getSegmenter } from \"./segmenter\";\nimport {\n addNonWord,\n addWhitespace,\n classifyNonWordSegment,\n createNonWordCollection,\n mergeNonWordCollections,\n} from \"./non-words\";\nimport { appendAll } from \"../utils/append-all\";\nimport type {\n CharCollectorBreakdown,\n ChunkAnalysis,\n CollectorBreakdown,\n LocaleChunk,\n NonWordCollection,\n} from \"./types\";\n\ntype CharAnalysis = LocaleChunk & { chars: number; nonWords?: NonWordCollection };\ntype CharChunkAnalysis = CharAnalysis & { wordChars: number; nonWordChars: number };\n\nexport function analyzeChunk(\n chunk: LocaleChunk,\n collectNonWords?: boolean,\n includeWhitespace?: boolean,\n): ChunkAnalysis {\n const segmenter = getSegmenter(chunk.locale);\n const segments: string[] = [];\n const nonWords: NonWordCollection | null = collectNonWords\n ? createNonWordCollection()\n : null;\n for (const part of segmenter.segment(chunk.text)) {\n if (part.isWordLike) {\n segments.push(part.segment);\n } else if (collectNonWords && nonWords) {\n if (includeWhitespace) {\n addWhitespace(nonWords, part.segment);\n }\n const category = classifyNonWordSegment(part.segment);\n if (category) {\n addNonWord(nonWords, category, part.segment);\n }\n }\n }\n return {\n locale: chunk.locale,\n text: chunk.text,\n segments,\n words: segments.length,\n nonWords: nonWords ?? undefined,\n };\n}\n\nexport function analyzeCharChunk(\n chunk: LocaleChunk,\n collectNonWords?: boolean,\n includeWhitespace?: boolean,\n): CharChunkAnalysis {\n const segmenter = getSegmenter(chunk.locale);\n const nonWords: NonWordCollection | null = collectNonWords\n ? createNonWordCollection()\n : null;\n let chars = 0;\n let wordChars = 0;\n let nonWordChars = 0;\n\n for (const part of segmenter.segment(chunk.text)) {\n if (part.isWordLike) {\n const count = countCharsForLocale(part.segment, chunk.locale);\n chars += count;\n wordChars += count;\n continue;\n }\n\n if (collectNonWords && nonWords) {\n let whitespaceCount = 0;\n if (includeWhitespace) {\n whitespaceCount = addWhitespace(nonWords, part.segment);\n }\n const category = classifyNonWordSegment(part.segment);\n if (category) {\n addNonWord(nonWords, category, part.segment);\n }\n if (category || whitespaceCount > 0) {\n const count = countCharsForLocale(part.segment, chunk.locale);\n chars += count;\n nonWordChars += count;\n }\n }\n }\n\n return {\n locale: chunk.locale,\n text: chunk.text,\n chars,\n wordChars,\n nonWordChars,\n nonWords: nonWords ?? undefined,\n };\n}\n\nexport function aggregateCharsByLocale(\n chunks: CharChunkAnalysis[],\n): Array<CharCollectorBreakdown & { wordChars: number; nonWordChars: number }> {\n const order: string[] = [];\n const map = new Map<\n string,\n CharCollectorBreakdown & { wordChars: number; nonWordChars: number }\n >();\n\n for (const chunk of chunks) {\n const existing = map.get(chunk.locale);\n if (existing) {\n existing.chars += chunk.chars;\n existing.wordChars += chunk.wordChars;\n existing.nonWordChars += chunk.nonWordChars;\n if (chunk.nonWords) {\n if (!existing.nonWords) {\n existing.nonWords = createNonWordCollection();\n }\n mergeNonWordCollections(existing.nonWords, chunk.nonWords);\n }\n continue;\n }\n\n order.push(chunk.locale);\n map.set(chunk.locale, {\n locale: chunk.locale,\n chars: chunk.chars,\n wordChars: chunk.wordChars,\n nonWordChars: chunk.nonWordChars,\n nonWords: chunk.nonWords\n ? mergeNonWordCollections(createNonWordCollection(), chunk.nonWords)\n : undefined,\n });\n }\n\n return order.map((locale) => map.get(locale)!);\n}\n\nexport function aggregateByLocale(\n chunks: ChunkAnalysis[]\n): CollectorBreakdown[] {\n const order: string[] = [];\n const map = new Map<string, CollectorBreakdown>();\n\n for (const chunk of chunks) {\n const existing = map.get(chunk.locale);\n if (existing) {\n existing.words += chunk.words;\n appendAll(existing.segments, chunk.segments);\n continue;\n }\n\n order.push(chunk.locale);\n map.set(chunk.locale, {\n locale: chunk.locale,\n words: chunk.words,\n segments: [...chunk.segments],\n });\n }\n\n return order.map((locale) => map.get(locale)!);\n}\n","import type { WordCounterMode } from \"./types\";\n\nconst MODE_ALIASES: Record<string, WordCounterMode> = {\n chunk: \"chunk\",\n chunks: \"chunk\",\n segments: \"segments\",\n segment: \"segments\",\n seg: \"segments\",\n collector: \"collector\",\n collect: \"collector\",\n colle: \"collector\",\n char: \"char\",\n chars: \"char\",\n character: \"char\",\n characters: \"char\",\n \"char-collector\": \"char-collector\",\n};\n\nconst CHAR_MODE_ALIASES = new Set([\"char\", \"chars\", \"character\", \"characters\"]);\nconst COLLECTOR_MODE_ALIASES = new Set([\"collector\", \"collect\", \"colle\", \"col\"]);\n\nfunction collapseSeparators(value: string): string {\n return value.replace(/[-_\\s]+/g, \"\");\n}\n\nfunction isComposedCharCollectorFromTokens(value: string): boolean {\n const tokens = value\n .split(/[-_\\s]+/)\n .map((token) => token.trim())\n .filter((token) => token.length > 0);\n if (tokens.length < 2) {\n return false;\n }\n\n let hasCharAlias = false;\n let hasCollectorAlias = false;\n for (const token of tokens) {\n if (CHAR_MODE_ALIASES.has(token)) {\n hasCharAlias = true;\n continue;\n }\n if (COLLECTOR_MODE_ALIASES.has(token)) {\n hasCollectorAlias = true;\n continue;\n }\n return false;\n }\n\n return hasCharAlias && hasCollectorAlias;\n}\n\nfunction isComposedCharCollectorCompact(value: string): boolean {\n for (const charAlias of CHAR_MODE_ALIASES) {\n for (const collectorAlias of COLLECTOR_MODE_ALIASES) {\n if (value === `${charAlias}${collectorAlias}` || value === `${collectorAlias}${charAlias}`) {\n return true;\n }\n }\n }\n return false;\n}\n\nexport function normalizeMode(input?: string | null): WordCounterMode | null {\n if (!input) {\n return null;\n }\n const normalized = input.trim().toLowerCase();\n const direct = MODE_ALIASES[normalized];\n if (direct) {\n return direct;\n }\n\n if (isComposedCharCollectorFromTokens(normalized)) {\n return \"char-collector\";\n }\n\n const compact = collapseSeparators(normalized);\n if (isComposedCharCollectorCompact(compact)) {\n return \"char-collector\";\n }\n\n return MODE_ALIASES[compact] ?? null;\n}\n\nexport function resolveMode(\n input?: string | null,\n fallback: WordCounterMode = \"chunk\",\n): WordCounterMode {\n return normalizeMode(input) ?? fallback;\n}\n","import type { LatinHintRule } from \"./types\";\n\nconst DEFAULT_LATIN_HINT_RULES_SOURCE = [\n { tag: \"de\", pattern: \"[äöüÄÖÜß]\" },\n { tag: \"es\", pattern: \"[ñÑ¿¡]\" },\n { tag: \"pt\", pattern: \"[ãõÃÕ]\" },\n { tag: \"fr\", pattern: \"[œŒæÆ]\" },\n { tag: \"pl\", pattern: \"[ąćęłńśźżĄĆĘŁŃŚŹŻ]\" },\n { tag: \"tr\", pattern: \"[ıİğĞşŞ]\" },\n { tag: \"ro\", pattern: \"[ăĂâÂîÎșȘțȚ]\" },\n { tag: \"hu\", pattern: \"[őŐűŰ]\" },\n { tag: \"is\", pattern: \"[ðÐþÞ]\" },\n] satisfies LatinHintRule[];\n\nexport const DEFAULT_LATIN_HINT_RULES: ReadonlyArray<Readonly<LatinHintRule>> = Object.freeze(\n DEFAULT_LATIN_HINT_RULES_SOURCE.map((rule) => Object.freeze({ ...rule })),\n);\n","import type { LatinHintRule } from \"./types\";\nimport { DEFAULT_LATIN_HINT_RULES } from \"./latin-hints\";\n\nexport const DEFAULT_LOCALE = \"und-Latn\";\nexport const DEFAULT_HAN_TAG = \"und-Hani\";\n\nconst MAX_LATIN_HINT_PATTERN_LENGTH = 256;\n\nexport interface LocaleDetectOptions {\n latinLanguageHint?: string;\n latinTagHint?: string;\n latinLocaleHint?: string;\n latinHintRules?: LatinHintRule[];\n useDefaultLatinHints?: boolean;\n hanLanguageHint?: string;\n hanTagHint?: string;\n}\n\ntype ResolvedLatinHintRule = {\n tag: string;\n pattern: RegExp;\n priority: number;\n order: number;\n};\n\nexport type LocaleDetectContext = {\n latinHint?: string;\n hanHint?: string;\n latinHintRules: ResolvedLatinHintRule[];\n latinLocales: Set<string>;\n};\n\nconst regex = {\n hiragana: /\\p{Script=Hiragana}/u,\n katakana: /\\p{Script=Katakana}/u,\n hangul: /\\p{Script=Hangul}/u,\n han: /\\p{Script=Han}/u,\n latin: /\\p{Script=Latin}/u,\n arabic: /\\p{Script=Arabic}/u,\n cyrillic: /\\p{Script=Cyrillic}/u,\n devanagari: /\\p{Script=Devanagari}/u,\n thai: /\\p{Script=Thai}/u,\n};\n\nconst defaultLatinLocales = new Set<string>([\n DEFAULT_LOCALE,\n ...DEFAULT_LATIN_HINT_RULES.map((hint) => hint.tag),\n]);\n\nexport function isLatinLocale(locale: string, context?: LocaleDetectContext): boolean {\n if (context) {\n return context.latinLocales.has(locale);\n }\n return defaultLatinLocales.has(locale);\n}\n\nfunction resolveLatinHint(options: LocaleDetectOptions): string | undefined {\n const latinTagHint = options.latinTagHint?.trim();\n if (latinTagHint) {\n return latinTagHint;\n }\n\n const latinLanguageHint = options.latinLanguageHint?.trim();\n if (latinLanguageHint) {\n return latinLanguageHint;\n }\n\n const latinLocaleHint = options.latinLocaleHint?.trim();\n if (latinLocaleHint) {\n return latinLocaleHint;\n }\n\n return undefined;\n}\n\nfunction resolveHanHint(options: LocaleDetectOptions): string | undefined {\n const hanTagHint = options.hanTagHint?.trim();\n if (hanTagHint) {\n return hanTagHint;\n }\n\n const hanLanguageHint = options.hanLanguageHint?.trim();\n if (hanLanguageHint) {\n return hanLanguageHint;\n }\n\n return undefined;\n}\n\nfunction compileLatinHintPattern(\n pattern: string | RegExp,\n label: string,\n): RegExp {\n const source = typeof pattern === \"string\" ? pattern : pattern.source;\n const hasUnicodeMode =\n typeof pattern !== \"string\" &&\n (pattern.flags.includes(\"u\") || pattern.flags.includes(\"v\"));\n const flags =\n typeof pattern === \"string\"\n ? \"u\"\n : hasUnicodeMode\n ? pattern.flags\n : `${pattern.flags}u`;\n if (source.length === 0) {\n throw new Error(`${label}: pattern must not be empty.`);\n }\n if (source.length > MAX_LATIN_HINT_PATTERN_LENGTH) {\n throw new Error(\n `${label}: pattern must be at most ${MAX_LATIN_HINT_PATTERN_LENGTH} characters.`,\n );\n }\n try {\n return new RegExp(source, flags);\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n throw new Error(`${label}: invalid Unicode regex pattern (${message}).`);\n }\n}\n\nfunction normalizeLatinHintPriority(priority: unknown, label: string): number {\n if (priority === undefined) {\n return 0;\n }\n if (typeof priority !== \"number\" || !Number.isFinite(priority)) {\n throw new Error(`${label}: priority must be a finite number when provided.`);\n }\n return priority;\n}\n\nfunction compileLatinHintRule(\n rule: Readonly<LatinHintRule>,\n order: number,\n label: string,\n): ResolvedLatinHintRule {\n const tag = typeof rule.tag === \"string\" ? rule.tag.trim() : \"\";\n if (!tag) {\n throw new Error(`${label}: tag must be a non-empty string.`);\n }\n const pattern = compileLatinHintPattern(rule.pattern, label);\n const priority = normalizeLatinHintPriority(rule.priority, label);\n return {\n tag,\n pattern,\n priority,\n order,\n };\n}\n\nfunction resolveLatinHintRules(options: LocaleDetectOptions): ResolvedLatinHintRule[] {\n const useDefaultLatinHints = options.useDefaultLatinHints !== false;\n const customRules = options.latinHintRules ?? [];\n const combinedRules: Array<{ rule: Readonly<LatinHintRule>; label: string }> = [];\n\n for (let index = 0; index < customRules.length; index += 1) {\n const rule = customRules[index];\n if (!rule) {\n continue;\n }\n combinedRules.push({\n rule,\n label: `Invalid custom Latin hint rule at index ${index}`,\n });\n }\n\n if (useDefaultLatinHints) {\n for (let index = 0; index < DEFAULT_LATIN_HINT_RULES.length; index += 1) {\n const rule = DEFAULT_LATIN_HINT_RULES[index];\n if (!rule) {\n continue;\n }\n combinedRules.push({\n rule,\n label: `Invalid default Latin hint rule at index ${index}`,\n });\n }\n }\n\n const resolvedRules = combinedRules.map((entry, index) =>\n compileLatinHintRule(entry.rule, index, entry.label),\n );\n\n resolvedRules.sort((left, right) => {\n if (left.priority !== right.priority) {\n return right.priority - left.priority;\n }\n return left.order - right.order;\n });\n\n return resolvedRules;\n}\n\nexport function resolveLocaleDetectContext(\n options: LocaleDetectOptions = {},\n): LocaleDetectContext {\n const latinHint = resolveLatinHint(options);\n const latinHintRules = resolveLatinHintRules(options);\n const latinLocales = new Set<string>([DEFAULT_LOCALE]);\n for (const rule of latinHintRules) {\n latinLocales.add(rule.tag);\n }\n if (latinHint) {\n latinLocales.add(latinHint);\n }\n\n return {\n latinHint,\n hanHint: resolveHanHint(options),\n latinHintRules,\n latinLocales,\n };\n}\n\nfunction detectLatinLocale(char: string, context: LocaleDetectContext): string {\n for (const hint of context.latinHintRules) {\n hint.pattern.lastIndex = 0;\n if (hint.pattern.test(char)) {\n return hint.tag;\n }\n }\n return DEFAULT_LOCALE;\n}\n\nexport function detectLocaleForChar(\n char: string,\n previousLocale?: string | null,\n options: LocaleDetectOptions = {},\n context: LocaleDetectContext = resolveLocaleDetectContext(options),\n allowLatinLocaleCarry = true,\n allowJapaneseHanCarry = true,\n): string | null {\n if (regex.hiragana.test(char) || regex.katakana.test(char)) {\n return \"ja\";\n }\n if (regex.hangul.test(char)) {\n return \"ko\";\n }\n if (regex.arabic.test(char)) {\n return \"ar\";\n }\n if (regex.cyrillic.test(char)) {\n return \"ru\";\n }\n if (regex.devanagari.test(char)) {\n return \"hi\";\n }\n if (regex.thai.test(char)) {\n return \"th\";\n }\n\n if (regex.han.test(char)) {\n if (allowJapaneseHanCarry && previousLocale && previousLocale.startsWith(\"ja\")) {\n return previousLocale;\n }\n return context.hanHint ?? DEFAULT_HAN_TAG;\n }\n\n if (regex.latin.test(char)) {\n const hintedLocale = detectLatinLocale(char, context);\n if (hintedLocale !== DEFAULT_LOCALE) {\n return hintedLocale;\n }\n if (\n allowLatinLocaleCarry &&\n previousLocale &&\n isLatinLocale(previousLocale, context) &&\n previousLocale !== DEFAULT_LOCALE\n ) {\n return previousLocale;\n }\n if (context.latinHint) {\n return context.latinHint;\n }\n return DEFAULT_LOCALE;\n }\n\n return null;\n}\n","import {\n DEFAULT_LOCALE,\n detectLocaleForChar,\n isLatinLocale,\n resolveLocaleDetectContext,\n type LocaleDetectOptions,\n} from \"./locale-detect\";\nimport type { LocaleChunk } from \"./types\";\n\nconst HARD_BOUNDARY_REGEX = /[\\r\\n,.!?;:,、。!?;:.。、]/u;\nconst LATIN_PROMOTION_BREAK_REGEX = /[\\s,.!?;:,、。!?;:.。、]/u;\n\nexport function segmentTextByLocale(\n text: string,\n options: LocaleDetectOptions = {}\n): LocaleChunk[] {\n const context = resolveLocaleDetectContext(options);\n const chunks: LocaleChunk[] = [];\n // Keep currentLocale as a non-null string to simplify type-narrowing.\n let currentLocale: string = DEFAULT_LOCALE;\n let buffer = \"\";\n let bufferHasScript = false;\n let sawCarryBoundary = false;\n\n const updateCarryBoundaryState = (detected: string | null, char: string): void => {\n if (detected !== null) {\n sawCarryBoundary = false;\n return;\n }\n if (HARD_BOUNDARY_REGEX.test(char)) {\n sawCarryBoundary = true;\n }\n };\n\n for (const char of text) {\n const detected = detectLocaleForChar(\n char,\n currentLocale,\n options,\n context,\n !sawCarryBoundary,\n !sawCarryBoundary,\n );\n const targetLocale = detected ?? currentLocale;\n\n // If buffer is empty, this is the first character for a new chunk.\n if (buffer === \"\") {\n currentLocale = targetLocale;\n buffer = char;\n bufferHasScript = detected !== null;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n if (detected !== null && !bufferHasScript) {\n currentLocale = targetLocale;\n buffer += char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n if (targetLocale !== currentLocale && detected !== null) {\n if (currentLocale === DEFAULT_LOCALE && isLatinLocale(targetLocale, context)) {\n const promotionBreakIndex = findLastLatinPromotionBreakIndex(buffer);\n if (promotionBreakIndex === -1) {\n currentLocale = targetLocale;\n buffer += char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n const prefix = buffer.slice(0, promotionBreakIndex + 1);\n const suffix = buffer.slice(promotionBreakIndex + 1);\n if (prefix.length > 0) {\n chunks.push({ locale: currentLocale, text: prefix });\n }\n currentLocale = targetLocale;\n buffer = `${suffix}${char}`;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n // currentLocale is guaranteed to be a string here.\n chunks.push({ locale: currentLocale, text: buffer });\n currentLocale = targetLocale;\n buffer = char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n buffer += char;\n if (detected !== null) {\n bufferHasScript = true;\n }\n updateCarryBoundaryState(detected, char);\n }\n\n if (buffer.length > 0) {\n chunks.push({ locale: currentLocale, text: buffer });\n }\n\n return mergeAdjacentChunks(chunks);\n}\n\nfunction findLastLatinPromotionBreakIndex(buffer: string): number {\n for (let index = buffer.length - 1; index >= 0; index -= 1) {\n const char = buffer[index];\n if (!char) {\n continue;\n }\n if (LATIN_PROMOTION_BREAK_REGEX.test(char)) {\n return index;\n }\n }\n return -1;\n}\n\nfunction mergeAdjacentChunks(chunks: LocaleChunk[]): LocaleChunk[] {\n if (chunks.length === 0) {\n return chunks;\n }\n\n const merged: LocaleChunk[] = [];\n // We already returned for empty arrays above, so the first element is present.\n let last = chunks[0]!;\n\n for (let i = 1; i < chunks.length; i++) {\n const chunk = chunks[i]!;\n if (chunk.locale === last.locale) {\n last = { locale: last.locale, text: last.text + chunk.text };\n } else {\n merged.push(last);\n last = chunk;\n }\n }\n\n merged.push(last);\n return merged;\n}\n","import {\n analyzeCharChunk,\n analyzeChunk,\n aggregateByLocale,\n aggregateCharsByLocale,\n} from \"./analyze\";\nimport { resolveMode } from \"./mode\";\nimport { segmentTextByLocale } from \"./segment\";\nimport { countCharsForLocale, countWordsForLocale } from \"./segmenter\";\nimport { createNonWordCollection, mergeNonWordCollections } from \"./non-words\";\nimport type {\n CharBreakdown,\n CharCollectorBreakdown,\n ChunkBreakdown,\n ChunkWithSegments,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n} from \"./types\";\n\nexport type {\n LatinHintRule,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n WordCounterBreakdown,\n} from \"./types\";\n\nexport { countCharsForLocale, countWordsForLocale, segmentTextByLocale };\nexport { DEFAULT_LATIN_HINT_RULES } from \"./latin-hints\";\n\nexport function wordCounter(\n text: string,\n options: WordCounterOptions = {}\n): WordCounterResult {\n const mode: WordCounterMode = resolveMode(options.mode, \"chunk\");\n const collectNonWords = Boolean(options.nonWords);\n const includeWhitespace = Boolean(options.includeWhitespace);\n const chunks = segmentTextByLocale(text, {\n latinLanguageHint: options.latinLanguageHint,\n latinTagHint: options.latinTagHint,\n latinLocaleHint: options.latinLocaleHint,\n latinHintRules: options.latinHintRules,\n useDefaultLatinHints: options.useDefaultLatinHints,\n hanLanguageHint: options.hanLanguageHint,\n hanTagHint: options.hanTagHint,\n });\n\n if (mode === \"char\" || mode === \"char-collector\") {\n const analyzed = chunks.map((chunk) =>\n analyzeCharChunk(chunk, collectNonWords, includeWhitespace),\n );\n const total = analyzed.reduce((sum, chunk) => sum + chunk.chars, 0);\n const counts = collectNonWords\n ? {\n words: analyzed.reduce((sum, chunk) => sum + chunk.wordChars, 0),\n nonWords: analyzed.reduce((sum, chunk) => sum + chunk.nonWordChars, 0),\n total,\n }\n : undefined;\n\n if (mode === \"char\") {\n const items: CharBreakdown[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n chars: chunk.chars,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n const aggregated = aggregateCharsByLocale(analyzed);\n const items: CharCollectorBreakdown[] = aggregated.map((chunk) => ({\n locale: chunk.locale,\n chars: chunk.chars,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n const analyzed = chunks.map((chunk) =>\n analyzeChunk(chunk, collectNonWords, includeWhitespace),\n );\n const wordsTotal = analyzed.reduce((sum, chunk) => sum + chunk.words, 0);\n const nonWordsTotal = collectNonWords\n ? analyzed.reduce((sum, chunk) => {\n if (!chunk.nonWords) {\n return sum;\n }\n return sum + getNonWordTotal(chunk.nonWords);\n }, 0)\n : 0;\n const total = analyzed.reduce((sum, chunk) => {\n let chunkTotal = chunk.words;\n if (collectNonWords && chunk.nonWords) {\n chunkTotal += getNonWordTotal(chunk.nonWords);\n }\n return sum + chunkTotal;\n }, 0);\n\n const counts = collectNonWords ? { words: wordsTotal, nonWords: nonWordsTotal, total } : undefined;\n\n if (mode === \"segments\") {\n const items: ChunkWithSegments[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n words: chunk.words,\n segments: chunk.segments,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n if (mode === \"collector\") {\n const items = aggregateByLocale(analyzed);\n const nonWords = collectNonWordsAggregate(analyzed, collectNonWords);\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n nonWords,\n },\n };\n }\n\n const items: ChunkBreakdown[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n words: chunk.words,\n nonWords: chunk.nonWords,\n }));\n\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n}\n\nfunction getNonWordTotal(nonWords: NonWordCollection): number {\n return (\n nonWords.counts.emoji +\n nonWords.counts.symbols +\n nonWords.counts.punctuation +\n (nonWords.counts.whitespace ?? 0)\n );\n}\n\n\nfunction collectNonWordsAggregate(\n analyzed: Array<{ nonWords?: NonWordCollection }>,\n enabled: boolean,\n): NonWordCollection | undefined {\n if (!enabled) {\n return undefined;\n }\n const collection = createNonWordCollection();\n for (const chunk of analyzed) {\n if (!chunk.nonWords) {\n continue;\n }\n mergeNonWordCollections(collection, chunk.nonWords);\n }\n return collection;\n}\n","import { wordCounter } from \"./wc\";\n\nexport default wordCounter;\nexport { countCharsForLocale, countWordsForLocale, segmentTextByLocale } from \"./wc\";\nexport { DEFAULT_LATIN_HINT_RULES } from \"./wc\";\nexport type {\n LatinHintRule,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n WordCounterBreakdown,\n} from \"./wc\";\n","export function ensureArrayContainer(\n result: Record<string, unknown>,\n key: string,\n): Record<string, unknown>[] {\n const existing = result[key];\n if (Array.isArray(existing)) {\n return existing as Record<string, unknown>[];\n }\n const list: Record<string, unknown>[] = [];\n result[key] = list;\n return list;\n}\n\nexport function flattenArrayTables(result: Record<string, unknown>): void {\n for (const [key, value] of Object.entries(result)) {\n if (!Array.isArray(value)) {\n continue;\n }\n const flattened = value\n .map((entry) =>\n Object.entries(entry)\n .map(([entryKey, entryValue]) => `${entryKey}=${entryValue}`)\n .join(\", \"),\n )\n .join(\" | \");\n result[key] = flattened;\n }\n}\n","function stripKeyQuotes(key: string): string {\n const trimmed = key.trim();\n if (\n (trimmed.startsWith(\"\\\"\") && trimmed.endsWith(\"\\\"\")) ||\n (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n ) {\n return trimmed.slice(1, -1);\n }\n return trimmed;\n}\n\nexport function normalizeKeyPath(key: string): string | null {\n const trimmed = key.trim();\n if (!trimmed) {\n return null;\n }\n\n if (\n (trimmed.startsWith(\"\\\"\") && trimmed.endsWith(\"\\\"\")) ||\n (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n ) {\n const unquoted = stripKeyQuotes(trimmed);\n return unquoted ? unquoted : null;\n }\n\n const segments = trimmed.split(\".\").map((segment) => segment.trim());\n if (segments.some((segment) => !segment)) {\n return null;\n }\n return segments.join(\".\");\n}\n","export function stripInlineComment(line: string): string {\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n\n for (let i = 0; i < line.length; i += 1) {\n const char = line[i] ?? \"\";\n\n if (inString) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n continue;\n }\n\n if (inString === \"single\" && char === \"'\") {\n inString = null;\n continue;\n }\n\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n continue;\n }\n\n if (char === \"#\") {\n return line.slice(0, i).trimEnd();\n }\n }\n\n return line;\n}\n\nfunction unescapeBasic(input: string): string {\n return input\n .replace(/\\\\\\\\/g, \"\\\\\")\n .replace(/\\\\\"/g, \"\\\"\")\n .replace(/\\\\n/g, \"\\n\")\n .replace(/\\\\t/g, \"\\t\")\n .replace(/\\\\r/g, \"\\r\");\n}\n\nexport function parseStringLiteral(value: string): string | null {\n if (value.startsWith('\"\"\"') && value.endsWith('\"\"\"')) {\n const inner = value.slice(3, -3);\n return unescapeBasic(inner);\n }\n\n if (value.startsWith(\"'''\") && value.endsWith(\"'''\")) {\n return value.slice(3, -3);\n }\n\n if (value.startsWith(\"\\\"\") && value.endsWith(\"\\\"\")) {\n return unescapeBasic(value.slice(1, -1));\n }\n\n if (value.startsWith(\"'\") && value.endsWith(\"'\")) {\n return value.slice(1, -1);\n }\n\n return null;\n}\n","import { normalizeKeyPath } from \"./keys\";\nimport { parseStringLiteral } from \"./strings\";\nimport type { TomlValue } from \"./types\";\n\nfunction parsePrimitive(raw: string): string | number | boolean | null {\n const value = raw.trim();\n if (!value) {\n return null;\n }\n\n const stringLiteral = parseStringLiteral(value);\n if (stringLiteral !== null) {\n return stringLiteral;\n }\n\n if (value === \"true\") {\n return true;\n }\n\n if (value === \"false\") {\n return false;\n }\n\n if (/^[+-]?\\d+(?:\\.\\d+)?(?:[eE][+-]?\\d+)?$/.test(value)) {\n return Number(value);\n }\n\n if (/^\\d{4}-\\d{2}-\\d{2}/.test(value)) {\n return value;\n }\n\n return value;\n}\n\nfunction parseArray(raw: string): Array<string | number | boolean> | null {\n const value = raw.trim();\n if (!value.startsWith(\"[\") || !value.endsWith(\"]\")) {\n return null;\n }\n\n const inner = value.slice(1, -1).trim();\n if (!inner) {\n return [];\n }\n\n const items: Array<string | number | boolean> = [];\n let current = \"\";\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n\n for (let i = 0; i < inner.length; i += 1) {\n const char = inner[i] ?? \"\";\n\n if (inString) {\n current += char;\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n } else if (inString === \"single\" && char === \"'\") {\n inString = null;\n }\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n current += char;\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n current += char;\n continue;\n }\n\n if (char === \",\") {\n const item = parsePrimitive(current);\n if (item === null) {\n return null;\n }\n items.push(item);\n current = \"\";\n continue;\n }\n\n current += char;\n }\n\n const finalItem = parsePrimitive(current);\n if (finalItem === null) {\n return null;\n }\n items.push(finalItem);\n\n return items;\n}\n\nfunction parseInlineTable(raw: string): Record<string, TomlValue> | null {\n const trimmed = raw.trim();\n if (!trimmed.startsWith(\"{\") || !trimmed.endsWith(\"}\")) {\n return null;\n }\n\n const inner = trimmed.slice(1, -1).trim();\n if (!inner) {\n return {};\n }\n\n const pairs: string[] = [];\n let current = \"\";\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n let bracketDepth = 0;\n let braceDepth = 0;\n\n for (let i = 0; i < inner.length; i += 1) {\n const char = inner[i] ?? \"\";\n\n if (inString) {\n current += char;\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n } else if (inString === \"single\" && char === \"'\") {\n inString = null;\n }\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n current += char;\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n current += char;\n continue;\n }\n\n if (char === \"[\") {\n bracketDepth += 1;\n current += char;\n continue;\n }\n\n if (char === \"]\") {\n if (bracketDepth > 0) {\n bracketDepth -= 1;\n }\n current += char;\n continue;\n }\n\n if (char === \"{\") {\n braceDepth += 1;\n current += char;\n continue;\n }\n\n if (char === \"}\") {\n if (braceDepth > 0) {\n braceDepth -= 1;\n }\n current += char;\n continue;\n }\n\n if (char === \",\" && bracketDepth === 0 && braceDepth === 0) {\n pairs.push(current);\n current = \"\";\n continue;\n }\n\n current += char;\n }\n\n if (current.trim()) {\n pairs.push(current);\n }\n\n const output: Record<string, TomlValue> = {};\n for (const pair of pairs) {\n const separatorIndex = pair.indexOf(\"=\");\n if (separatorIndex === -1) {\n return null;\n }\n const key = normalizeKeyPath(pair.slice(0, separatorIndex));\n if (!key) {\n return null;\n }\n const valueRaw = pair.slice(separatorIndex + 1).trim();\n if (!valueRaw) {\n return null;\n }\n if (valueRaw.startsWith(\"{\")) {\n return null;\n }\n const normalized = normalizeValue(valueRaw);\n if (normalized === null) {\n return null;\n }\n if (typeof normalized === \"object\" && !Array.isArray(normalized)) {\n return null;\n }\n output[key] = normalized;\n }\n\n return output;\n}\n\nexport function normalizeValue(value: string): TomlValue | null {\n if (!value) {\n return null;\n }\n\n const trimmed = value.trim();\n if (trimmed.startsWith(\"{\") && trimmed.endsWith(\"}\")) {\n return parseInlineTable(trimmed);\n }\n\n const array = parseArray(trimmed);\n if (array) {\n return array;\n }\n\n if (trimmed.startsWith(\"[\") && trimmed.endsWith(\"]\")) {\n return null;\n }\n\n return parsePrimitive(trimmed);\n}\n\nexport function toPlainText(value: unknown): string {\n if (value == null) {\n return \"\";\n }\n if (Array.isArray(value)) {\n return value.map((item) => String(item)).join(\", \");\n }\n return String(value);\n}\n","import { ensureArrayContainer, flattenArrayTables } from \"./arrays\";\nimport { normalizeKeyPath } from \"./keys\";\nimport { stripInlineComment } from \"./strings\";\nimport { normalizeValue, toPlainText } from \"./values\";\n\nexport function parseTomlFrontmatter(frontmatter: string): Record<string, unknown> | null {\n const result: Record<string, unknown> = {};\n const lines = frontmatter.split(\"\\n\");\n let tablePrefix = \"\";\n let tableTarget: Record<string, unknown> | null = null;\n let tablePrefixInList = false;\n\n for (let index = 0; index < lines.length; index += 1) {\n const rawLine = lines[index] ?? \"\";\n const trimmedLine = rawLine.trim();\n if (!trimmedLine || trimmedLine.startsWith(\"#\")) {\n continue;\n }\n\n if (trimmedLine.startsWith(\"[[\")) {\n const match = trimmedLine.match(/^\\[\\[([^\\]]+)]]$/);\n if (!match) {\n return null;\n }\n const normalizedTable = normalizeKeyPath(match[1] ?? \"\");\n if (!normalizedTable) {\n return null;\n }\n const list = ensureArrayContainer(result, normalizedTable);\n const newEntry: Record<string, unknown> = {};\n list.push(newEntry);\n tableTarget = newEntry;\n tablePrefix = normalizedTable;\n tablePrefixInList = true;\n continue;\n }\n\n const tableMatch = trimmedLine.match(/^\\[([^\\]]+)]$/);\n if (tableMatch) {\n const normalizedTable = normalizeKeyPath(tableMatch[1] ?? \"\");\n if (!normalizedTable) {\n return null;\n }\n tablePrefix = normalizedTable;\n tablePrefixInList = false;\n tableTarget = null;\n continue;\n }\n\n const lineForParsing = /(\"\"\"|''')/.test(rawLine) ? rawLine : stripInlineComment(rawLine);\n const separatorIndex = lineForParsing.indexOf(\"=\");\n if (separatorIndex === -1) {\n return null;\n }\n\n const keyRaw = lineForParsing.slice(0, separatorIndex);\n const key = normalizeKeyPath(keyRaw);\n let valueRaw = lineForParsing.slice(separatorIndex + 1).trim();\n if (!key) {\n return null;\n }\n\n const tripleDelimiter = valueRaw.startsWith('\"\"\"') ? '\"\"\"' : valueRaw.startsWith(\"'''\") ? \"'''\" : null;\n if (tripleDelimiter) {\n const closingIndex = valueRaw.indexOf(tripleDelimiter, tripleDelimiter.length);\n if (closingIndex !== -1) {\n const after = valueRaw.slice(closingIndex + tripleDelimiter.length);\n const strippedAfter = stripInlineComment(after);\n valueRaw = `${valueRaw.slice(0, closingIndex + tripleDelimiter.length)}${strippedAfter}`;\n } else {\n const delimiter = tripleDelimiter;\n let combined = valueRaw;\n let closed = false;\n while (index + 1 < lines.length) {\n index += 1;\n const nextLine = lines[index] ?? \"\";\n combined += `\\n${nextLine}`;\n if (new RegExp(`${delimiter}\\\\s*$`).test(nextLine)) {\n closed = true;\n break;\n }\n }\n if (!closed) {\n return null;\n }\n valueRaw = combined;\n }\n }\n\n const normalized = normalizeValue(valueRaw);\n if (normalized === null) {\n return null;\n }\n\n const fullKey = tablePrefix ? `${tablePrefix}.${key}` : key;\n if (typeof normalized === \"object\" && !Array.isArray(normalized)) {\n for (const [inlineKey, inlineValue] of Object.entries(normalized)) {\n const entryKey = tablePrefixInList ? `${key}.${inlineKey}` : `${fullKey}.${inlineKey}`;\n if (tablePrefixInList && tableTarget) {\n tableTarget[entryKey] = toPlainText(inlineValue);\n } else {\n result[entryKey] = toPlainText(inlineValue);\n }\n }\n continue;\n }\n\n if (tablePrefixInList && tableTarget) {\n tableTarget[key] = toPlainText(normalized);\n continue;\n }\n\n result[fullKey] = toPlainText(normalized);\n }\n\n flattenArrayTables(result);\n\n return result;\n}\n","import { parseDocument } from \"yaml\";\nimport { parseTomlFrontmatter } from \"./toml-simple\";\nimport type { FrontmatterType, ParsedMarkdown } from \"./types\";\n\nconst FENCE_TO_TYPE: Record<string, FrontmatterType> = {\n \"---\": \"yaml\",\n \"+++\": \"toml\",\n \";;;\": \"json\",\n};\n\nfunction normalizeNewlines(input: string): string {\n return input.replace(/\\r\\n/g, \"\\n\");\n}\n\nfunction stripBom(line: string): string {\n return line.startsWith(\"\\uFEFF\") ? line.slice(1) : line;\n}\n\nfunction getFenceType(line: string): FrontmatterType | null {\n const match = line.match(/^[\\t ]*(---|\\+\\+\\+|;;;)[\\t ]*$/);\n if (!match) {\n return null;\n }\n return FENCE_TO_TYPE[match[1] ?? \"\"] ?? null;\n}\n\nfunction parseFrontmatter(frontmatter: string, type: FrontmatterType | null): Record<string, unknown> | null {\n if (!type) {\n return null;\n }\n\n if (type === \"json\") {\n try {\n return JSON.parse(frontmatter) as Record<string, unknown>;\n } catch {\n return null;\n }\n }\n\n if (type === \"yaml\") {\n const doc = parseDocument(frontmatter, { prettyErrors: false });\n if (doc.errors.length > 0) {\n return null;\n }\n const data = doc.toJSON();\n if (!data || typeof data !== \"object\" || Array.isArray(data)) {\n return null;\n }\n return data as Record<string, unknown>;\n }\n\n if (type === \"toml\") {\n return parseTomlFrontmatter(frontmatter);\n }\n\n return null;\n}\n\nfunction extractJsonBlock(text: string, startIndex: number): { jsonText: string; endIndex: number } | null {\n let depth = 0;\n let inString = false;\n let escaped = false;\n\n for (let i = startIndex; i < text.length; i += 1) {\n const char = text[i] ?? \"\";\n\n if (inString) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\") {\n escaped = true;\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = false;\n }\n\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = true;\n continue;\n }\n\n if (char === \"{\") {\n depth += 1;\n continue;\n }\n\n if (char === \"}\") {\n depth -= 1;\n if (depth === 0) {\n const jsonText = text.slice(startIndex, i + 1);\n return { jsonText, endIndex: i };\n }\n }\n }\n\n return null;\n}\n\nexport function parseMarkdown(input: string): ParsedMarkdown {\n const normalized = normalizeNewlines(input);\n const lines = normalized.split(\"\\n\");\n if (lines.length === 0) {\n return { frontmatter: null, content: normalized, data: null, frontmatterType: null };\n }\n\n lines[0] = stripBom(lines[0] ?? \"\");\n const normalizedWithoutBom = lines.join(\"\\n\");\n\n const openingType = getFenceType(lines[0] ?? \"\");\n if (!openingType) {\n const leadingWhitespace = normalizedWithoutBom.match(/^[\\t \\n]*/)?.[0] ?? \"\";\n const jsonStart = leadingWhitespace.length;\n if (normalizedWithoutBom[jsonStart] !== \"{\") {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const jsonBlock = extractJsonBlock(normalizedWithoutBom, jsonStart);\n if (!jsonBlock) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const frontmatter = jsonBlock.jsonText;\n let content = normalizedWithoutBom.slice(jsonBlock.endIndex + 1);\n if (content.startsWith(\"\\n\")) {\n content = content.slice(1);\n }\n const data = parseFrontmatter(frontmatter, \"json\");\n if (!data) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n return {\n frontmatter,\n content,\n data,\n frontmatterType: \"json\",\n };\n }\n\n let closingIndex = -1;\n for (let i = 1; i < lines.length; i += 1) {\n if (getFenceType(lines[i] ?? \"\") === openingType) {\n closingIndex = i;\n break;\n }\n }\n\n if (closingIndex === -1) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const frontmatter = lines.slice(1, closingIndex).join(\"\\n\");\n const content = lines.slice(closingIndex + 1).join(\"\\n\");\n const data = parseFrontmatter(frontmatter, openingType);\n\n return {\n frontmatter,\n content,\n data,\n frontmatterType: openingType,\n };\n}\n","import type { WordCounterMode, WordCounterOptions, WordCounterResult } from \"../wc/types\";\nimport wordCounter from \"../wc\";\nimport { parseMarkdown } from \"./parse-markdown\";\nimport type { SectionMode, SectionedResult } from \"./types\";\n\nfunction normalizeText(value: unknown): string {\n if (value == null) {\n return \"\";\n }\n if (typeof value === \"string\") {\n return value;\n }\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n return String(value);\n }\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n}\n\nfunction buildPerKeyItems(\n data: Record<string, unknown> | null,\n mode: WordCounterMode,\n options: WordCounterOptions,\n): Array<{ name: string; source: \"frontmatter\"; result: WordCounterResult }> {\n if (!data || typeof data !== \"object\" || Array.isArray(data)) {\n return [];\n }\n\n return Object.entries(data).map(([key, value]) => {\n const valueText = normalizeText(value);\n const text = valueText ? `${key}: ${valueText}` : key;\n return {\n name: key,\n source: \"frontmatter\",\n result: wordCounter(text, options),\n };\n });\n}\n\nfunction buildSingleItem(\n name: string,\n text: string,\n mode: WordCounterMode,\n options: WordCounterOptions,\n source: \"frontmatter\" | \"content\",\n) {\n return [{ name, source, result: wordCounter(text, options) }];\n}\n\nfunction sumTotals(items: Array<{ result: WordCounterResult }>): number {\n return items.reduce((sum, item) => sum + item.result.total, 0);\n}\n\nexport function countSections(\n input: string,\n section: SectionMode,\n options: WordCounterOptions = {},\n): SectionedResult {\n const mode: WordCounterMode = options.mode ?? \"chunk\";\n if (section === \"all\") {\n const result = wordCounter(input, options);\n return {\n section,\n total: result.total,\n frontmatterType: null,\n items: [{ name: \"all\", source: \"content\", result }],\n };\n }\n\n const parsed = parseMarkdown(input);\n const frontmatterText = parsed.frontmatter ?? \"\";\n const contentText = parsed.content ?? \"\";\n\n let items: Array<{ name: string; source: \"frontmatter\" | \"content\"; result: WordCounterResult }> = [];\n\n if (section === \"frontmatter\") {\n items = buildSingleItem(\"frontmatter\", frontmatterText, mode, options, \"frontmatter\");\n } else if (section === \"content\") {\n items = buildSingleItem(\"content\", contentText, mode, options, \"content\");\n } else if (section === \"split\") {\n items = [\n ...buildSingleItem(\"frontmatter\", frontmatterText, mode, options, \"frontmatter\"),\n ...buildSingleItem(\"content\", contentText, mode, options, \"content\"),\n ];\n } else if (section === \"per-key\") {\n items = buildPerKeyItems(parsed.data, mode, options);\n } else if (section === \"split-per-key\") {\n items = [\n ...buildPerKeyItems(parsed.data, mode, options),\n ...buildSingleItem(\"content\", contentText, mode, options, \"content\"),\n ];\n }\n\n return {\n section,\n total: sumTotals(items),\n frontmatterType: parsed.frontmatterType,\n items,\n };\n}\n","export function showSingularOrPluralWord(count: number, word: string): string {\n return `${count} ${word}${count === 1 ? \"\" : \"s\"}`;\n}\n","import wordCounter, {\n countCharsForLocale,\n countWordsForLocale,\n segmentTextByLocale,\n} from \"./wc\";\nimport { parseMarkdown, countSections } from \"./markdown\";\nimport { showSingularOrPluralWord } from \"./utils\";\n\nconst cjsExports = Object.assign(wordCounter, {\n default: wordCounter,\n wordCounter,\n countCharsForLocale,\n countWordsForLocale,\n segmentTextByLocale,\n parseMarkdown,\n countSections,\n showSingularOrPluralWord,\n});\n\nexport = cjsExports;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.cjs","names":["wordCounter","wordCounter"],"sources":["../../src/wc/segmenter.ts","../../src/utils/append-all.ts","../../src/wc/non-words.ts","../../src/wc/analyze.ts","../../src/wc/mode.ts","../../src/wc/latin-hints.ts","../../src/wc/locale-detect.ts","../../src/wc/segment.ts","../../src/wc/wc.ts","../../src/wc/index.ts","../../src/markdown/toml/arrays.ts","../../src/markdown/toml/keys.ts","../../src/markdown/toml/strings.ts","../../src/markdown/toml/values.ts","../../src/markdown/toml/parse-frontmatter.ts","../../src/markdown/parse-markdown.ts","../../src/markdown/section-count.ts","../../src/utils/show-singular-or-plural-word.ts","../../src/index.cjs.ts"],"sourcesContent":["const segmenterCache = new Map<string, Intl.Segmenter>();\nconst graphemeSegmenterCache = new Map<string, Intl.Segmenter>();\n\nexport function getSegmenter(locale: string): Intl.Segmenter {\n const cached = segmenterCache.get(locale);\n if (cached) {\n return cached;\n }\n const segmenter = new Intl.Segmenter(locale, { granularity: \"word\" });\n segmenterCache.set(locale, segmenter);\n return segmenter;\n}\n\nexport function getGraphemeSegmenter(locale: string): Intl.Segmenter {\n const cached = graphemeSegmenterCache.get(locale);\n if (cached) {\n return cached;\n }\n const segmenter = new Intl.Segmenter(locale, { granularity: \"grapheme\" });\n graphemeSegmenterCache.set(locale, segmenter);\n return segmenter;\n}\n\nfunction supportsSegmenter(): boolean {\n return typeof Intl !== \"undefined\" && typeof Intl.Segmenter === \"function\";\n}\n\nexport function countWordsForLocale(text: string, locale: string): number {\n const segmenter = getSegmenter(locale);\n let count = 0;\n for (const segment of segmenter.segment(text)) {\n if (segment.isWordLike) {\n count++;\n }\n }\n return count;\n}\n\nexport function countCharsForLocale(text: string, locale: string): number {\n if (!supportsSegmenter()) {\n return Array.from(text).length;\n }\n const segmenter = getGraphemeSegmenter(locale);\n let count = 0;\n for (const _segment of segmenter.segment(text)) {\n count++;\n }\n return count;\n}\n","export function appendAll<T>(target: T[], source: readonly T[]): void {\n for (const item of source) {\n target.push(item);\n }\n}\n","import type { NonWordCollection, WhitespaceCounts } from \"./types\";\nimport { appendAll } from \"../utils/append-all\";\n\nconst emojiRegex = /(?:\\p{Extended_Pictographic}|\\p{Emoji_Presentation})/u;\nconst emojiPresentationRegex = /\\p{Emoji_Presentation}/u;\nconst keycapEmojiRegex = /[0-9#*]\\uFE0F?\\u20E3/u;\nconst symbolRegex = /\\p{S}/u;\nconst punctuationRegex = /\\p{P}/u;\nconst whitespaceRegex = /\\s/u;\nconst newlineChars = new Set([\"\\n\", \"\\r\", \"\\u2028\", \"\\u2029\"]);\n\nexport function createNonWordCollection(): NonWordCollection {\n return {\n emoji: [],\n symbols: [],\n punctuation: [],\n counts: {\n emoji: 0,\n symbols: 0,\n punctuation: 0,\n },\n };\n}\n\nexport function addNonWord(\n collection: NonWordCollection,\n category: \"emoji\" | \"symbol\" | \"punctuation\",\n segment: string,\n): void {\n if (category === \"emoji\") {\n collection.emoji.push(segment);\n collection.counts.emoji += 1;\n return;\n }\n if (category === \"symbol\") {\n collection.symbols.push(segment);\n collection.counts.symbols += 1;\n return;\n }\n collection.punctuation.push(segment);\n collection.counts.punctuation += 1;\n}\n\nexport function addWhitespace(\n collection: NonWordCollection,\n segment: string,\n): number {\n let whitespace = collection.whitespace;\n let count = 0;\n for (const char of segment) {\n if (char === \" \") {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.spaces += 1;\n count += 1;\n continue;\n }\n if (char === \"\\t\") {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.tabs += 1;\n count += 1;\n continue;\n }\n if (newlineChars.has(char)) {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.newlines += 1;\n count += 1;\n continue;\n }\n if (whitespaceRegex.test(char)) {\n whitespace = whitespace ?? createWhitespaceCounts();\n whitespace.other += 1;\n count += 1;\n }\n }\n\n if (count > 0) {\n collection.whitespace = whitespace ?? createWhitespaceCounts();\n collection.counts.whitespace = (collection.counts.whitespace ?? 0) + count;\n }\n\n return count;\n}\n\nexport function classifyNonWordSegment(\n segment: string,\n): \"emoji\" | \"symbol\" | \"punctuation\" | null {\n const hasEmojiVariationSelector = segment.includes(\"\\uFE0F\");\n if (\n keycapEmojiRegex.test(segment) ||\n emojiPresentationRegex.test(segment) ||\n (hasEmojiVariationSelector && emojiRegex.test(segment))\n ) {\n return \"emoji\";\n }\n if (symbolRegex.test(segment)) {\n return \"symbol\";\n }\n if (punctuationRegex.test(segment)) {\n return \"punctuation\";\n }\n return null;\n}\n\nexport function mergeNonWordCollections(\n target: NonWordCollection,\n source: NonWordCollection,\n): NonWordCollection {\n if (source.counts.emoji > 0) {\n appendAll(target.emoji, source.emoji);\n target.counts.emoji += source.counts.emoji;\n }\n if (source.counts.symbols > 0) {\n appendAll(target.symbols, source.symbols);\n target.counts.symbols += source.counts.symbols;\n }\n if (source.counts.punctuation > 0) {\n appendAll(target.punctuation, source.punctuation);\n target.counts.punctuation += source.counts.punctuation;\n }\n if (source.counts.whitespace && source.counts.whitespace > 0 && source.whitespace) {\n const whitespace = target.whitespace ?? createWhitespaceCounts();\n whitespace.spaces += source.whitespace.spaces;\n whitespace.tabs += source.whitespace.tabs;\n whitespace.newlines += source.whitespace.newlines;\n whitespace.other += source.whitespace.other;\n target.whitespace = whitespace;\n target.counts.whitespace = (target.counts.whitespace ?? 0) + source.counts.whitespace;\n }\n return target;\n}\n\nfunction createWhitespaceCounts(): WhitespaceCounts {\n return { spaces: 0, tabs: 0, newlines: 0, other: 0 };\n}\n","import { countCharsForLocale, getSegmenter } from \"./segmenter\";\nimport {\n addNonWord,\n addWhitespace,\n classifyNonWordSegment,\n createNonWordCollection,\n mergeNonWordCollections,\n} from \"./non-words\";\nimport { appendAll } from \"../utils/append-all\";\nimport type {\n CharCollectorBreakdown,\n ChunkAnalysis,\n CollectorBreakdown,\n LocaleChunk,\n NonWordCollection,\n} from \"./types\";\n\ntype CharAnalysis = LocaleChunk & { chars: number; nonWords?: NonWordCollection };\ntype CharChunkAnalysis = CharAnalysis & { wordChars: number; nonWordChars: number };\n\nexport function analyzeChunk(\n chunk: LocaleChunk,\n collectNonWords?: boolean,\n includeWhitespace?: boolean,\n): ChunkAnalysis {\n const segmenter = getSegmenter(chunk.locale);\n const segments: string[] = [];\n const nonWords: NonWordCollection | null = collectNonWords\n ? createNonWordCollection()\n : null;\n for (const part of segmenter.segment(chunk.text)) {\n if (part.isWordLike) {\n segments.push(part.segment);\n } else if (collectNonWords && nonWords) {\n if (includeWhitespace) {\n addWhitespace(nonWords, part.segment);\n }\n const category = classifyNonWordSegment(part.segment);\n if (category) {\n addNonWord(nonWords, category, part.segment);\n }\n }\n }\n return {\n locale: chunk.locale,\n text: chunk.text,\n segments,\n words: segments.length,\n nonWords: nonWords ?? undefined,\n };\n}\n\nexport function analyzeCharChunk(\n chunk: LocaleChunk,\n collectNonWords?: boolean,\n includeWhitespace?: boolean,\n): CharChunkAnalysis {\n const segmenter = getSegmenter(chunk.locale);\n const nonWords: NonWordCollection | null = collectNonWords\n ? createNonWordCollection()\n : null;\n let chars = 0;\n let wordChars = 0;\n let nonWordChars = 0;\n\n for (const part of segmenter.segment(chunk.text)) {\n if (part.isWordLike) {\n const count = countCharsForLocale(part.segment, chunk.locale);\n chars += count;\n wordChars += count;\n continue;\n }\n\n if (collectNonWords && nonWords) {\n let whitespaceCount = 0;\n if (includeWhitespace) {\n whitespaceCount = addWhitespace(nonWords, part.segment);\n }\n const category = classifyNonWordSegment(part.segment);\n if (category) {\n addNonWord(nonWords, category, part.segment);\n }\n if (category || whitespaceCount > 0) {\n const count = countCharsForLocale(part.segment, chunk.locale);\n chars += count;\n nonWordChars += count;\n }\n }\n }\n\n return {\n locale: chunk.locale,\n text: chunk.text,\n chars,\n wordChars,\n nonWordChars,\n nonWords: nonWords ?? undefined,\n };\n}\n\nexport function aggregateCharsByLocale(\n chunks: CharChunkAnalysis[],\n): Array<CharCollectorBreakdown & { wordChars: number; nonWordChars: number }> {\n const order: string[] = [];\n const map = new Map<\n string,\n CharCollectorBreakdown & { wordChars: number; nonWordChars: number }\n >();\n\n for (const chunk of chunks) {\n const existing = map.get(chunk.locale);\n if (existing) {\n existing.chars += chunk.chars;\n existing.wordChars += chunk.wordChars;\n existing.nonWordChars += chunk.nonWordChars;\n if (chunk.nonWords) {\n if (!existing.nonWords) {\n existing.nonWords = createNonWordCollection();\n }\n mergeNonWordCollections(existing.nonWords, chunk.nonWords);\n }\n continue;\n }\n\n order.push(chunk.locale);\n map.set(chunk.locale, {\n locale: chunk.locale,\n chars: chunk.chars,\n wordChars: chunk.wordChars,\n nonWordChars: chunk.nonWordChars,\n nonWords: chunk.nonWords\n ? mergeNonWordCollections(createNonWordCollection(), chunk.nonWords)\n : undefined,\n });\n }\n\n return order.map((locale) => map.get(locale)!);\n}\n\nexport function aggregateByLocale(\n chunks: ChunkAnalysis[]\n): CollectorBreakdown[] {\n const order: string[] = [];\n const map = new Map<string, CollectorBreakdown>();\n\n for (const chunk of chunks) {\n const existing = map.get(chunk.locale);\n if (existing) {\n existing.words += chunk.words;\n appendAll(existing.segments, chunk.segments);\n continue;\n }\n\n order.push(chunk.locale);\n map.set(chunk.locale, {\n locale: chunk.locale,\n words: chunk.words,\n segments: [...chunk.segments],\n });\n }\n\n return order.map((locale) => map.get(locale)!);\n}\n","import type { WordCounterMode } from \"./types\";\n\nconst MODE_ALIASES: Record<string, WordCounterMode> = {\n chunk: \"chunk\",\n chunks: \"chunk\",\n segments: \"segments\",\n segment: \"segments\",\n seg: \"segments\",\n collector: \"collector\",\n collect: \"collector\",\n colle: \"collector\",\n char: \"char\",\n chars: \"char\",\n character: \"char\",\n characters: \"char\",\n \"char-collector\": \"char-collector\",\n};\n\nconst CHAR_MODE_ALIASES = new Set([\"char\", \"chars\", \"character\", \"characters\"]);\nconst COLLECTOR_MODE_ALIASES = new Set([\"collector\", \"collect\", \"colle\", \"col\"]);\n\nfunction collapseSeparators(value: string): string {\n return value.replace(/[-_\\s]+/g, \"\");\n}\n\nfunction isComposedCharCollectorFromTokens(value: string): boolean {\n const tokens = value\n .split(/[-_\\s]+/)\n .map((token) => token.trim())\n .filter((token) => token.length > 0);\n if (tokens.length < 2) {\n return false;\n }\n\n let hasCharAlias = false;\n let hasCollectorAlias = false;\n for (const token of tokens) {\n if (CHAR_MODE_ALIASES.has(token)) {\n hasCharAlias = true;\n continue;\n }\n if (COLLECTOR_MODE_ALIASES.has(token)) {\n hasCollectorAlias = true;\n continue;\n }\n return false;\n }\n\n return hasCharAlias && hasCollectorAlias;\n}\n\nfunction isComposedCharCollectorCompact(value: string): boolean {\n for (const charAlias of CHAR_MODE_ALIASES) {\n for (const collectorAlias of COLLECTOR_MODE_ALIASES) {\n if (value === `${charAlias}${collectorAlias}` || value === `${collectorAlias}${charAlias}`) {\n return true;\n }\n }\n }\n return false;\n}\n\nexport function normalizeMode(input?: string | null): WordCounterMode | null {\n if (!input) {\n return null;\n }\n const normalized = input.trim().toLowerCase();\n const direct = MODE_ALIASES[normalized];\n if (direct) {\n return direct;\n }\n\n if (isComposedCharCollectorFromTokens(normalized)) {\n return \"char-collector\";\n }\n\n const compact = collapseSeparators(normalized);\n if (isComposedCharCollectorCompact(compact)) {\n return \"char-collector\";\n }\n\n return MODE_ALIASES[compact] ?? null;\n}\n\nexport function resolveMode(\n input?: string | null,\n fallback: WordCounterMode = \"chunk\",\n): WordCounterMode {\n return normalizeMode(input) ?? fallback;\n}\n","import type { LatinHintRule } from \"./types\";\n\nconst DEFAULT_LATIN_HINT_RULES_SOURCE = [\n { tag: \"de\", pattern: \"[äöüÄÖÜß]\" },\n { tag: \"es\", pattern: \"[ñÑ¿¡]\" },\n { tag: \"pt\", pattern: \"[ãõÃÕ]\" },\n { tag: \"fr\", pattern: \"[œŒæÆ]\" },\n { tag: \"pl\", pattern: \"[ąćęłńśźżĄĆĘŁŃŚŹŻ]\" },\n { tag: \"tr\", pattern: \"[ıİğĞşŞ]\" },\n { tag: \"ro\", pattern: \"[ăĂâÂîÎșȘțȚ]\" },\n { tag: \"hu\", pattern: \"[őŐűŰ]\" },\n { tag: \"is\", pattern: \"[ðÐþÞ]\" },\n] satisfies LatinHintRule[];\n\nexport const DEFAULT_LATIN_HINT_RULES: ReadonlyArray<Readonly<LatinHintRule>> = Object.freeze(\n DEFAULT_LATIN_HINT_RULES_SOURCE.map((rule) => Object.freeze({ ...rule })),\n);\n","import type { LatinHintRule } from \"./types\";\nimport { DEFAULT_LATIN_HINT_RULES } from \"./latin-hints\";\n\nexport const DEFAULT_LOCALE = \"und-Latn\";\nexport const DEFAULT_HAN_TAG = \"und-Hani\";\n\nconst MAX_LATIN_HINT_PATTERN_LENGTH = 256;\n\nexport interface LocaleDetectOptions {\n latinLanguageHint?: string;\n latinTagHint?: string;\n latinLocaleHint?: string;\n latinHintRules?: LatinHintRule[];\n useDefaultLatinHints?: boolean;\n hanLanguageHint?: string;\n hanTagHint?: string;\n}\n\ntype ResolvedLatinHintRule = {\n tag: string;\n pattern: RegExp;\n priority: number;\n order: number;\n};\n\nexport type LocaleDetectContext = {\n latinHint?: string;\n hanHint?: string;\n latinHintRules: ResolvedLatinHintRule[];\n latinLocales: Set<string>;\n};\n\nconst regex = {\n hiragana: /\\p{Script=Hiragana}/u,\n katakana: /\\p{Script=Katakana}/u,\n hangul: /\\p{Script=Hangul}/u,\n han: /\\p{Script=Han}/u,\n latin: /\\p{Script=Latin}/u,\n arabic: /\\p{Script=Arabic}/u,\n cyrillic: /\\p{Script=Cyrillic}/u,\n devanagari: /\\p{Script=Devanagari}/u,\n thai: /\\p{Script=Thai}/u,\n};\n\nconst defaultLatinLocales = new Set<string>([\n DEFAULT_LOCALE,\n ...DEFAULT_LATIN_HINT_RULES.map((hint) => hint.tag),\n]);\n\nexport function isLatinLocale(locale: string, context?: LocaleDetectContext): boolean {\n if (context) {\n return context.latinLocales.has(locale);\n }\n return defaultLatinLocales.has(locale);\n}\n\nfunction resolveLatinHint(options: LocaleDetectOptions): string | undefined {\n const latinTagHint = options.latinTagHint?.trim();\n if (latinTagHint) {\n return latinTagHint;\n }\n\n const latinLanguageHint = options.latinLanguageHint?.trim();\n if (latinLanguageHint) {\n return latinLanguageHint;\n }\n\n const latinLocaleHint = options.latinLocaleHint?.trim();\n if (latinLocaleHint) {\n return latinLocaleHint;\n }\n\n return undefined;\n}\n\nfunction resolveHanHint(options: LocaleDetectOptions): string | undefined {\n const hanTagHint = options.hanTagHint?.trim();\n if (hanTagHint) {\n return hanTagHint;\n }\n\n const hanLanguageHint = options.hanLanguageHint?.trim();\n if (hanLanguageHint) {\n return hanLanguageHint;\n }\n\n return undefined;\n}\n\nfunction compileLatinHintPattern(\n pattern: string | RegExp,\n label: string,\n): RegExp {\n const source = typeof pattern === \"string\" ? pattern : pattern.source;\n const hasUnicodeMode =\n typeof pattern !== \"string\" &&\n (pattern.flags.includes(\"u\") || pattern.flags.includes(\"v\"));\n const flags =\n typeof pattern === \"string\"\n ? \"u\"\n : hasUnicodeMode\n ? pattern.flags\n : `${pattern.flags}u`;\n if (source.length === 0) {\n throw new Error(`${label}: pattern must not be empty.`);\n }\n if (source.length > MAX_LATIN_HINT_PATTERN_LENGTH) {\n throw new Error(\n `${label}: pattern must be at most ${MAX_LATIN_HINT_PATTERN_LENGTH} characters.`,\n );\n }\n try {\n return new RegExp(source, flags);\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n throw new Error(`${label}: invalid Unicode regex pattern (${message}).`);\n }\n}\n\nfunction normalizeLatinHintPriority(priority: unknown, label: string): number {\n if (priority === undefined) {\n return 0;\n }\n if (typeof priority !== \"number\" || !Number.isFinite(priority)) {\n throw new Error(`${label}: priority must be a finite number when provided.`);\n }\n return priority;\n}\n\nfunction compileLatinHintRule(\n rule: Readonly<LatinHintRule>,\n order: number,\n label: string,\n): ResolvedLatinHintRule {\n const tag = typeof rule.tag === \"string\" ? rule.tag.trim() : \"\";\n if (!tag) {\n throw new Error(`${label}: tag must be a non-empty string.`);\n }\n const pattern = compileLatinHintPattern(rule.pattern, label);\n const priority = normalizeLatinHintPriority(rule.priority, label);\n return {\n tag,\n pattern,\n priority,\n order,\n };\n}\n\nfunction resolveLatinHintRules(options: LocaleDetectOptions): ResolvedLatinHintRule[] {\n const useDefaultLatinHints = options.useDefaultLatinHints !== false;\n const customRules = options.latinHintRules ?? [];\n const combinedRules: Array<{ rule: Readonly<LatinHintRule>; label: string }> = [];\n\n for (let index = 0; index < customRules.length; index += 1) {\n const rule = customRules[index];\n if (!rule) {\n continue;\n }\n combinedRules.push({\n rule,\n label: `Invalid custom Latin hint rule at index ${index}`,\n });\n }\n\n if (useDefaultLatinHints) {\n for (let index = 0; index < DEFAULT_LATIN_HINT_RULES.length; index += 1) {\n const rule = DEFAULT_LATIN_HINT_RULES[index];\n if (!rule) {\n continue;\n }\n combinedRules.push({\n rule,\n label: `Invalid default Latin hint rule at index ${index}`,\n });\n }\n }\n\n const resolvedRules = combinedRules.map((entry, index) =>\n compileLatinHintRule(entry.rule, index, entry.label),\n );\n\n resolvedRules.sort((left, right) => {\n if (left.priority !== right.priority) {\n return right.priority - left.priority;\n }\n return left.order - right.order;\n });\n\n return resolvedRules;\n}\n\nexport function resolveLocaleDetectContext(\n options: LocaleDetectOptions = {},\n): LocaleDetectContext {\n const latinHint = resolveLatinHint(options);\n const latinHintRules = resolveLatinHintRules(options);\n const latinLocales = new Set<string>([DEFAULT_LOCALE]);\n for (const rule of latinHintRules) {\n latinLocales.add(rule.tag);\n }\n if (latinHint) {\n latinLocales.add(latinHint);\n }\n\n return {\n latinHint,\n hanHint: resolveHanHint(options),\n latinHintRules,\n latinLocales,\n };\n}\n\nfunction detectLatinLocale(char: string, context: LocaleDetectContext): string {\n for (const hint of context.latinHintRules) {\n hint.pattern.lastIndex = 0;\n if (hint.pattern.test(char)) {\n return hint.tag;\n }\n }\n return DEFAULT_LOCALE;\n}\n\nexport function detectLocaleForChar(\n char: string,\n previousLocale?: string | null,\n options: LocaleDetectOptions = {},\n context: LocaleDetectContext = resolveLocaleDetectContext(options),\n allowLatinLocaleCarry = true,\n allowJapaneseHanCarry = true,\n): string | null {\n if (regex.hiragana.test(char) || regex.katakana.test(char)) {\n return \"ja\";\n }\n if (regex.hangul.test(char)) {\n return \"ko\";\n }\n if (regex.arabic.test(char)) {\n return \"ar\";\n }\n if (regex.cyrillic.test(char)) {\n return \"ru\";\n }\n if (regex.devanagari.test(char)) {\n return \"hi\";\n }\n if (regex.thai.test(char)) {\n return \"th\";\n }\n\n if (regex.han.test(char)) {\n if (allowJapaneseHanCarry && previousLocale && previousLocale.startsWith(\"ja\")) {\n return previousLocale;\n }\n return context.hanHint ?? DEFAULT_HAN_TAG;\n }\n\n if (regex.latin.test(char)) {\n const hintedLocale = detectLatinLocale(char, context);\n if (hintedLocale !== DEFAULT_LOCALE) {\n return hintedLocale;\n }\n if (\n allowLatinLocaleCarry &&\n previousLocale &&\n isLatinLocale(previousLocale, context) &&\n previousLocale !== DEFAULT_LOCALE\n ) {\n return previousLocale;\n }\n if (context.latinHint) {\n return context.latinHint;\n }\n return DEFAULT_LOCALE;\n }\n\n return null;\n}\n","import {\n DEFAULT_LOCALE,\n detectLocaleForChar,\n isLatinLocale,\n resolveLocaleDetectContext,\n type LocaleDetectOptions,\n} from \"./locale-detect\";\nimport type { LocaleChunk } from \"./types\";\n\nconst HARD_BOUNDARY_REGEX = /[\\r\\n,.!?;:,、。!?;:.。、]/u;\nconst LATIN_PROMOTION_BREAK_REGEX = /[\\s,.!?;:,、。!?;:.。、]/u;\n\nexport function segmentTextByLocale(\n text: string,\n options: LocaleDetectOptions = {}\n): LocaleChunk[] {\n const context = resolveLocaleDetectContext(options);\n const chunks: LocaleChunk[] = [];\n // Keep currentLocale as a non-null string to simplify type-narrowing.\n let currentLocale: string = DEFAULT_LOCALE;\n let buffer = \"\";\n let bufferHasScript = false;\n let sawCarryBoundary = false;\n\n const updateCarryBoundaryState = (detected: string | null, char: string): void => {\n if (detected !== null) {\n sawCarryBoundary = false;\n return;\n }\n if (HARD_BOUNDARY_REGEX.test(char)) {\n sawCarryBoundary = true;\n }\n };\n\n for (const char of text) {\n const detected = detectLocaleForChar(\n char,\n currentLocale,\n options,\n context,\n !sawCarryBoundary,\n !sawCarryBoundary,\n );\n const targetLocale = detected ?? currentLocale;\n\n // If buffer is empty, this is the first character for a new chunk.\n if (buffer === \"\") {\n currentLocale = targetLocale;\n buffer = char;\n bufferHasScript = detected !== null;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n if (detected !== null && !bufferHasScript) {\n currentLocale = targetLocale;\n buffer += char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n if (targetLocale !== currentLocale && detected !== null) {\n if (currentLocale === DEFAULT_LOCALE && isLatinLocale(targetLocale, context)) {\n const promotionBreakIndex = findLastLatinPromotionBreakIndex(buffer);\n if (promotionBreakIndex === -1) {\n currentLocale = targetLocale;\n buffer += char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n const prefix = buffer.slice(0, promotionBreakIndex + 1);\n const suffix = buffer.slice(promotionBreakIndex + 1);\n if (prefix.length > 0) {\n chunks.push({ locale: currentLocale, text: prefix });\n }\n currentLocale = targetLocale;\n buffer = `${suffix}${char}`;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n // currentLocale is guaranteed to be a string here.\n chunks.push({ locale: currentLocale, text: buffer });\n currentLocale = targetLocale;\n buffer = char;\n bufferHasScript = true;\n updateCarryBoundaryState(detected, char);\n continue;\n }\n\n buffer += char;\n if (detected !== null) {\n bufferHasScript = true;\n }\n updateCarryBoundaryState(detected, char);\n }\n\n if (buffer.length > 0) {\n chunks.push({ locale: currentLocale, text: buffer });\n }\n\n return mergeAdjacentChunks(chunks);\n}\n\nfunction findLastLatinPromotionBreakIndex(buffer: string): number {\n for (let index = buffer.length - 1; index >= 0; index -= 1) {\n const char = buffer[index];\n if (!char) {\n continue;\n }\n if (LATIN_PROMOTION_BREAK_REGEX.test(char)) {\n return index;\n }\n }\n return -1;\n}\n\nfunction mergeAdjacentChunks(chunks: LocaleChunk[]): LocaleChunk[] {\n if (chunks.length === 0) {\n return chunks;\n }\n\n const merged: LocaleChunk[] = [];\n // We already returned for empty arrays above, so the first element is present.\n let last = chunks[0]!;\n\n for (let i = 1; i < chunks.length; i++) {\n const chunk = chunks[i]!;\n if (chunk.locale === last.locale) {\n last = { locale: last.locale, text: last.text + chunk.text };\n } else {\n merged.push(last);\n last = chunk;\n }\n }\n\n merged.push(last);\n return merged;\n}\n","import {\n analyzeCharChunk,\n analyzeChunk,\n aggregateByLocale,\n aggregateCharsByLocale,\n} from \"./analyze\";\nimport { resolveMode } from \"./mode\";\nimport { segmentTextByLocale } from \"./segment\";\nimport { countCharsForLocale, countWordsForLocale } from \"./segmenter\";\nimport { createNonWordCollection, mergeNonWordCollections } from \"./non-words\";\nimport type {\n CharBreakdown,\n CharCollectorBreakdown,\n ChunkBreakdown,\n ChunkWithSegments,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n} from \"./types\";\n\nexport type {\n LatinHintRule,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n WordCounterBreakdown,\n} from \"./types\";\n\nexport { countCharsForLocale, countWordsForLocale, segmentTextByLocale };\nexport { DEFAULT_LATIN_HINT_RULES } from \"./latin-hints\";\n\nexport function wordCounter(\n text: string,\n options: WordCounterOptions = {}\n): WordCounterResult {\n const mode: WordCounterMode = resolveMode(options.mode, \"chunk\");\n const collectNonWords = Boolean(options.nonWords);\n const includeWhitespace = Boolean(options.includeWhitespace);\n const chunks = segmentTextByLocale(text, {\n latinLanguageHint: options.latinLanguageHint,\n latinTagHint: options.latinTagHint,\n latinLocaleHint: options.latinLocaleHint,\n latinHintRules: options.latinHintRules,\n useDefaultLatinHints: options.useDefaultLatinHints,\n hanLanguageHint: options.hanLanguageHint,\n hanTagHint: options.hanTagHint,\n });\n\n if (mode === \"char\" || mode === \"char-collector\") {\n const analyzed = chunks.map((chunk) =>\n analyzeCharChunk(chunk, collectNonWords, includeWhitespace),\n );\n const total = analyzed.reduce((sum, chunk) => sum + chunk.chars, 0);\n const counts = collectNonWords\n ? {\n words: analyzed.reduce((sum, chunk) => sum + chunk.wordChars, 0),\n nonWords: analyzed.reduce((sum, chunk) => sum + chunk.nonWordChars, 0),\n total,\n }\n : undefined;\n\n if (mode === \"char\") {\n const items: CharBreakdown[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n chars: chunk.chars,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n const aggregated = aggregateCharsByLocale(analyzed);\n const items: CharCollectorBreakdown[] = aggregated.map((chunk) => ({\n locale: chunk.locale,\n chars: chunk.chars,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n const analyzed = chunks.map((chunk) =>\n analyzeChunk(chunk, collectNonWords, includeWhitespace),\n );\n const wordsTotal = analyzed.reduce((sum, chunk) => sum + chunk.words, 0);\n const nonWordsTotal = collectNonWords\n ? analyzed.reduce((sum, chunk) => {\n if (!chunk.nonWords) {\n return sum;\n }\n return sum + getNonWordTotal(chunk.nonWords);\n }, 0)\n : 0;\n const total = analyzed.reduce((sum, chunk) => {\n let chunkTotal = chunk.words;\n if (collectNonWords && chunk.nonWords) {\n chunkTotal += getNonWordTotal(chunk.nonWords);\n }\n return sum + chunkTotal;\n }, 0);\n\n const counts = collectNonWords ? { words: wordsTotal, nonWords: nonWordsTotal, total } : undefined;\n\n if (mode === \"segments\") {\n const items: ChunkWithSegments[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n words: chunk.words,\n segments: chunk.segments,\n nonWords: chunk.nonWords,\n }));\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n }\n\n if (mode === \"collector\") {\n const items = aggregateByLocale(analyzed);\n const nonWords = collectNonWordsAggregate(analyzed, collectNonWords);\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n nonWords,\n },\n };\n }\n\n const items: ChunkBreakdown[] = analyzed.map((chunk) => ({\n locale: chunk.locale,\n text: chunk.text,\n words: chunk.words,\n nonWords: chunk.nonWords,\n }));\n\n return {\n total,\n counts,\n breakdown: {\n mode,\n items,\n },\n };\n}\n\nfunction getNonWordTotal(nonWords: NonWordCollection): number {\n return (\n nonWords.counts.emoji +\n nonWords.counts.symbols +\n nonWords.counts.punctuation +\n (nonWords.counts.whitespace ?? 0)\n );\n}\n\n\nfunction collectNonWordsAggregate(\n analyzed: Array<{ nonWords?: NonWordCollection }>,\n enabled: boolean,\n): NonWordCollection | undefined {\n if (!enabled) {\n return undefined;\n }\n const collection = createNonWordCollection();\n for (const chunk of analyzed) {\n if (!chunk.nonWords) {\n continue;\n }\n mergeNonWordCollections(collection, chunk.nonWords);\n }\n return collection;\n}\n","import { wordCounter } from \"./wc\";\n\nexport default wordCounter;\nexport { countCharsForLocale, countWordsForLocale, segmentTextByLocale } from \"./wc\";\nexport { DEFAULT_LATIN_HINT_RULES } from \"./wc\";\nexport type {\n LatinHintRule,\n NonWordCollection,\n WordCounterMode,\n WordCounterOptions,\n WordCounterResult,\n WordCounterBreakdown,\n} from \"./wc\";\n","export function ensureArrayContainer(\n result: Record<string, unknown>,\n key: string,\n): Record<string, unknown>[] {\n const existing = result[key];\n if (Array.isArray(existing)) {\n return existing as Record<string, unknown>[];\n }\n const list: Record<string, unknown>[] = [];\n result[key] = list;\n return list;\n}\n\nexport function flattenArrayTables(result: Record<string, unknown>): void {\n for (const [key, value] of Object.entries(result)) {\n if (!Array.isArray(value)) {\n continue;\n }\n const flattened = value\n .map((entry) =>\n Object.entries(entry)\n .map(([entryKey, entryValue]) => `${entryKey}=${entryValue}`)\n .join(\", \"),\n )\n .join(\" | \");\n result[key] = flattened;\n }\n}\n","function stripKeyQuotes(key: string): string {\n const trimmed = key.trim();\n if (\n (trimmed.startsWith(\"\\\"\") && trimmed.endsWith(\"\\\"\")) ||\n (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n ) {\n return trimmed.slice(1, -1);\n }\n return trimmed;\n}\n\nexport function normalizeKeyPath(key: string): string | null {\n const trimmed = key.trim();\n if (!trimmed) {\n return null;\n }\n\n if (\n (trimmed.startsWith(\"\\\"\") && trimmed.endsWith(\"\\\"\")) ||\n (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n ) {\n const unquoted = stripKeyQuotes(trimmed);\n return unquoted ? unquoted : null;\n }\n\n const segments = trimmed.split(\".\").map((segment) => segment.trim());\n if (segments.some((segment) => !segment)) {\n return null;\n }\n return segments.join(\".\");\n}\n","export function stripInlineComment(line: string): string {\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n\n for (let i = 0; i < line.length; i += 1) {\n const char = line[i] ?? \"\";\n\n if (inString) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n continue;\n }\n\n if (inString === \"single\" && char === \"'\") {\n inString = null;\n continue;\n }\n\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n continue;\n }\n\n if (char === \"#\") {\n return line.slice(0, i).trimEnd();\n }\n }\n\n return line;\n}\n\nfunction unescapeBasic(input: string): string {\n return input\n .replace(/\\\\\\\\/g, \"\\\\\")\n .replace(/\\\\\"/g, \"\\\"\")\n .replace(/\\\\n/g, \"\\n\")\n .replace(/\\\\t/g, \"\\t\")\n .replace(/\\\\r/g, \"\\r\");\n}\n\nexport function parseStringLiteral(value: string): string | null {\n if (value.startsWith('\"\"\"') && value.endsWith('\"\"\"')) {\n const inner = value.slice(3, -3);\n return unescapeBasic(inner);\n }\n\n if (value.startsWith(\"'''\") && value.endsWith(\"'''\")) {\n return value.slice(3, -3);\n }\n\n if (value.startsWith(\"\\\"\") && value.endsWith(\"\\\"\")) {\n return unescapeBasic(value.slice(1, -1));\n }\n\n if (value.startsWith(\"'\") && value.endsWith(\"'\")) {\n return value.slice(1, -1);\n }\n\n return null;\n}\n","import { normalizeKeyPath } from \"./keys\";\nimport { parseStringLiteral } from \"./strings\";\nimport type { TomlValue } from \"./types\";\n\nfunction parsePrimitive(raw: string): string | number | boolean | null {\n const value = raw.trim();\n if (!value) {\n return null;\n }\n\n const stringLiteral = parseStringLiteral(value);\n if (stringLiteral !== null) {\n return stringLiteral;\n }\n\n if (value === \"true\") {\n return true;\n }\n\n if (value === \"false\") {\n return false;\n }\n\n if (/^[+-]?\\d+(?:\\.\\d+)?(?:[eE][+-]?\\d+)?$/.test(value)) {\n return Number(value);\n }\n\n if (/^\\d{4}-\\d{2}-\\d{2}/.test(value)) {\n return value;\n }\n\n return value;\n}\n\nfunction parseArray(raw: string): Array<string | number | boolean> | null {\n const value = raw.trim();\n if (!value.startsWith(\"[\") || !value.endsWith(\"]\")) {\n return null;\n }\n\n const inner = value.slice(1, -1).trim();\n if (!inner) {\n return [];\n }\n\n const items: Array<string | number | boolean> = [];\n let current = \"\";\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n\n for (let i = 0; i < inner.length; i += 1) {\n const char = inner[i] ?? \"\";\n\n if (inString) {\n current += char;\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n } else if (inString === \"single\" && char === \"'\") {\n inString = null;\n }\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n current += char;\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n current += char;\n continue;\n }\n\n if (char === \",\") {\n const item = parsePrimitive(current);\n if (item === null) {\n return null;\n }\n items.push(item);\n current = \"\";\n continue;\n }\n\n current += char;\n }\n\n const finalItem = parsePrimitive(current);\n if (finalItem === null) {\n return null;\n }\n items.push(finalItem);\n\n return items;\n}\n\nfunction parseInlineTable(raw: string): Record<string, TomlValue> | null {\n const trimmed = raw.trim();\n if (!trimmed.startsWith(\"{\") || !trimmed.endsWith(\"}\")) {\n return null;\n }\n\n const inner = trimmed.slice(1, -1).trim();\n if (!inner) {\n return {};\n }\n\n const pairs: string[] = [];\n let current = \"\";\n let inString: \"single\" | \"double\" | null = null;\n let escaped = false;\n let bracketDepth = 0;\n let braceDepth = 0;\n\n for (let i = 0; i < inner.length; i += 1) {\n const char = inner[i] ?? \"\";\n\n if (inString) {\n current += char;\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\" && inString === \"double\") {\n escaped = true;\n continue;\n }\n\n if (inString === \"double\" && char === \"\\\"\") {\n inString = null;\n } else if (inString === \"single\" && char === \"'\") {\n inString = null;\n }\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = \"double\";\n current += char;\n continue;\n }\n\n if (char === \"'\") {\n inString = \"single\";\n current += char;\n continue;\n }\n\n if (char === \"[\") {\n bracketDepth += 1;\n current += char;\n continue;\n }\n\n if (char === \"]\") {\n if (bracketDepth > 0) {\n bracketDepth -= 1;\n }\n current += char;\n continue;\n }\n\n if (char === \"{\") {\n braceDepth += 1;\n current += char;\n continue;\n }\n\n if (char === \"}\") {\n if (braceDepth > 0) {\n braceDepth -= 1;\n }\n current += char;\n continue;\n }\n\n if (char === \",\" && bracketDepth === 0 && braceDepth === 0) {\n pairs.push(current);\n current = \"\";\n continue;\n }\n\n current += char;\n }\n\n if (current.trim()) {\n pairs.push(current);\n }\n\n const output: Record<string, TomlValue> = {};\n for (const pair of pairs) {\n const separatorIndex = pair.indexOf(\"=\");\n if (separatorIndex === -1) {\n return null;\n }\n const key = normalizeKeyPath(pair.slice(0, separatorIndex));\n if (!key) {\n return null;\n }\n const valueRaw = pair.slice(separatorIndex + 1).trim();\n if (!valueRaw) {\n return null;\n }\n if (valueRaw.startsWith(\"{\")) {\n return null;\n }\n const normalized = normalizeValue(valueRaw);\n if (normalized === null) {\n return null;\n }\n if (typeof normalized === \"object\" && !Array.isArray(normalized)) {\n return null;\n }\n output[key] = normalized;\n }\n\n return output;\n}\n\nexport function normalizeValue(value: string): TomlValue | null {\n if (!value) {\n return null;\n }\n\n const trimmed = value.trim();\n if (trimmed.startsWith(\"{\") && trimmed.endsWith(\"}\")) {\n return parseInlineTable(trimmed);\n }\n\n const array = parseArray(trimmed);\n if (array) {\n return array;\n }\n\n if (trimmed.startsWith(\"[\") && trimmed.endsWith(\"]\")) {\n return null;\n }\n\n return parsePrimitive(trimmed);\n}\n\nexport function toPlainText(value: unknown): string {\n if (value == null) {\n return \"\";\n }\n if (Array.isArray(value)) {\n return value.map((item) => String(item)).join(\", \");\n }\n return String(value);\n}\n","import { ensureArrayContainer, flattenArrayTables } from \"./arrays\";\nimport { normalizeKeyPath } from \"./keys\";\nimport { stripInlineComment } from \"./strings\";\nimport { normalizeValue, toPlainText } from \"./values\";\n\nexport function parseTomlFrontmatter(frontmatter: string): Record<string, unknown> | null {\n const result: Record<string, unknown> = {};\n const lines = frontmatter.split(\"\\n\");\n let tablePrefix = \"\";\n let tableTarget: Record<string, unknown> | null = null;\n let tablePrefixInList = false;\n\n for (let index = 0; index < lines.length; index += 1) {\n const rawLine = lines[index] ?? \"\";\n const trimmedLine = rawLine.trim();\n if (!trimmedLine || trimmedLine.startsWith(\"#\")) {\n continue;\n }\n\n if (trimmedLine.startsWith(\"[[\")) {\n const match = trimmedLine.match(/^\\[\\[([^\\]]+)]]$/);\n if (!match) {\n return null;\n }\n const normalizedTable = normalizeKeyPath(match[1] ?? \"\");\n if (!normalizedTable) {\n return null;\n }\n const list = ensureArrayContainer(result, normalizedTable);\n const newEntry: Record<string, unknown> = {};\n list.push(newEntry);\n tableTarget = newEntry;\n tablePrefix = normalizedTable;\n tablePrefixInList = true;\n continue;\n }\n\n const tableMatch = trimmedLine.match(/^\\[([^\\]]+)]$/);\n if (tableMatch) {\n const normalizedTable = normalizeKeyPath(tableMatch[1] ?? \"\");\n if (!normalizedTable) {\n return null;\n }\n tablePrefix = normalizedTable;\n tablePrefixInList = false;\n tableTarget = null;\n continue;\n }\n\n const lineForParsing = /(\"\"\"|''')/.test(rawLine) ? rawLine : stripInlineComment(rawLine);\n const separatorIndex = lineForParsing.indexOf(\"=\");\n if (separatorIndex === -1) {\n return null;\n }\n\n const keyRaw = lineForParsing.slice(0, separatorIndex);\n const key = normalizeKeyPath(keyRaw);\n let valueRaw = lineForParsing.slice(separatorIndex + 1).trim();\n if (!key) {\n return null;\n }\n\n const tripleDelimiter = valueRaw.startsWith('\"\"\"') ? '\"\"\"' : valueRaw.startsWith(\"'''\") ? \"'''\" : null;\n if (tripleDelimiter) {\n const closingIndex = valueRaw.indexOf(tripleDelimiter, tripleDelimiter.length);\n if (closingIndex !== -1) {\n const after = valueRaw.slice(closingIndex + tripleDelimiter.length);\n const strippedAfter = stripInlineComment(after);\n valueRaw = `${valueRaw.slice(0, closingIndex + tripleDelimiter.length)}${strippedAfter}`;\n } else {\n const delimiter = tripleDelimiter;\n let combined = valueRaw;\n let closed = false;\n while (index + 1 < lines.length) {\n index += 1;\n const nextLine = lines[index] ?? \"\";\n combined += `\\n${nextLine}`;\n if (new RegExp(`${delimiter}\\\\s*$`).test(nextLine)) {\n closed = true;\n break;\n }\n }\n if (!closed) {\n return null;\n }\n valueRaw = combined;\n }\n }\n\n const normalized = normalizeValue(valueRaw);\n if (normalized === null) {\n return null;\n }\n\n const fullKey = tablePrefix ? `${tablePrefix}.${key}` : key;\n if (typeof normalized === \"object\" && !Array.isArray(normalized)) {\n for (const [inlineKey, inlineValue] of Object.entries(normalized)) {\n const entryKey = tablePrefixInList ? `${key}.${inlineKey}` : `${fullKey}.${inlineKey}`;\n if (tablePrefixInList && tableTarget) {\n tableTarget[entryKey] = toPlainText(inlineValue);\n } else {\n result[entryKey] = toPlainText(inlineValue);\n }\n }\n continue;\n }\n\n if (tablePrefixInList && tableTarget) {\n tableTarget[key] = toPlainText(normalized);\n continue;\n }\n\n result[fullKey] = toPlainText(normalized);\n }\n\n flattenArrayTables(result);\n\n return result;\n}\n","import { parseDocument } from \"yaml\";\nimport { parseTomlFrontmatter } from \"./toml-simple\";\nimport type { FrontmatterType, ParsedMarkdown } from \"./types\";\n\nconst FENCE_TO_TYPE: Record<string, FrontmatterType> = {\n \"---\": \"yaml\",\n \"+++\": \"toml\",\n \";;;\": \"json\",\n};\n\nfunction normalizeNewlines(input: string): string {\n return input.replace(/\\r\\n/g, \"\\n\");\n}\n\nfunction stripBom(line: string): string {\n return line.startsWith(\"\\uFEFF\") ? line.slice(1) : line;\n}\n\nfunction getFenceType(line: string): FrontmatterType | null {\n const match = line.match(/^[\\t ]*(---|\\+\\+\\+|;;;)[\\t ]*$/);\n if (!match) {\n return null;\n }\n return FENCE_TO_TYPE[match[1] ?? \"\"] ?? null;\n}\n\nfunction parseFrontmatter(frontmatter: string, type: FrontmatterType | null): Record<string, unknown> | null {\n if (!type) {\n return null;\n }\n\n if (type === \"json\") {\n try {\n return JSON.parse(frontmatter) as Record<string, unknown>;\n } catch {\n return null;\n }\n }\n\n if (type === \"yaml\") {\n const doc = parseDocument(frontmatter, { prettyErrors: false });\n if (doc.errors.length > 0) {\n return null;\n }\n const data = doc.toJSON();\n if (!data || typeof data !== \"object\" || Array.isArray(data)) {\n return null;\n }\n return data as Record<string, unknown>;\n }\n\n if (type === \"toml\") {\n return parseTomlFrontmatter(frontmatter);\n }\n\n return null;\n}\n\nfunction extractJsonBlock(text: string, startIndex: number): { jsonText: string; endIndex: number } | null {\n let depth = 0;\n let inString = false;\n let escaped = false;\n\n for (let i = startIndex; i < text.length; i += 1) {\n const char = text[i] ?? \"\";\n\n if (inString) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === \"\\\\\") {\n escaped = true;\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = false;\n }\n\n continue;\n }\n\n if (char === \"\\\"\") {\n inString = true;\n continue;\n }\n\n if (char === \"{\") {\n depth += 1;\n continue;\n }\n\n if (char === \"}\") {\n depth -= 1;\n if (depth === 0) {\n const jsonText = text.slice(startIndex, i + 1);\n return { jsonText, endIndex: i };\n }\n }\n }\n\n return null;\n}\n\nexport function parseMarkdown(input: string): ParsedMarkdown {\n const normalized = normalizeNewlines(input);\n const lines = normalized.split(\"\\n\");\n if (lines.length === 0) {\n return { frontmatter: null, content: normalized, data: null, frontmatterType: null };\n }\n\n lines[0] = stripBom(lines[0] ?? \"\");\n const normalizedWithoutBom = lines.join(\"\\n\");\n\n const openingType = getFenceType(lines[0] ?? \"\");\n if (!openingType) {\n const leadingWhitespace = normalizedWithoutBom.match(/^[\\t \\n]*/)?.[0] ?? \"\";\n const jsonStart = leadingWhitespace.length;\n if (normalizedWithoutBom[jsonStart] !== \"{\") {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const jsonBlock = extractJsonBlock(normalizedWithoutBom, jsonStart);\n if (!jsonBlock) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const frontmatter = jsonBlock.jsonText;\n let content = normalizedWithoutBom.slice(jsonBlock.endIndex + 1);\n if (content.startsWith(\"\\n\")) {\n content = content.slice(1);\n }\n const data = parseFrontmatter(frontmatter, \"json\");\n if (!data) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n return {\n frontmatter,\n content,\n data,\n frontmatterType: \"json\",\n };\n }\n\n let closingIndex = -1;\n for (let i = 1; i < lines.length; i += 1) {\n if (getFenceType(lines[i] ?? \"\") === openingType) {\n closingIndex = i;\n break;\n }\n }\n\n if (closingIndex === -1) {\n return { frontmatter: null, content: normalizedWithoutBom, data: null, frontmatterType: null };\n }\n\n const frontmatter = lines.slice(1, closingIndex).join(\"\\n\");\n const content = lines.slice(closingIndex + 1).join(\"\\n\");\n const data = parseFrontmatter(frontmatter, openingType);\n\n return {\n frontmatter,\n content,\n data,\n frontmatterType: openingType,\n };\n}\n","import type { WordCounterMode, WordCounterOptions, WordCounterResult } from \"../wc/types\";\nimport wordCounter from \"../wc\";\nimport { parseMarkdown } from \"./parse-markdown\";\nimport type { SectionMode, SectionedResult } from \"./types\";\n\nfunction normalizeText(value: unknown): string {\n if (value == null) {\n return \"\";\n }\n if (typeof value === \"string\") {\n return value;\n }\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n return String(value);\n }\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n}\n\nfunction buildPerKeyItems(\n data: Record<string, unknown> | null,\n mode: WordCounterMode,\n options: WordCounterOptions,\n): Array<{ name: string; source: \"frontmatter\"; result: WordCounterResult }> {\n if (!data || typeof data !== \"object\" || Array.isArray(data)) {\n return [];\n }\n\n return Object.entries(data).map(([key, value]) => {\n const valueText = normalizeText(value);\n const text = valueText ? `${key}: ${valueText}` : key;\n return {\n name: key,\n source: \"frontmatter\",\n result: wordCounter(text, options),\n };\n });\n}\n\nfunction buildSingleItem(\n name: string,\n text: string,\n mode: WordCounterMode,\n options: WordCounterOptions,\n source: \"frontmatter\" | \"content\",\n) {\n return [{ name, source, result: wordCounter(text, options) }];\n}\n\nfunction sumTotals(items: Array<{ result: WordCounterResult }>): number {\n return items.reduce((sum, item) => sum + item.result.total, 0);\n}\n\nexport function countSections(\n input: string,\n section: SectionMode,\n options: WordCounterOptions = {},\n): SectionedResult {\n const mode: WordCounterMode = options.mode ?? \"chunk\";\n if (section === \"all\") {\n const result = wordCounter(input, options);\n return {\n section,\n total: result.total,\n frontmatterType: null,\n items: [{ name: \"all\", source: \"content\", result }],\n };\n }\n\n const parsed = parseMarkdown(input);\n const frontmatterText = parsed.frontmatter ?? \"\";\n const contentText = parsed.content ?? \"\";\n\n let items: Array<{ name: string; source: \"frontmatter\" | \"content\"; result: WordCounterResult }> = [];\n\n if (section === \"frontmatter\") {\n items = buildSingleItem(\"frontmatter\", frontmatterText, mode, options, \"frontmatter\");\n } else if (section === \"content\") {\n items = buildSingleItem(\"content\", contentText, mode, options, \"content\");\n } else if (section === \"split\") {\n items = [\n ...buildSingleItem(\"frontmatter\", frontmatterText, mode, options, \"frontmatter\"),\n ...buildSingleItem(\"content\", contentText, mode, options, \"content\"),\n ];\n } else if (section === \"per-key\") {\n items = buildPerKeyItems(parsed.data, mode, options);\n } else if (section === \"split-per-key\") {\n items = [\n ...buildPerKeyItems(parsed.data, mode, options),\n ...buildSingleItem(\"content\", contentText, mode, options, \"content\"),\n ];\n }\n\n return {\n section,\n total: sumTotals(items),\n frontmatterType: parsed.frontmatterType,\n items,\n };\n}\n","export function showSingularOrPluralWord(count: number, word: string): string {\n return `${count} ${word}${count === 1 ? \"\" : \"s\"}`;\n}\n","import wordCounter, {\n countCharsForLocale,\n countWordsForLocale,\n segmentTextByLocale,\n} from \"./wc\";\nimport { parseMarkdown, countSections } from \"./markdown\";\nimport { showSingularOrPluralWord } from \"./utils\";\n\nconst cjsExports = Object.assign(wordCounter, {\n default: wordCounter,\n wordCounter,\n countCharsForLocale,\n countWordsForLocale,\n segmentTextByLocale,\n parseMarkdown,\n countSections,\n showSingularOrPluralWord,\n});\n\nexport = cjsExports;\n"],"mappings":"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"}
|