vitepress-allyouneed 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js ADDED
@@ -0,0 +1,1441 @@
1
+ // src/core/slugify.ts
2
+ import { slugify as mditVueSlugify } from "@mdit-vue/shared";
3
+ function defaultSlugify(text) {
4
+ return mditVueSlugify(text);
5
+ }
6
+ var CUSTOM_ID_RE = /\s*\{#([^}\s]+)\}\s*$/;
7
+ function extractCustomId(headingText) {
8
+ const m = headingText.match(CUSTOM_ID_RE);
9
+ if (!m) return { text: headingText, customId: void 0 };
10
+ return {
11
+ text: headingText.replace(CUSTOM_ID_RE, ""),
12
+ customId: m[1]
13
+ };
14
+ }
15
+
16
+ // src/core/config-bridge.ts
17
+ var DEFAULT_ASSET_EXTENSIONS = [
18
+ // 位图
19
+ "bmp",
20
+ "gif",
21
+ "jpeg",
22
+ "jpg",
23
+ "png",
24
+ "svg",
25
+ "webp",
26
+ "avif",
27
+ "ico",
28
+ // 视频
29
+ "mp4",
30
+ "webm",
31
+ "mov",
32
+ "m4v",
33
+ // 音频
34
+ "mp3",
35
+ "wav",
36
+ "ogg",
37
+ "m4a",
38
+ "flac",
39
+ // 文档
40
+ "pdf",
41
+ // Obsidian 专属
42
+ "canvas",
43
+ "excalidraw"
44
+ ];
45
+ var DEFAULT_IMAGE_EXTENSIONS = [
46
+ "bmp",
47
+ "gif",
48
+ "jpeg",
49
+ "jpg",
50
+ "png",
51
+ "svg",
52
+ "webp",
53
+ "avif",
54
+ "ico"
55
+ ];
56
+ function resolveOptions(user = {}, ctx = {}) {
57
+ const srcDir = user.srcDir ?? ctx.srcDir ?? process.cwd();
58
+ let base = user.base ?? ctx.base ?? "/";
59
+ if (!base.startsWith("/")) base = "/" + base;
60
+ if (!base.endsWith("/")) base = base + "/";
61
+ const cleanUrls = user.cleanUrls ?? ctx.cleanUrls ?? false;
62
+ const slugify = user.slugify ?? ctx.externalSlugify ?? defaultSlugify;
63
+ const wikilinksUser = user.wikilinks ?? {};
64
+ const embedsUser = user.embeds ?? {};
65
+ const scanUser = user.scan ?? {};
66
+ const assetsUser = user.assets ?? {};
67
+ const modulesUser = user.modules ?? {};
68
+ const wikilinksHtmlAttrs = wikilinksUser.htmlAttributes ?? {};
69
+ const embedsHtmlAttrs = embedsUser.htmlAttributes ?? {};
70
+ return {
71
+ srcDir,
72
+ base,
73
+ cleanUrls,
74
+ caseSensitive: user.caseSensitive ?? false,
75
+ deadLink: user.deadLink ?? "warn",
76
+ onConflict: user.onConflict ?? "shortest",
77
+ onAliasConflict: user.onAliasConflict ?? "first",
78
+ scan: {
79
+ include: scanUser.include ?? ["**/*.md", "**/*.markdown"],
80
+ exclude: scanUser.exclude ?? [],
81
+ followSymlinks: scanUser.followSymlinks ?? false,
82
+ respectGitignore: scanUser.respectGitignore ?? true,
83
+ assetExtensions: scanUser.assetExtensions ?? DEFAULT_ASSET_EXTENSIONS
84
+ },
85
+ assets: {
86
+ mode: assetsUser.mode ?? "auto",
87
+ preserveAssetPaths: assetsUser.preserveAssetPaths ?? false,
88
+ outputDir: assetsUser.outputDir ?? "_assets"
89
+ },
90
+ wikilinks: {
91
+ postProcessLinkTarget: wikilinksUser.postProcessLinkTarget ?? ((t) => t.trim()),
92
+ postProcessLinkLabel: wikilinksUser.postProcessLinkLabel ?? ((l) => l.trim()),
93
+ allowLinkLabelFormatting: wikilinksUser.allowLinkLabelFormatting ?? false,
94
+ linkText: wikilinksUser.linkText ?? "basename",
95
+ htmlAttributes: wikilinksHtmlAttrs
96
+ },
97
+ embeds: {
98
+ imageFileExt: embedsUser.imageFileExt ?? DEFAULT_IMAGE_EXTENSIONS,
99
+ defaultAltText: embedsUser.defaultAltText ?? false,
100
+ postProcessImageTarget: embedsUser.postProcessImageTarget ?? ((t) => t.trim()),
101
+ postProcessAltText: embedsUser.postProcessAltText ?? ((a) => a.trim()),
102
+ uriSuffix: embedsUser.uriSuffix ?? "",
103
+ transclusionMaxDepth: embedsUser.transclusionMaxDepth ?? 8,
104
+ htmlAttributes: embedsHtmlAttrs
105
+ },
106
+ modules: {
107
+ wikilinks: modulesUser.wikilinks ?? true,
108
+ embeds: modulesUser.embeds ?? true
109
+ },
110
+ slugify
111
+ };
112
+ }
113
+
114
+ // src/core/vault/index.ts
115
+ import fs3 from "fs";
116
+ import nodePath4 from "path";
117
+
118
+ // src/utils/path.ts
119
+ import nodePath from "path";
120
+ function toPosix(p) {
121
+ return p.replace(/\\/g, "/");
122
+ }
123
+ function relative(from, to) {
124
+ return toPosix(nodePath.relative(from, to));
125
+ }
126
+ function stripMarkdownExt(target) {
127
+ return target.replace(/\.(md|markdown)$/i, "");
128
+ }
129
+ function basename(p, stripExt = false) {
130
+ const idx = p.lastIndexOf("/");
131
+ const file = idx === -1 ? p : p.slice(idx + 1);
132
+ if (!stripExt) return file;
133
+ const dot = file.lastIndexOf(".");
134
+ return dot <= 0 ? file : file.slice(0, dot);
135
+ }
136
+ function extname(p) {
137
+ const file = basename(p);
138
+ const dot = file.lastIndexOf(".");
139
+ if (dot <= 0) return "";
140
+ return file.slice(dot + 1).toLowerCase();
141
+ }
142
+ function splitPath(p) {
143
+ return toPosix(p).split("/").filter(Boolean);
144
+ }
145
+ function pathDepth(relPath) {
146
+ return splitPath(relPath).length;
147
+ }
148
+
149
+ // src/utils/url.ts
150
+ function encodePath(s) {
151
+ return encodeURI(s);
152
+ }
153
+ function buildUrl(base, pathSegments, anchor) {
154
+ let normBase = base || "/";
155
+ if (!normBase.startsWith("/")) normBase = "/" + normBase;
156
+ if (!normBase.endsWith("/")) normBase = normBase + "/";
157
+ const joined = pathSegments.map((s) => s.replace(/^\/+|\/+$/g, "")).filter(Boolean).join("/");
158
+ let url = normBase + joined;
159
+ url = url.replace(/\/{2,}/g, "/");
160
+ url = encodePath(url);
161
+ if (anchor) {
162
+ url += "#" + encodeURIComponent(anchor).replace(/%2F/g, "/");
163
+ }
164
+ return url;
165
+ }
166
+ function applyCleanUrls(path, cleanUrls) {
167
+ if (cleanUrls) return path;
168
+ if (/\.html$/i.test(path)) return path;
169
+ if (path.endsWith("/")) return path + "index.html";
170
+ return path + ".html";
171
+ }
172
+
173
+ // src/core/vault/scan.ts
174
+ import fs from "fs";
175
+ import nodePath2 from "path";
176
+ function walk(srcDir, isIgnored, followSymlinks) {
177
+ const out = [];
178
+ const seenInodes = /* @__PURE__ */ new Set();
179
+ function visit(dir) {
180
+ let entries;
181
+ try {
182
+ entries = fs.readdirSync(dir, { withFileTypes: true });
183
+ } catch {
184
+ return;
185
+ }
186
+ for (const ent of entries) {
187
+ const full = nodePath2.join(dir, ent.name);
188
+ const posix = toPosix(full);
189
+ if (isIgnored(posix)) continue;
190
+ let isDir = ent.isDirectory();
191
+ let isFile = ent.isFile();
192
+ if (ent.isSymbolicLink()) {
193
+ if (!followSymlinks) continue;
194
+ try {
195
+ const stat = fs.statSync(full);
196
+ isDir = stat.isDirectory();
197
+ isFile = stat.isFile();
198
+ } catch {
199
+ continue;
200
+ }
201
+ }
202
+ if (isDir) {
203
+ if (followSymlinks) {
204
+ try {
205
+ const stat = fs.statSync(full);
206
+ const key = `${stat.dev}:${stat.ino}`;
207
+ if (seenInodes.has(key)) continue;
208
+ seenInodes.add(key);
209
+ } catch {
210
+ continue;
211
+ }
212
+ }
213
+ visit(full);
214
+ } else if (isFile) {
215
+ try {
216
+ const stat = fs.statSync(full);
217
+ out.push({
218
+ absolutePath: posix,
219
+ size: stat.size,
220
+ mtime: stat.mtimeMs,
221
+ extension: extname(posix)
222
+ });
223
+ } catch {
224
+ }
225
+ }
226
+ }
227
+ }
228
+ visit(srcDir);
229
+ return out;
230
+ }
231
+
232
+ // src/core/vault/ignore.ts
233
+ import fs2 from "fs";
234
+ import nodePath3 from "path";
235
+ import picomatch from "picomatch";
236
+ var HARD_IGNORE_DIRS = /* @__PURE__ */ new Set([
237
+ "node_modules",
238
+ ".git",
239
+ ".svn",
240
+ ".hg",
241
+ ".obsidian",
242
+ ".trash",
243
+ ".vitepress",
244
+ ".next",
245
+ ".nuxt",
246
+ ".cache",
247
+ ".idea",
248
+ ".vscode",
249
+ "dist",
250
+ "build"
251
+ ]);
252
+ function buildIgnorer(srcDir, userExclude, respectGitignore) {
253
+ const patterns = [...userExclude];
254
+ if (respectGitignore) {
255
+ const gitignore = nodePath3.join(srcDir, ".gitignore");
256
+ try {
257
+ const content = fs2.readFileSync(gitignore, "utf8");
258
+ for (const line of content.split(/\r?\n/)) {
259
+ const trimmed = line.trim();
260
+ if (!trimmed || trimmed.startsWith("#")) continue;
261
+ patterns.push(trimmed.endsWith("/") ? trimmed + "**" : trimmed);
262
+ }
263
+ } catch {
264
+ }
265
+ }
266
+ const matchers = patterns.map(
267
+ (p) => picomatch(p, { dot: true, nocase: false })
268
+ );
269
+ return (absPath) => {
270
+ const rel = toPosix(relative(srcDir, absPath));
271
+ if (!rel || rel.startsWith("..")) return true;
272
+ for (const seg of rel.split("/")) {
273
+ if (HARD_IGNORE_DIRS.has(seg)) return true;
274
+ }
275
+ for (const m of matchers) {
276
+ if (m(rel)) return true;
277
+ }
278
+ return false;
279
+ };
280
+ }
281
+
282
+ // src/core/vault/frontmatter.ts
283
+ import matter from "gray-matter";
284
+ function parseFrontmatter(raw) {
285
+ try {
286
+ const { data, content } = matter(raw);
287
+ return { data: data ?? {}, content };
288
+ } catch (err) {
289
+ return {
290
+ data: {},
291
+ content: raw,
292
+ error: err instanceof Error ? err.message : String(err)
293
+ };
294
+ }
295
+ }
296
+ function normalizeAliases(raw) {
297
+ if (raw == null) return [];
298
+ if (typeof raw === "string") return raw.trim() ? [raw.trim()] : [];
299
+ if (Array.isArray(raw)) {
300
+ return raw.filter((v) => typeof v === "string").map((s) => s.trim()).filter(Boolean);
301
+ }
302
+ return [];
303
+ }
304
+ function normalizeTags(raw) {
305
+ if (raw == null) return [];
306
+ if (typeof raw === "string") {
307
+ return raw.split(/[,\s]+/).map((s) => s.trim().replace(/^#/, "")).filter(Boolean);
308
+ }
309
+ if (Array.isArray(raw)) {
310
+ return raw.filter((v) => typeof v === "string").map((s) => s.trim().replace(/^#/, "")).filter(Boolean);
311
+ }
312
+ return [];
313
+ }
314
+
315
+ // src/core/vault/headings.ts
316
+ var HEADING_RE = /^(#{1,6})\s+(.+?)\s*$/;
317
+ var FENCE_RE = /^(`{3,}|~{3,})/;
318
+ function collectHeadings(content, slugify) {
319
+ const lines = content.split(/\r?\n/);
320
+ const out = [];
321
+ let inFence = false;
322
+ let fenceMarker = "";
323
+ for (let i = 0; i < lines.length; i++) {
324
+ const line = lines[i];
325
+ const fenceMatch = line.match(FENCE_RE);
326
+ if (fenceMatch) {
327
+ if (!inFence) {
328
+ inFence = true;
329
+ fenceMarker = fenceMatch[1];
330
+ } else if (line.startsWith(fenceMarker)) {
331
+ inFence = false;
332
+ fenceMarker = "";
333
+ }
334
+ continue;
335
+ }
336
+ if (inFence) continue;
337
+ const m = line.match(HEADING_RE);
338
+ if (!m) continue;
339
+ const level = m[1].length;
340
+ const rawText = m[2];
341
+ const { text, customId } = extractCustomId(rawText);
342
+ const slug = customId ?? slugify(text);
343
+ out.push({ level, text, slug, line: i });
344
+ }
345
+ return out;
346
+ }
347
+
348
+ // src/core/vault/index.ts
349
+ var MD_EXTENSIONS = /* @__PURE__ */ new Set(["md", "markdown"]);
350
+ function createEmptyIndex(srcDir = "", base = "/", cleanUrls = false) {
351
+ return {
352
+ files: /* @__PURE__ */ new Map(),
353
+ assets: /* @__PURE__ */ new Map(),
354
+ byBasename: /* @__PURE__ */ new Map(),
355
+ byBasenameLower: /* @__PURE__ */ new Map(),
356
+ byAlias: /* @__PURE__ */ new Map(),
357
+ byRelativePath: /* @__PURE__ */ new Map(),
358
+ byUrl: /* @__PURE__ */ new Map(),
359
+ assetsByBasename: /* @__PURE__ */ new Map(),
360
+ assetsByBasenameLower: /* @__PURE__ */ new Map(),
361
+ assetsByRelativePath: /* @__PURE__ */ new Map(),
362
+ tags: /* @__PURE__ */ new Map(),
363
+ backlinks: /* @__PURE__ */ new Map(),
364
+ headings: /* @__PURE__ */ new Map(),
365
+ srcDir,
366
+ base,
367
+ cleanUrls,
368
+ scannedAt: Date.now(),
369
+ warnings: []
370
+ };
371
+ }
372
+ function scanVault(options) {
373
+ const srcDir = toPosix(nodePath4.resolve(options.srcDir));
374
+ const index = createEmptyIndex(srcDir, options.base, options.cleanUrls);
375
+ const isIgnored = buildIgnorer(
376
+ srcDir,
377
+ options.scan.exclude,
378
+ options.scan.respectGitignore
379
+ );
380
+ const assetExtSet = new Set(
381
+ options.scan.assetExtensions.map((e) => e.toLowerCase())
382
+ );
383
+ const entries = walk(srcDir, isIgnored, options.scan.followSymlinks);
384
+ for (const ent of entries) {
385
+ const ext = ent.extension;
386
+ if (MD_EXTENSIONS.has(ext)) {
387
+ ingestMarkdown(index, ent.absolutePath, ent.size, ent.mtime, options);
388
+ } else if (assetExtSet.has(ext)) {
389
+ ingestAsset(index, ent.absolutePath, ent.size, ent.mtime, ext);
390
+ }
391
+ }
392
+ index.scannedAt = Date.now();
393
+ return index;
394
+ }
395
+ function ingestMarkdown(index, absPath, size, mtime, options) {
396
+ let raw;
397
+ try {
398
+ raw = fs3.readFileSync(absPath, "utf8");
399
+ } catch (err) {
400
+ index.warnings.push({
401
+ kind: "unreadable-file",
402
+ message: `\u65E0\u6CD5\u8BFB\u53D6\u6587\u4EF6: ${absPath} (${err instanceof Error ? err.message : String(err)})`,
403
+ affected: [absPath]
404
+ });
405
+ return;
406
+ }
407
+ const { data, content, error } = parseFrontmatter(raw);
408
+ if (error) {
409
+ index.warnings.push({
410
+ kind: "invalid-frontmatter",
411
+ message: `frontmatter \u89E3\u6790\u5931\u8D25 (${absPath}): ${error}`,
412
+ affected: [absPath]
413
+ });
414
+ }
415
+ const aliases = normalizeAliases(data.aliases);
416
+ const tags = normalizeTags(data.tags);
417
+ const headings = collectHeadings(content, options.slugify);
418
+ const rel = relative(index.srcDir, absPath);
419
+ const base = basename(absPath, true);
420
+ const ext = extname(absPath);
421
+ const url = computeUrl(rel, options);
422
+ const entry = {
423
+ absolutePath: absPath,
424
+ relativePath: rel,
425
+ basename: base,
426
+ extension: ext,
427
+ url,
428
+ frontmatter: data,
429
+ aliases,
430
+ tags,
431
+ headings,
432
+ mtime,
433
+ size,
434
+ content
435
+ };
436
+ registerFileEntry(index, entry, options);
437
+ }
438
+ function ingestAsset(index, absPath, size, mtime, ext) {
439
+ const rel = relative(index.srcDir, absPath);
440
+ const base = basename(absPath);
441
+ const entry = {
442
+ absolutePath: absPath,
443
+ relativePath: rel,
444
+ basename: base,
445
+ extension: ext,
446
+ mtime,
447
+ size,
448
+ referencedBy: /* @__PURE__ */ new Set()
449
+ };
450
+ index.assets.set(absPath, entry);
451
+ index.assetsByRelativePath.set(rel, entry);
452
+ pushToArrayMap(index.assetsByBasename, base, entry);
453
+ pushToArrayMap(index.assetsByBasenameLower, base.toLowerCase(), entry);
454
+ }
455
+ function registerFileEntry(index, entry, options) {
456
+ index.files.set(entry.absolutePath, entry);
457
+ index.byRelativePath.set(entry.relativePath, entry);
458
+ const existingAtUrl = index.byUrl.get(entry.url);
459
+ if (existingAtUrl && existingAtUrl.absolutePath !== entry.absolutePath) {
460
+ index.warnings.push({
461
+ kind: "unknown",
462
+ message: `URL \u51B2\u7A81:\u6587\u4EF6 "${entry.relativePath}" \u548C "${existingAtUrl.relativePath}" \u90FD\u8DEF\u7531\u5230 "${entry.url}"\u3002VitePress \u4F1A\u8BA9\u5176\u4E2D\u4E00\u4E2A 404\u3002\u5EFA\u8BAE\u5728 .vitepress/config \u52A0 srcExclude: ['${entry.relativePath}'](\u6216\u53E6\u4E00\u4E2A)\u3002`,
463
+ affected: [existingAtUrl.absolutePath, entry.absolutePath]
464
+ });
465
+ }
466
+ index.byUrl.set(entry.url, entry);
467
+ index.headings.set(entry.absolutePath, entry.headings);
468
+ pushToArrayMap(index.byBasename, entry.basename, entry);
469
+ pushToArrayMap(index.byBasenameLower, entry.basename.toLowerCase(), entry);
470
+ for (const alias of entry.aliases) {
471
+ const key = options.caseSensitive ? alias : alias.toLowerCase();
472
+ if (index.byAlias.has(key)) {
473
+ index.warnings.push({
474
+ kind: "duplicate-alias",
475
+ message: `alias "${alias}" \u540C\u65F6\u88AB\u591A\u4E2A\u6587\u4EF6\u58F0\u660E,\u6309 onAliasConflict='${options.onAliasConflict}' \u5904\u7406`,
476
+ affected: [index.byAlias.get(key).absolutePath, entry.absolutePath]
477
+ });
478
+ if (options.onAliasConflict === "first") continue;
479
+ }
480
+ index.byAlias.set(key, entry);
481
+ }
482
+ for (const tag of entry.tags) {
483
+ pushToArrayMap(index.tags, tag, entry);
484
+ }
485
+ }
486
+ function computeUrl(rel, options) {
487
+ const noExt = rel.replace(/\.(md|markdown)$/i, "");
488
+ const isIndex = /(^|\/)(index|README)$/i.test(noExt);
489
+ const pathPart = isIndex ? noExt.replace(/(^|\/)(index|README)$/i, "$1") : noExt;
490
+ const segments = pathPart.split("/").filter(Boolean);
491
+ if (segments.length === 0) {
492
+ return options.base;
493
+ }
494
+ const last = segments[segments.length - 1];
495
+ if (!isIndex) {
496
+ segments[segments.length - 1] = applyCleanUrls(last, options.cleanUrls);
497
+ } else if (!options.cleanUrls) {
498
+ segments.push("index.html");
499
+ }
500
+ return buildUrl(options.base, segments);
501
+ }
502
+ function pushToArrayMap(m, k, v) {
503
+ const arr = m.get(k);
504
+ if (arr) arr.push(v);
505
+ else m.set(k, [v]);
506
+ }
507
+ function updateFile(index, absPath, options) {
508
+ const posix = toPosix(absPath);
509
+ removeFile(index, posix, options);
510
+ let stat;
511
+ try {
512
+ stat = fs3.statSync(posix);
513
+ } catch {
514
+ return;
515
+ }
516
+ const ext = extname(posix);
517
+ if (MD_EXTENSIONS.has(ext)) {
518
+ ingestMarkdown(index, posix, stat.size, stat.mtimeMs, options);
519
+ } else if (new Set(options.scan.assetExtensions.map((e) => e.toLowerCase())).has(ext)) {
520
+ ingestAsset(index, posix, stat.size, stat.mtimeMs, ext);
521
+ }
522
+ }
523
+ function removeFile(index, absPath, options) {
524
+ const posix = toPosix(absPath);
525
+ const file = index.files.get(posix);
526
+ if (file) {
527
+ index.files.delete(posix);
528
+ index.byRelativePath.delete(file.relativePath);
529
+ index.byUrl.delete(file.url);
530
+ index.headings.delete(posix);
531
+ removeFromArrayMap(index.byBasename, file.basename, file);
532
+ removeFromArrayMap(
533
+ index.byBasenameLower,
534
+ file.basename.toLowerCase(),
535
+ file
536
+ );
537
+ for (const alias of file.aliases) {
538
+ const key = options.caseSensitive ? alias : alias.toLowerCase();
539
+ if (index.byAlias.get(key) === file) index.byAlias.delete(key);
540
+ }
541
+ for (const tag of file.tags) {
542
+ removeFromArrayMap(index.tags, tag, file);
543
+ }
544
+ return;
545
+ }
546
+ const asset = index.assets.get(posix);
547
+ if (asset) {
548
+ index.assets.delete(posix);
549
+ index.assetsByRelativePath.delete(asset.relativePath);
550
+ removeFromArrayMap(index.assetsByBasename, asset.basename, asset);
551
+ removeFromArrayMap(
552
+ index.assetsByBasenameLower,
553
+ asset.basename.toLowerCase(),
554
+ asset
555
+ );
556
+ }
557
+ }
558
+ function removeFromArrayMap(m, k, v) {
559
+ const arr = m.get(k);
560
+ if (!arr) return;
561
+ const idx = arr.indexOf(v);
562
+ if (idx >= 0) arr.splice(idx, 1);
563
+ if (arr.length === 0) m.delete(k);
564
+ }
565
+ function sortByShortestPath(items) {
566
+ return [...items].sort((a, b) => {
567
+ const da = pathDepth(a.relativePath);
568
+ const db = pathDepth(b.relativePath);
569
+ if (da !== db) return da - db;
570
+ return a.relativePath.localeCompare(b.relativePath);
571
+ });
572
+ }
573
+
574
+ // src/core/resolver.ts
575
+ function resolveWikilink(rawTarget, index, options, kind = "page") {
576
+ let target = toPosix(rawTarget).trim();
577
+ const hashIdx = target.indexOf("#");
578
+ let headingPart = "";
579
+ if (hashIdx >= 0) {
580
+ headingPart = target.slice(hashIdx + 1).trim();
581
+ target = target.slice(0, hashIdx).trim();
582
+ }
583
+ target = stripMarkdownExt(target);
584
+ const entry = lookupEntry(target, index, options);
585
+ if (!entry) {
586
+ return {
587
+ url: buildDeadUrl(rawTarget, options),
588
+ defaultLabel: defaultLabel(target, headingPart, void 0, options),
589
+ isDead: true,
590
+ hasUnmatchedAnchor: false,
591
+ kind
592
+ };
593
+ }
594
+ let url = entry.url;
595
+ let hasUnmatchedAnchor = false;
596
+ if (headingPart) {
597
+ const heading = entry.headings.find(
598
+ (h) => h.text === headingPart || h.slug === headingPart || h.slug === options.slugify(headingPart)
599
+ );
600
+ if (heading) {
601
+ url = entry.url + "#" + heading.slug;
602
+ } else {
603
+ hasUnmatchedAnchor = true;
604
+ url = entry.url + "#" + encodeURIComponent(headingPart);
605
+ }
606
+ }
607
+ return {
608
+ url,
609
+ defaultLabel: defaultLabel(target, headingPart, entry, options),
610
+ isDead: false,
611
+ hasUnmatchedAnchor,
612
+ target: entry,
613
+ kind
614
+ };
615
+ }
616
+ function lookupEntry(target, index, options) {
617
+ if (!target) return void 0;
618
+ if (target.includes("/")) {
619
+ const variants = [
620
+ target,
621
+ target + ".md",
622
+ target + ".markdown",
623
+ target + "/index.md",
624
+ target + "/index.markdown"
625
+ ];
626
+ for (const v of variants) {
627
+ const e = index.byRelativePath.get(v);
628
+ if (e) return e;
629
+ }
630
+ return void 0;
631
+ }
632
+ const aliasKey = options.caseSensitive ? target : target.toLowerCase();
633
+ const aliased = index.byAlias.get(aliasKey);
634
+ if (aliased) return aliased;
635
+ const bnMap = options.caseSensitive ? index.byBasename : index.byBasenameLower;
636
+ const bnKey = options.caseSensitive ? target : target.toLowerCase();
637
+ const candidates = bnMap.get(bnKey);
638
+ if (!candidates || candidates.length === 0) return void 0;
639
+ if (candidates.length === 1) return candidates[0];
640
+ switch (options.onConflict) {
641
+ case "shortest": {
642
+ const sorted = sortByShortestPath(candidates);
643
+ return sorted[0];
644
+ }
645
+ case "first":
646
+ return candidates[0];
647
+ case "error":
648
+ return void 0;
649
+ }
650
+ }
651
+ function buildDeadUrl(rawTarget, options) {
652
+ const safe = encodeURIComponent(stripMarkdownExt(rawTarget).split("#")[0]);
653
+ return options.base + safe;
654
+ }
655
+ function defaultLabel(target, headingPart, entry, options) {
656
+ const lt = options.wikilinks.linkText;
657
+ let base;
658
+ if (typeof lt === "function") {
659
+ if (entry) {
660
+ base = lt(entry, target);
661
+ } else {
662
+ base = basename(target);
663
+ }
664
+ } else if (lt === "fullPath") {
665
+ base = entry ? entry.relativePath.replace(/\.(md|markdown)$/i, "") : target;
666
+ } else {
667
+ base = entry ? entry.basename : basename(target);
668
+ }
669
+ if (headingPart) {
670
+ return `${base} > ${headingPart}`;
671
+ }
672
+ return base;
673
+ }
674
+ function resolveAsset(rawTarget, index, options) {
675
+ const target = toPosix(rawTarget).trim();
676
+ if (target.includes("/")) {
677
+ return {
678
+ asset: index.assetsByRelativePath.get(target),
679
+ rawBasename: basename(target)
680
+ };
681
+ }
682
+ const bn = options.caseSensitive ? target : target.toLowerCase();
683
+ const map = options.caseSensitive ? index.assetsByBasename : index.assetsByBasenameLower;
684
+ const candidates = map.get(bn);
685
+ if (!candidates || candidates.length === 0) {
686
+ return { asset: void 0, rawBasename: target };
687
+ }
688
+ if (candidates.length === 1) {
689
+ return { asset: candidates[0], rawBasename: target };
690
+ }
691
+ switch (options.onConflict) {
692
+ case "shortest": {
693
+ const sorted = sortByShortestPath(candidates);
694
+ return { asset: sorted[0], rawBasename: target };
695
+ }
696
+ case "first":
697
+ return { asset: candidates[0], rawBasename: target };
698
+ case "error":
699
+ return { asset: void 0, rawBasename: target };
700
+ }
701
+ }
702
+
703
+ // src/modules/wikilinks/render.ts
704
+ function renderPageLink(state, result, label, env) {
705
+ const open = state.push("link_open", "a", 1);
706
+ const classes = ["wikilink"];
707
+ if (result.hasUnmatchedAnchor) classes.push("wikilink--unmatched-anchor");
708
+ const baseAttrs = {
709
+ href: result.url,
710
+ class: classes.join(" "),
711
+ "data-wikilink-target": result.target ? result.target.relativePath : ""
712
+ };
713
+ const extra = resolveExtraAttrs(env.options.wikilinks.htmlAttributes, {
714
+ originalHref: result.url,
715
+ label,
716
+ target: result.target,
717
+ isDead: result.isDead,
718
+ hasUnmatchedAnchor: result.hasUnmatchedAnchor
719
+ });
720
+ applyAttrs(open, baseAttrs, extra);
721
+ emitLabel(state, label, env);
722
+ state.push("link_close", "a", -1);
723
+ return true;
724
+ }
725
+ function renderDeadLink(state, url, label, rawTarget, env) {
726
+ const open = state.push("link_open", "a", 1);
727
+ const baseAttrs = {
728
+ href: url,
729
+ class: "wikilink wikilink--dead",
730
+ "data-wikilink-target": rawTarget,
731
+ title: `\u6B7B\u94FE:\u627E\u4E0D\u5230 [[${rawTarget}]]`
732
+ };
733
+ const extra = resolveExtraAttrs(env.options.wikilinks.htmlAttributes, {
734
+ originalHref: url,
735
+ label,
736
+ target: void 0,
737
+ isDead: true,
738
+ hasUnmatchedAnchor: false
739
+ });
740
+ applyAttrs(open, baseAttrs, extra);
741
+ emitLabel(state, label, env);
742
+ state.push("link_close", "a", -1);
743
+ return true;
744
+ }
745
+ function emitLabel(state, label, env) {
746
+ if (env.options.wikilinks.allowLinkLabelFormatting) {
747
+ const depth = env._labelDepth ?? 0;
748
+ if (depth < 3) {
749
+ ;
750
+ env._labelDepth = depth + 1;
751
+ const md = state.md;
752
+ const html = md.renderInline(label, env);
753
+ const token = state.push("html_inline", "", 0);
754
+ token.content = html;
755
+ env._labelDepth = depth;
756
+ return;
757
+ }
758
+ }
759
+ const t = state.push("text", "", 0);
760
+ t.content = label;
761
+ }
762
+ function resolveExtraAttrs(htmlAttrs, ctx) {
763
+ if (typeof htmlAttrs === "function") return htmlAttrs(ctx);
764
+ return htmlAttrs ?? {};
765
+ }
766
+ function applyAttrs(token, base, extra) {
767
+ const merged = { ...base };
768
+ for (const [k, v] of Object.entries(extra)) {
769
+ if (k === "class" && merged.class) {
770
+ merged.class = merged.class + " " + v;
771
+ } else {
772
+ merged[k] = v;
773
+ }
774
+ }
775
+ for (const [k, v] of Object.entries(merged)) {
776
+ token.attrSet(k, v);
777
+ }
778
+ }
779
+
780
+ // src/utils/escape.ts
781
+ var HTML_ESCAPE_MAP = {
782
+ "&": "&amp;",
783
+ "<": "&lt;",
784
+ ">": "&gt;",
785
+ '"': "&quot;",
786
+ "'": "&#39;"
787
+ };
788
+ var HTML_ESCAPE_RE = /[&<>"']/g;
789
+ function escapeHtml(s) {
790
+ return s.replace(HTML_ESCAPE_RE, (c) => HTML_ESCAPE_MAP[c]);
791
+ }
792
+
793
+ // src/core/asset-pipeline/build-emit.ts
794
+ var ASSET_PLACEHOLDER_PREFIX = "/__ayn_asset__/";
795
+ function buildPlaceholderUrl(asset, options) {
796
+ const id = encodeURI(asset.relativePath);
797
+ return options.base + ASSET_PLACEHOLDER_PREFIX.slice(1) + id;
798
+ }
799
+
800
+ // src/modules/embeds/image.ts
801
+ function renderImageHtml(rawTarget, aliasParts, env) {
802
+ const { index, options } = env;
803
+ const { altText, dim } = parseAltAndDim(aliasParts);
804
+ const processedTarget = options.embeds.postProcessImageTarget(rawTarget);
805
+ const { asset } = resolveAsset(processedTarget, index, options);
806
+ let src;
807
+ if (asset) {
808
+ asset.referencedBy.add(env.currentPath ?? "<unknown>");
809
+ env.referencedAssets?.add(asset);
810
+ src = buildPlaceholderUrl(asset, options) + options.embeds.uriSuffix;
811
+ } else {
812
+ src = options.base + encodeURIComponent(basename(processedTarget)) + options.embeds.uriSuffix;
813
+ }
814
+ const finalAlt = determineAlt(altText, processedTarget, options);
815
+ const attrs = {
816
+ src,
817
+ alt: finalAlt ?? ""
818
+ };
819
+ if (dim.width !== void 0) attrs.width = String(dim.width);
820
+ if (dim.height !== void 0) attrs.height = String(dim.height);
821
+ const extra = resolveExtra(options.embeds.htmlAttributes, {
822
+ originalHref: src,
823
+ altText: finalAlt,
824
+ dimensions: dim.raw,
825
+ embedType: "image"
826
+ });
827
+ for (const [k, v] of Object.entries(extra)) {
828
+ if (k === "class" && attrs.class) {
829
+ attrs.class = attrs.class + " " + v;
830
+ } else {
831
+ attrs[k] = v;
832
+ }
833
+ }
834
+ return "<img " + Object.entries(attrs).map(([k, v]) => `${escapeAttrName(k)}="${escapeHtml(v)}"`).join(" ") + " />";
835
+ }
836
+ function handleImageEmbed(state, rawTarget, aliasParts, env) {
837
+ const html = renderImageHtml(rawTarget, aliasParts, env);
838
+ const token = state.push("html_inline", "", 0);
839
+ token.content = html;
840
+ return true;
841
+ }
842
+ function parseAltAndDim(parts) {
843
+ if (parts.length === 0) return { altText: "", dim: { raw: "" } };
844
+ const last = parts[parts.length - 1];
845
+ const parsedLast = tryParseDimension(last);
846
+ if (parsedLast) {
847
+ const alt = parts.slice(0, -1).join("|").trim();
848
+ return { altText: alt, dim: { ...parsedLast, raw: last } };
849
+ }
850
+ return { altText: parts.join("|").trim(), dim: { raw: "" } };
851
+ }
852
+ function tryParseDimension(s) {
853
+ const trimmed = s.trim().toLowerCase();
854
+ if (!trimmed) return void 0;
855
+ if (trimmed.includes("x")) {
856
+ const [w, h] = trimmed.split("x");
857
+ const wOk = w === "" || /^\d+$/.test(w);
858
+ const hOk = h === "" || /^\d+$/.test(h);
859
+ if (!wOk || !hOk) return void 0;
860
+ if (w === "" && h === "") return void 0;
861
+ return {
862
+ width: w === "" ? void 0 : Number(w),
863
+ height: h === "" ? void 0 : Number(h)
864
+ };
865
+ }
866
+ if (/^\d+$/.test(trimmed)) {
867
+ return { width: Number(trimmed) };
868
+ }
869
+ return void 0;
870
+ }
871
+ function determineAlt(rawAlt, target, options) {
872
+ if (rawAlt && rawAlt !== "") {
873
+ return options.embeds.postProcessAltText(rawAlt);
874
+ }
875
+ const def = options.embeds.defaultAltText;
876
+ if (def === false) return void 0;
877
+ if (def === true) {
878
+ const bn = basename(target);
879
+ const dot = bn.lastIndexOf(".");
880
+ const noExt = dot > 0 ? bn.slice(0, dot) : bn;
881
+ return options.embeds.postProcessAltText(noExt);
882
+ }
883
+ if (typeof def === "string") {
884
+ return def === "" ? "" : options.embeds.postProcessAltText(def);
885
+ }
886
+ return options.embeds.postProcessAltText(basename(target));
887
+ }
888
+ function resolveExtra(attrs, ctx) {
889
+ if (typeof attrs === "function") return attrs(ctx);
890
+ return attrs ?? {};
891
+ }
892
+ function escapeAttrName(k) {
893
+ return k.replace(/[^a-zA-Z0-9_-]/g, "_");
894
+ }
895
+
896
+ // src/modules/embeds/transclusion.ts
897
+ function renderTransclusionHtml(md, rawTarget, aliasParts, env) {
898
+ const { index, options } = env;
899
+ const result = resolveWikilink(rawTarget, index, options, "transclusion");
900
+ if (result.isDead || !result.target) {
901
+ return `<div class="transclusion transclusion--dead" data-target="${escapeHtml(
902
+ rawTarget
903
+ )}">\u26A0\uFE0F \u627E\u4E0D\u5230\u7B14\u8BB0 <code>${escapeHtml(rawTarget)}</code></div>`;
904
+ }
905
+ const target = result.target;
906
+ const arr = index.backlinks.get(target.absolutePath) ?? [];
907
+ if (env.currentPath) {
908
+ arr.push({
909
+ fromPath: env.currentPath,
910
+ fromUrl: index.files.get(env.currentPath)?.url ?? "",
911
+ context: "",
912
+ isEmbed: true,
913
+ line: -1
914
+ });
915
+ }
916
+ index.backlinks.set(target.absolutePath, arr);
917
+ const stack = env.transclusionStack ?? [];
918
+ if (stack.includes(target.absolutePath)) {
919
+ return `<div class="transclusion transclusion--cycle" data-target="${escapeHtml(
920
+ rawTarget
921
+ )}">\u26A0\uFE0F \u5FAA\u73AF\u5F15\u7528:<code>${escapeHtml(
922
+ rawTarget
923
+ )}</code> \u5DF2\u5728 transclusion \u94FE\u4E0A</div>`;
924
+ }
925
+ const depth = env.transclusionDepth ?? 0;
926
+ if (depth >= options.embeds.transclusionMaxDepth) {
927
+ return `<div class="transclusion transclusion--too-deep" data-target="${escapeHtml(
928
+ rawTarget
929
+ )}">\u26A0\uFE0F transclusion \u5D4C\u5957\u8FC7\u6DF1(&gt; ${options.embeds.transclusionMaxDepth})</div>`;
930
+ }
931
+ const headingPart = extractHeading(rawTarget);
932
+ const fragment = headingPart ? sliceByHeading(target, headingPart, options.slugify) : target.content;
933
+ if (fragment == null) {
934
+ return `<div class="transclusion transclusion--unmatched-anchor" data-target="${escapeHtml(
935
+ rawTarget
936
+ )}">\u26A0\uFE0F \u627E\u4E0D\u5230\u7AE0\u8282 <code>#${escapeHtml(
937
+ headingPart
938
+ )}</code></div>`;
939
+ }
940
+ const cacheKey = `${target.absolutePath}::${headingPart ?? ""}`;
941
+ const cache = getCache(env);
942
+ let inner;
943
+ const cached = cache.get(cacheKey);
944
+ if (cached) {
945
+ inner = cached.html;
946
+ } else {
947
+ const childEnv = {
948
+ index: env.index,
949
+ options: env.options,
950
+ currentPath: target.absolutePath,
951
+ transclusionStack: [...stack, target.absolutePath],
952
+ transclusionDepth: depth + 1,
953
+ // 跨递归共享:asset 引用集合(build 时要 emit 所有被引用 asset)
954
+ referencedAssets: env.referencedAssets
955
+ };
956
+ childEnv._transclusionCache = env._transclusionCache;
957
+ inner = md.render(fragment, childEnv);
958
+ cache.set(cacheKey, { html: inner });
959
+ }
960
+ const sourceUrl = headingPart ? `${target.url}#${escapeHtml(options.slugify(headingPart))}` : target.url;
961
+ const aliasData = aliasParts.length ? ` data-caption="${escapeHtml(aliasParts.join("|"))}"` : "";
962
+ return `<div class="transclusion" data-source="${escapeHtml(
963
+ target.relativePath
964
+ )}" data-source-url="${escapeHtml(sourceUrl)}"${aliasData}>${inner}</div>`;
965
+ }
966
+ function handleTransclusion(state, rawTarget, aliasParts, env) {
967
+ if (env.options.deadLink !== "silent") {
968
+ console.warn(
969
+ `vitepress-allyouneed: ![[${rawTarget}]] \u5728\u6BB5\u843D\u4E2D\u65E0\u6CD5 transclude(\u4F1A\u4EA7\u751F\u4E0D\u5408\u6CD5 HTML),\u5DF2\u964D\u7EA7\u4E3A\u94FE\u63A5\u3002\u8BF7\u5355\u72EC\u653E\u4E00\u884C\u3002`
970
+ );
971
+ }
972
+ const { index, options } = env;
973
+ const result = resolveWikilink(rawTarget, index, options, "page");
974
+ const url = result.url;
975
+ const label = aliasParts.length ? aliasParts.join("|").trim() : result.defaultLabel;
976
+ const html = `<a class="wikilink wikilink--inline-transclusion-degraded" href="${escapeHtml(url)}" data-wikilink-target="${escapeHtml(rawTarget)}" title="\u884C\u5185 transclusion \u5DF2\u964D\u7EA7,\u89C1\u63A7\u5236\u53F0">${escapeHtml(label)}</a>`;
977
+ const token = state.push("html_inline", "", 0);
978
+ token.content = html;
979
+ return true;
980
+ }
981
+ function extractHeading(raw) {
982
+ const hashIdx = raw.indexOf("#");
983
+ if (hashIdx < 0) return "";
984
+ return raw.slice(hashIdx + 1).trim();
985
+ }
986
+ function sliceByHeading(target, headingPart, slugify) {
987
+ const matched = target.headings.find(
988
+ (h) => h.text === headingPart || h.slug === headingPart || h.slug === slugify(headingPart)
989
+ );
990
+ if (!matched) return void 0;
991
+ const lines = target.content.split(/\r?\n/);
992
+ const startLine = matched.line + 1;
993
+ let endLine = lines.length;
994
+ for (let i = startLine; i < lines.length; i++) {
995
+ const l = lines[i];
996
+ const m = l.match(/^(#{1,6})\s+/);
997
+ if (m && m[1].length <= matched.level) {
998
+ endLine = i;
999
+ break;
1000
+ }
1001
+ }
1002
+ return lines.slice(startLine, endLine).join("\n").trim();
1003
+ }
1004
+ function getCache(env) {
1005
+ const e = env;
1006
+ if (!e._transclusionCache) e._transclusionCache = /* @__PURE__ */ new Map();
1007
+ return e._transclusionCache;
1008
+ }
1009
+
1010
+ // src/modules/wikilinks/rule.ts
1011
+ function makeWikilinkRule(scope) {
1012
+ return function wikilinkRule(state, silent) {
1013
+ const src = state.src;
1014
+ const start = state.pos;
1015
+ const max = state.posMax;
1016
+ let isEmbed = false;
1017
+ let inner;
1018
+ if (src.charCodeAt(start) === 33 && src.charCodeAt(start + 1) === 91 && src.charCodeAt(start + 2) === 91) {
1019
+ if (scope === "wikilinks-only") return false;
1020
+ isEmbed = true;
1021
+ } else if (src.charCodeAt(start) === 91 && src.charCodeAt(start + 1) === 91) {
1022
+ if (scope === "embeds-only") return false;
1023
+ } else {
1024
+ return false;
1025
+ }
1026
+ const innerStart = start + (isEmbed ? 3 : 2);
1027
+ const closeIdx = src.indexOf("]]", innerStart);
1028
+ if (closeIdx < 0 || closeIdx >= max) return false;
1029
+ inner = src.slice(innerStart, closeIdx);
1030
+ if (inner.includes("\n")) return false;
1031
+ if (!inner.trim()) return false;
1032
+ if (silent) return true;
1033
+ const env = state.env;
1034
+ if (!env || !env.index || !env.options) {
1035
+ return false;
1036
+ }
1037
+ state.pos = closeIdx + 2;
1038
+ const parts = inner.split("|").map((p) => p.trim());
1039
+ const rawTarget = parts[0];
1040
+ const aliasParts = parts.slice(1);
1041
+ if (isEmbed) {
1042
+ const ext = extractExt(rawTarget);
1043
+ const isImage = ext && env.options.embeds.imageFileExt.includes(ext.toLowerCase());
1044
+ if (isImage) {
1045
+ return handleImageEmbed(state, rawTarget, aliasParts, env);
1046
+ }
1047
+ return handleTransclusion(state, rawTarget, aliasParts, env);
1048
+ }
1049
+ return emitPageLink(state, rawTarget, aliasParts, env);
1050
+ };
1051
+ }
1052
+ function extractExt(target) {
1053
+ const cleaned = target.split("#")[0];
1054
+ const dot = cleaned.lastIndexOf(".");
1055
+ if (dot <= 0) return "";
1056
+ return cleaned.slice(dot + 1).toLowerCase();
1057
+ }
1058
+ function emitPageLink(state, rawTarget, aliasParts, env) {
1059
+ const { index, options } = env;
1060
+ const userAlias = aliasParts.length > 0 ? aliasParts.join("|").trim() : "";
1061
+ const processed = options.wikilinks.postProcessLinkTarget(rawTarget);
1062
+ const result = resolveWikilink(processed, index, options, "page");
1063
+ const label = userAlias ? options.wikilinks.postProcessLinkLabel(userAlias) : result.defaultLabel;
1064
+ registerBacklink(env, result.target?.absolutePath, false);
1065
+ if (result.isDead) {
1066
+ handleDeadLink(env, rawTarget);
1067
+ return renderDeadLink(state, result.url, label, rawTarget, env);
1068
+ }
1069
+ return renderPageLink(state, result, label, env);
1070
+ }
1071
+ function registerBacklink(env, targetPath, isEmbed) {
1072
+ if (!targetPath || !env.currentPath) return;
1073
+ const arr = env.index.backlinks.get(targetPath) ?? [];
1074
+ arr.push({
1075
+ fromPath: env.currentPath,
1076
+ fromUrl: env.index.files.get(env.currentPath)?.url ?? "",
1077
+ context: "",
1078
+ isEmbed,
1079
+ line: -1
1080
+ });
1081
+ env.index.backlinks.set(targetPath, arr);
1082
+ }
1083
+ function handleDeadLink(env, rawTarget) {
1084
+ const { options } = env;
1085
+ const msg = `vitepress-allyouneed: \u6B7B\u94FE [[${rawTarget}]]${env.currentPath ? ` (in ${env.currentPath})` : ""}`;
1086
+ if (options.deadLink === "silent") return;
1087
+ if (options.deadLink === "warn") {
1088
+ console.warn(msg);
1089
+ return;
1090
+ }
1091
+ env.index.warnings.push({
1092
+ kind: "unknown",
1093
+ message: msg,
1094
+ affected: env.currentPath ? [env.currentPath] : []
1095
+ });
1096
+ }
1097
+
1098
+ // src/modules/wikilinks/index.ts
1099
+ function registerWikilinks(md, scope) {
1100
+ md.inline.ruler.before(
1101
+ "link",
1102
+ "allyouneed_wikilinks",
1103
+ makeWikilinkRule(scope)
1104
+ );
1105
+ }
1106
+
1107
+ // src/modules/embeds/block-rule.ts
1108
+ var LINE_RE = /^!\[\[([^\n\]]+)\]\]\s*$/;
1109
+ function registerEmbedBlockRule(md) {
1110
+ md.block.ruler.before(
1111
+ "paragraph",
1112
+ "allyouneed_embed_block",
1113
+ makeRule(md),
1114
+ { alt: ["paragraph"] }
1115
+ );
1116
+ }
1117
+ function makeRule(md) {
1118
+ return function embedBlockRule(state, startLine, _endLine, silent) {
1119
+ const start = state.bMarks[startLine] + state.tShift[startLine];
1120
+ const max = state.eMarks[startLine];
1121
+ const lineText = state.src.slice(start, max);
1122
+ const m = lineText.match(LINE_RE);
1123
+ if (!m) return false;
1124
+ if (silent) return true;
1125
+ const env = state.env;
1126
+ if (!env || !env.index || !env.options) return false;
1127
+ const inner = m[1];
1128
+ const parts = inner.split("|").map((p) => p.trim());
1129
+ const rawTarget = parts[0];
1130
+ const aliasParts = parts.slice(1);
1131
+ const ext = extractExt2(rawTarget);
1132
+ const isImage = !!ext && env.options.embeds.imageFileExt.includes(ext.toLowerCase());
1133
+ let html;
1134
+ if (isImage) {
1135
+ html = renderImageHtml(rawTarget, aliasParts, env);
1136
+ } else {
1137
+ html = renderTransclusionHtml(md, rawTarget, aliasParts, env);
1138
+ }
1139
+ const token = state.push("html_block", "", 0);
1140
+ token.content = html + "\n";
1141
+ token.map = [startLine, startLine + 1];
1142
+ state.line = startLine + 1;
1143
+ return true;
1144
+ };
1145
+ }
1146
+ function extractExt2(target) {
1147
+ const cleaned = target.split("#")[0];
1148
+ const dot = cleaned.lastIndexOf(".");
1149
+ if (dot <= 0) return "";
1150
+ return cleaned.slice(dot + 1).toLowerCase();
1151
+ }
1152
+
1153
+ // src/modules/embeds/index.ts
1154
+ function registerEmbedsOnly(md) {
1155
+ md.inline.ruler.before(
1156
+ "link",
1157
+ "allyouneed_embeds",
1158
+ makeWikilinkRule("embeds-only")
1159
+ );
1160
+ registerEmbedBlockRule(md);
1161
+ }
1162
+
1163
+ // src/markdown-it.ts
1164
+ function allYouNeedMarkdownIt(md, options) {
1165
+ const resolved = options && isResolved(options) ? options : resolveOptions(options);
1166
+ const { wikilinks: wlOn, embeds: emOn } = resolved.modules;
1167
+ if (wlOn && emOn) {
1168
+ registerWikilinks(md, "both");
1169
+ registerEmbedBlockRule(md);
1170
+ } else if (wlOn) {
1171
+ registerWikilinks(md, "wikilinks-only");
1172
+ } else if (emOn) {
1173
+ registerEmbedsOnly(md);
1174
+ }
1175
+ }
1176
+ function isResolved(o) {
1177
+ return typeof o.slugify === "function" && typeof o.srcDir === "string" && typeof o.modules === "object" && o.modules !== null;
1178
+ }
1179
+ var markdown_it_default = allYouNeedMarkdownIt;
1180
+
1181
+ // src/vite.ts
1182
+ import fs5 from "fs";
1183
+ import nodePath6 from "path";
1184
+
1185
+ // src/core/asset-pipeline/dev-middleware.ts
1186
+ import fs4 from "fs";
1187
+ import nodePath5 from "path";
1188
+ function createDevMiddleware(index, options) {
1189
+ return (req, res, next) => {
1190
+ if (!req.url) return next();
1191
+ const cleanPath = req.url.split("?")[0].split("#")[0];
1192
+ let inSiteUrl = cleanPath;
1193
+ if (options.base !== "/" && cleanPath.startsWith(options.base)) {
1194
+ inSiteUrl = "/" + cleanPath.slice(options.base.length);
1195
+ }
1196
+ let decoded;
1197
+ try {
1198
+ decoded = decodeURIComponent(inSiteUrl);
1199
+ } catch {
1200
+ return next();
1201
+ }
1202
+ if (!/\.[a-zA-Z0-9]+$/.test(decoded)) return next();
1203
+ const relCandidate = decoded.replace(/^\/+/, "");
1204
+ let asset = index.assetsByRelativePath.get(relCandidate);
1205
+ if (!asset) {
1206
+ const bn = basename(decoded);
1207
+ const map = options.caseSensitive ? index.assetsByBasename : index.assetsByBasenameLower;
1208
+ const key = options.caseSensitive ? bn : bn.toLowerCase();
1209
+ const candidates = map.get(key);
1210
+ if (candidates && candidates.length > 0) asset = candidates[0];
1211
+ }
1212
+ if (!asset) return next();
1213
+ let stat;
1214
+ try {
1215
+ stat = fs4.statSync(asset.absolutePath);
1216
+ } catch {
1217
+ return next();
1218
+ }
1219
+ res.statusCode = 200;
1220
+ res.setHeader("Content-Type", guessMime(asset.extension));
1221
+ res.setHeader("Content-Length", String(stat.size));
1222
+ res.setHeader("Cache-Control", "no-cache");
1223
+ fs4.createReadStream(asset.absolutePath).on("error", () => next()).pipe(res);
1224
+ };
1225
+ }
1226
+ var MIME = {
1227
+ png: "image/png",
1228
+ jpg: "image/jpeg",
1229
+ jpeg: "image/jpeg",
1230
+ gif: "image/gif",
1231
+ webp: "image/webp",
1232
+ svg: "image/svg+xml",
1233
+ bmp: "image/bmp",
1234
+ avif: "image/avif",
1235
+ ico: "image/x-icon",
1236
+ mp4: "video/mp4",
1237
+ webm: "video/webm",
1238
+ mov: "video/quicktime",
1239
+ m4v: "video/x-m4v",
1240
+ mp3: "audio/mpeg",
1241
+ wav: "audio/wav",
1242
+ ogg: "audio/ogg",
1243
+ m4a: "audio/mp4",
1244
+ flac: "audio/flac",
1245
+ pdf: "application/pdf",
1246
+ canvas: "application/json",
1247
+ excalidraw: "application/json"
1248
+ };
1249
+ function guessMime(ext) {
1250
+ return MIME[ext.toLowerCase()] ?? "application/octet-stream";
1251
+ }
1252
+
1253
+ // src/vite.ts
1254
+ function stripQueryAndHash(id) {
1255
+ return id.split("?")[0].split("#")[0];
1256
+ }
1257
+ function viteAllYouNeed(userOptions = {}) {
1258
+ let resolved;
1259
+ let index;
1260
+ let viteConfig;
1261
+ const plugin = {
1262
+ name: "vitepress-allyouneed",
1263
+ /**
1264
+ * 在 Vite 配置定型前扩 server.fs.allow,让我们的 vault srcDir 可被 Vite
1265
+ * 服务(即便 srcDir 不在项目根下)。
1266
+ */
1267
+ config(_userViteConfig, _envCtx) {
1268
+ const srcDirOpt = userOptions.srcDir;
1269
+ if (!srcDirOpt) return void 0;
1270
+ const abs = toPosix(nodePath6.resolve(srcDirOpt));
1271
+ return {
1272
+ server: {
1273
+ fs: {
1274
+ allow: [abs]
1275
+ }
1276
+ }
1277
+ };
1278
+ },
1279
+ configResolved(cfg) {
1280
+ viteConfig = cfg;
1281
+ resolved = resolveOptions(userOptions, {
1282
+ srcDir: userOptions.srcDir ?? cfg.root,
1283
+ base: userOptions.base ?? cfg.base,
1284
+ cleanUrls: userOptions.cleanUrls
1285
+ });
1286
+ try {
1287
+ index = scanVault(resolved);
1288
+ if (index.warnings.length > 0) {
1289
+ const top = index.warnings.slice(0, 10);
1290
+ for (const w of top) {
1291
+ cfg.logger.warn(`[vitepress-allyouneed] ${w.message}`);
1292
+ }
1293
+ if (index.warnings.length > top.length) {
1294
+ cfg.logger.warn(
1295
+ `[vitepress-allyouneed] (...\u8FD8\u6709 ${index.warnings.length - top.length} \u6761\u544A\u8B66)`
1296
+ );
1297
+ }
1298
+ }
1299
+ } catch (err) {
1300
+ cfg.logger.error(
1301
+ `[vitepress-allyouneed] vault \u626B\u63CF\u5931\u8D25: ${err instanceof Error ? err.message : String(err)}`
1302
+ );
1303
+ index = void 0;
1304
+ }
1305
+ },
1306
+ configureServer(server) {
1307
+ if (!index) return;
1308
+ const mw = createDevMiddleware(index, resolved);
1309
+ return () => {
1310
+ server.middlewares.use(mw);
1311
+ };
1312
+ },
1313
+ handleHotUpdate(ctx) {
1314
+ if (!index) return;
1315
+ try {
1316
+ const stat = fs5.statSync(ctx.file);
1317
+ if (stat.isFile()) {
1318
+ updateFile(index, ctx.file, resolved);
1319
+ } else if (!fs5.existsSync(ctx.file)) {
1320
+ removeFile(index, ctx.file, resolved);
1321
+ }
1322
+ } catch {
1323
+ removeFile(index, ctx.file, resolved);
1324
+ }
1325
+ },
1326
+ /**
1327
+ * 拦截占位符 URL,**返回真实绝对文件路径**(POSIX 风格)。
1328
+ *
1329
+ * Vite 拿到文件路径会:
1330
+ * - dev:按文件系统服务,id 经 transform 后变成 `export default '<url>'`
1331
+ * - build:emit asset、Rollup 自动加 hash,导出最终 URL
1332
+ */
1333
+ resolveId(id) {
1334
+ if (!index) return null;
1335
+ const stripped = stripQueryAndHash(id);
1336
+ const ph = ASSET_PLACEHOLDER_PREFIX;
1337
+ const phIdx = stripped.indexOf(ph);
1338
+ if (phIdx < 0) return null;
1339
+ const encoded = stripped.slice(phIdx + ph.length);
1340
+ let relPath;
1341
+ try {
1342
+ relPath = decodeURI(encoded);
1343
+ } catch {
1344
+ return null;
1345
+ }
1346
+ const asset = index.assetsByRelativePath.get(relPath);
1347
+ if (!asset) return null;
1348
+ const query = id.slice(stripped.length);
1349
+ return asset.absolutePath + query;
1350
+ },
1351
+ /**
1352
+ * 给 vitepress.ts wrapper 用。
1353
+ */
1354
+ __getOptions() {
1355
+ return resolved;
1356
+ },
1357
+ __getIndex() {
1358
+ return index;
1359
+ }
1360
+ };
1361
+ void viteConfig;
1362
+ return plugin;
1363
+ }
1364
+
1365
+ // src/vitepress.ts
1366
+ function defineConfigWithAllYouNeed(config, pluginOptions = {}) {
1367
+ const vpExclude = Array.isArray(config.srcExclude) ? config.srcExclude : [];
1368
+ const mergedOptions = {
1369
+ ...pluginOptions,
1370
+ srcDir: pluginOptions.srcDir ?? config.srcDir,
1371
+ base: pluginOptions.base ?? config.base,
1372
+ cleanUrls: pluginOptions.cleanUrls ?? config.cleanUrls,
1373
+ scan: {
1374
+ ...pluginOptions.scan,
1375
+ exclude: [
1376
+ ...pluginOptions.scan?.exclude ?? [],
1377
+ ...vpExclude
1378
+ ]
1379
+ }
1380
+ };
1381
+ const vitePlugin = viteAllYouNeed(mergedOptions);
1382
+ const existingVite = typeof config.vite === "object" && config.vite !== null ? config.vite : {};
1383
+ const existingPlugins = Array.isArray(existingVite.plugins) ? existingVite.plugins : [];
1384
+ const newVite = {
1385
+ ...existingVite,
1386
+ plugins: [...existingPlugins, vitePlugin]
1387
+ };
1388
+ const existingMarkdown = config.markdown ?? {};
1389
+ const existingConfig = existingMarkdown.config;
1390
+ const newMarkdownConfig = (md) => {
1391
+ markdown_it_default(md, mergedOptions);
1392
+ md.core.ruler.before(
1393
+ "normalize",
1394
+ "allyouneed_env_inject",
1395
+ makeEnvInjector(vitePlugin)
1396
+ );
1397
+ if (typeof existingConfig === "function") {
1398
+ existingConfig(md);
1399
+ }
1400
+ };
1401
+ return {
1402
+ ...config,
1403
+ vite: newVite,
1404
+ markdown: {
1405
+ ...existingMarkdown,
1406
+ config: newMarkdownConfig
1407
+ }
1408
+ };
1409
+ }
1410
+ function makeEnvInjector(vitePlugin) {
1411
+ return (state) => {
1412
+ const env = state.env;
1413
+ if (!env) return;
1414
+ if (env.index && env.options) return;
1415
+ const index = vitePlugin.__getIndex();
1416
+ const options = vitePlugin.__getOptions();
1417
+ if (!index || !options) return;
1418
+ env.index = index;
1419
+ env.options = options;
1420
+ if (!env.currentPath) {
1421
+ env.currentPath = typeof env.realPath === "string" ? env.realPath : typeof env.path === "string" ? env.path : void 0;
1422
+ }
1423
+ if (!env.referencedAssets) env.referencedAssets = /* @__PURE__ */ new Set();
1424
+ };
1425
+ }
1426
+ export {
1427
+ allYouNeedMarkdownIt,
1428
+ createEmptyIndex,
1429
+ markdown_it_default as default,
1430
+ defaultSlugify,
1431
+ defineConfigWithAllYouNeed,
1432
+ extractCustomId,
1433
+ removeFile,
1434
+ resolveAsset,
1435
+ resolveOptions,
1436
+ resolveWikilink,
1437
+ scanVault,
1438
+ updateFile,
1439
+ viteAllYouNeed
1440
+ };
1441
+ //# sourceMappingURL=index.js.map