obsidian-dev-utils 5.3.1 → 6.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.
Files changed (37) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/lib/Async.cjs +2 -1
  3. package/dist/lib/Async.d.ts +4 -0
  4. package/dist/lib/String.cjs +3 -3
  5. package/dist/lib/String.d.ts +2 -2
  6. package/dist/lib/obsidian/AttachmentPath.cjs +3 -3
  7. package/dist/lib/obsidian/AttachmentPath.d.ts +3 -3
  8. package/dist/lib/obsidian/Backlink.cjs +3 -3
  9. package/dist/lib/obsidian/FileSystem.cjs +17 -17
  10. package/dist/lib/obsidian/FileSystem.d.ts +14 -14
  11. package/dist/lib/obsidian/Link.cjs +77 -74
  12. package/dist/lib/obsidian/Link.d.ts +61 -65
  13. package/dist/lib/obsidian/Loop.cjs +2 -2
  14. package/dist/lib/obsidian/Loop.d.ts +4 -4
  15. package/dist/lib/obsidian/MetadataCache.cjs +1 -16
  16. package/dist/lib/obsidian/MetadataCache.d.ts +0 -9
  17. package/dist/lib/obsidian/Plugin/ValueComponent.cjs +4 -4
  18. package/dist/lib/obsidian/Plugin/ValueComponent.d.ts +4 -4
  19. package/dist/lib/obsidian/RenameDeleteHandler.cjs +17 -19
  20. package/dist/lib/scripts/ESLint/ESLint.cjs +7 -7
  21. package/dist/lib/scripts/ESLint/ESLint.d.ts +2 -2
  22. package/dist/lib/scripts/ESLint/eslint.config.cjs +3 -2
  23. package/dist/lib/scripts/Exec.cjs +5 -5
  24. package/dist/lib/scripts/Exec.d.ts +11 -7
  25. package/dist/lib/scripts/JSON.cjs +3 -3
  26. package/dist/lib/scripts/JSON.d.ts +1 -1
  27. package/dist/lib/scripts/NodeModules.cjs +16 -1
  28. package/dist/lib/scripts/NodeModules.d.ts +5 -4
  29. package/dist/lib/scripts/Npm.cjs +28 -27
  30. package/dist/lib/scripts/Npm.d.ts +24 -56
  31. package/dist/lib/scripts/Root.cjs +20 -7
  32. package/dist/lib/scripts/Root.d.ts +3 -3
  33. package/dist/lib/scripts/cli.cjs +3 -3
  34. package/dist/lib/scripts/esbuild/Dependency.cjs +3 -3
  35. package/dist/lib/scripts/esbuild/ObsidianPluginBuilder.cjs +11 -4
  36. package/dist/lib/scripts/version.cjs +33 -30
  37. package/package.json +13 -13
@@ -74,16 +74,18 @@ var __process = globalThis["process"] ?? {
74
74
  const SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\x00\x08\x0B\x0C\x0E-\x1F ]/g;
75
75
  const SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\[\]<>_*~=`$]/g;
76
76
  function convertLink(options) {
77
- const pathOrFile = options.oldPathOrFile ? extractLinkFile(options.app, options.link, options.oldPathOrFile) : null;
77
+ const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);
78
+ if (!targetFile) {
79
+ return options.link.original;
80
+ }
78
81
  return updateLink({
79
82
  app: options.app,
80
- forceMarkdownLinks: options.forceMarkdownLinks,
81
83
  link: options.link,
82
- oldPathOrFile: pathOrFile ?? void 0,
83
- pathOrFile,
84
- renameMap: options.renameMap,
85
- shouldUpdateFilenameAlias: options.shouldUpdateFilenameAlias,
86
- sourcePathOrFile: options.sourcePathOrFile
84
+ newSourcePathOrFile: options.newSourcePathOrFile,
85
+ newTargetPathOrFile: targetFile,
86
+ oldSourcePathOrFile: options.oldSourcePathOrFile,
87
+ shouldForceMarkdownLinks: options.shouldForceMarkdownLinks,
88
+ shouldUpdateFilenameAlias: options.shouldUpdateFilenameAlias
87
89
  });
88
90
  }
89
91
  async function editBacklinks(app, pathOrFile, linkConverter, retryOptions = {}) {
@@ -117,43 +119,43 @@ async function editLinks(app, pathOrFile, linkConverter, retryOptions = {}) {
117
119
  return changes;
118
120
  }, retryOptions);
119
121
  }
120
- function extractLinkFile(app, link, notePathOrFile) {
122
+ function extractLinkFile(app, link, sourcePathOrFile) {
121
123
  const { linkPath } = splitSubpath(link.link);
122
- return app.metadataCache.getFirstLinkpathDest(linkPath, (0, import_FileSystem.getPath)(notePathOrFile));
124
+ return app.metadataCache.getFirstLinkpathDest(linkPath, (0, import_FileSystem.getPath)(sourcePathOrFile));
123
125
  }
124
126
  function generateMarkdownLink(options) {
125
127
  const { app } = options;
126
128
  const configurableDefaultOptionsFn = app.fileManager.generateMarkdownLink.defaultOptionsFn ?? (() => ({}));
127
129
  const configurableDefaultOptions = configurableDefaultOptionsFn();
128
130
  const DEFAULT_OPTIONS = {
129
- allowEmptyEmbedAlias: true
131
+ isEmptyEmbedAliasAllowed: true
130
132
  };
131
133
  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };
132
- const file = (0, import_FileSystem.getFile)(app, options.pathOrFile, options.allowNonExistingFile);
133
- return (0, import_MetadataCache.tempRegisterFileAndRun)(app, file, () => {
134
+ const targetFile = (0, import_FileSystem.getFile)(app, options.targetPathOrFile, options.isNonExistingFileAllowed);
135
+ return (0, import_MetadataCache.tempRegisterFileAndRun)(app, targetFile, () => {
134
136
  const sourcePath = (0, import_FileSystem.getPath)(options.sourcePathOrFile);
135
137
  const subpath = options.subpath ?? "";
136
138
  let alias = options.alias ?? "";
137
- const isEmbed = options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : void 0) ?? !(0, import_FileSystem.isMarkdownFile)(file);
139
+ const isEmbed = options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : void 0) ?? !(0, import_FileSystem.isMarkdownFile)(targetFile);
138
140
  const isWikilink = options.isWikilink ?? (options.originalLink ? testWikilink(options.originalLink) : void 0) ?? (0, import_ObsidianSettings.shouldUseWikilinks)(app);
139
- const forceRelativePath = options.forceRelativePath ?? (0, import_ObsidianSettings.shouldUseRelativeLinks)(app);
140
- const useLeadingDot = options.useLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : void 0) ?? false;
141
- const useAngleBrackets = options.useAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : void 0) ?? false;
142
- let linkText = file.path === sourcePath && subpath ? subpath : forceRelativePath ? (0, import_Path.relative)((0, import_Path.dirname)(sourcePath), isWikilink ? (0, import_FileSystem.trimMarkdownExtension)(file) : file.path) + subpath : app.metadataCache.fileToLinktext(file, sourcePath, isWikilink) + subpath;
143
- if (forceRelativePath && useLeadingDot && !linkText.startsWith(".") && !linkText.startsWith("#")) {
141
+ const shouldForceRelativePath = options.shouldForceRelativePath ?? (0, import_ObsidianSettings.shouldUseRelativeLinks)(app);
142
+ const shouldUseLeadingDot = options.shouldUseLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : void 0) ?? false;
143
+ const shouldUseAngleBrackets = options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : void 0) ?? false;
144
+ let linkText = targetFile.path === sourcePath && subpath ? subpath : shouldForceRelativePath ? (0, import_Path.relative)((0, import_Path.dirname)(sourcePath), isWikilink ? (0, import_FileSystem.trimMarkdownExtension)(targetFile) : targetFile.path) + subpath : app.metadataCache.fileToLinktext(targetFile, sourcePath, isWikilink) + subpath;
145
+ if (shouldForceRelativePath && shouldUseLeadingDot && !linkText.startsWith(".") && !linkText.startsWith("#")) {
144
146
  linkText = "./" + linkText;
145
147
  }
146
148
  const embedPrefix = isEmbed ? "!" : "";
147
149
  if (!isWikilink) {
148
- if (useAngleBrackets) {
150
+ if (shouldUseAngleBrackets) {
149
151
  linkText = `<${linkText}>`;
150
152
  } else {
151
153
  linkText = linkText.replace(SPECIAL_LINK_SYMBOLS_REGEXP, function(specialLinkSymbol) {
152
154
  return encodeURIComponent(specialLinkSymbol);
153
155
  });
154
156
  }
155
- if (!alias && (!isEmbed || !options.allowEmptyEmbedAlias)) {
156
- alias = !options.includeAttachmentExtensionToEmbedAlias || (0, import_FileSystem.isMarkdownFile)(file) ? file.basename : file.name;
157
+ if (!alias && (!isEmbed || !options.isEmptyEmbedAliasAllowed)) {
158
+ alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || (0, import_FileSystem.isMarkdownFile)(targetFile) ? targetFile.basename : targetFile.name;
157
159
  }
158
160
  alias = alias.replace(SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, "\\$&");
159
161
  return `${embedPrefix}[${alias}](${linkText})`;
@@ -249,30 +251,36 @@ function shouldResetAlias(options) {
249
251
  app,
250
252
  displayText,
251
253
  isWikilink,
252
- otherPathOrFiles,
253
- pathOrFile,
254
- sourcePathOrFile
254
+ newSourcePathOrFile,
255
+ oldSourcePathOrFile,
256
+ oldTargetPath,
257
+ targetPathOrFile
255
258
  } = options;
256
259
  if (isWikilink === false) {
257
260
  return false;
258
261
  }
259
- const file = (0, import_FileSystem.getFile)(app, pathOrFile);
260
262
  if (!displayText) {
261
263
  return true;
262
264
  }
263
- const sourcePath = (0, import_FileSystem.getPath)(sourcePathOrFile);
264
- const sourceDir = (0, import_Path.dirname)(sourcePath);
265
+ const targetFile = (0, import_FileSystem.getFile)(app, targetPathOrFile);
266
+ const newSourcePath = (0, import_FileSystem.getPath)(newSourcePathOrFile);
267
+ const oldSourcePath = (0, import_FileSystem.getPath)(oldSourcePathOrFile ?? newSourcePathOrFile);
268
+ const newSourceDir = (0, import_Path.dirname)(newSourcePath);
269
+ const oldSourceDir = (0, import_Path.dirname)(oldSourcePath);
265
270
  const aliasesToReset = /* @__PURE__ */ new Set();
266
- for (const pathOrFile2 of [file.path, ...otherPathOrFiles]) {
267
- if (!pathOrFile2) {
271
+ for (const pathOrFile of [targetFile.path, oldTargetPath]) {
272
+ if (!pathOrFile) {
268
273
  continue;
269
274
  }
270
- const path = (0, import_FileSystem.getPath)(pathOrFile2);
275
+ const path = (0, import_FileSystem.getPath)(pathOrFile);
271
276
  aliasesToReset.add(path);
272
277
  aliasesToReset.add((0, import_Path.basename)(path));
273
- aliasesToReset.add((0, import_Path.relative)(sourceDir, path));
278
+ aliasesToReset.add((0, import_Path.relative)(newSourceDir, path));
279
+ aliasesToReset.add((0, import_Path.relative)(oldSourceDir, path));
280
+ }
281
+ for (const sourcePath of [oldSourcePath, newSourcePath]) {
282
+ aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));
274
283
  }
275
- aliasesToReset.add(app.metadataCache.fileToLinktext(file, sourcePath, false));
276
284
  const cleanDisplayText = (0, import_obsidian.normalizePath)(displayText.split(" > ")[0] ?? "").replace(/^\.\//, "").toLowerCase();
277
285
  for (const alias of aliasesToReset) {
278
286
  if (alias.toLowerCase() === cleanDisplayText) {
@@ -312,80 +320,75 @@ function testWikilink(link) {
312
320
  function updateLink(options) {
313
321
  const {
314
322
  app,
315
- forceMarkdownLinks,
316
323
  link,
317
- oldPathOrFile,
318
- pathOrFile,
319
- renameMap,
320
- shouldUpdateFilenameAlias,
321
- sourcePathOrFile
324
+ newSourcePathOrFile,
325
+ newTargetPathOrFile,
326
+ oldSourcePathOrFile,
327
+ oldTargetPathOrFile,
328
+ shouldForceMarkdownLinks,
329
+ shouldUpdateFilenameAlias
322
330
  } = options;
323
- if (!pathOrFile) {
331
+ if (!newTargetPathOrFile) {
324
332
  return link.original;
325
333
  }
326
- let file = (0, import_FileSystem.getFile)(app, pathOrFile);
327
- const oldPath = (0, import_FileSystem.getPath)(oldPathOrFile ?? sourcePathOrFile);
328
- const isWikilink = testWikilink(link.original) && forceMarkdownLinks !== true;
334
+ const targetFile = (0, import_FileSystem.getFile)(app, newTargetPathOrFile);
335
+ const oldTargetPath = (0, import_FileSystem.getPath)(oldTargetPathOrFile ?? newTargetPathOrFile);
336
+ const isWikilink = testWikilink(link.original) && shouldForceMarkdownLinks !== true;
329
337
  const { subpath } = splitSubpath(link.link);
330
- const newPath = renameMap?.get(file.path);
338
+ if ((0, import_FileSystem.isCanvasFile)(newSourcePathOrFile)) {
339
+ return targetFile.path + subpath;
340
+ }
331
341
  let alias = shouldResetAlias({
332
342
  app,
333
343
  displayText: link.displayText,
334
344
  isWikilink,
335
- otherPathOrFiles: [oldPath, newPath],
336
- pathOrFile,
337
- sourcePathOrFile
345
+ newSourcePathOrFile,
346
+ oldSourcePathOrFile,
347
+ oldTargetPath,
348
+ targetPathOrFile: targetFile
338
349
  }) ? void 0 : link.displayText;
339
350
  if (shouldUpdateFilenameAlias ?? true) {
340
- if (alias?.toLowerCase() === (0, import_Path.basename)(oldPath, (0, import_Path.extname)(oldPath)).toLowerCase()) {
341
- alias = file.basename;
342
- } else if (alias?.toLowerCase() === (0, import_Path.basename)(oldPath).toLowerCase()) {
343
- alias = file.name;
351
+ if (alias?.toLowerCase() === (0, import_Path.basename)(oldTargetPath, (0, import_Path.extname)(oldTargetPath)).toLowerCase()) {
352
+ alias = targetFile.basename;
353
+ } else if (alias?.toLowerCase() === (0, import_Path.basename)(oldTargetPath).toLowerCase()) {
354
+ alias = targetFile.name;
344
355
  }
345
356
  }
346
- if (newPath) {
347
- file = (0, import_FileSystem.getFile)(app, newPath, true);
348
- }
349
- if ((0, import_FileSystem.isCanvasFile)(sourcePathOrFile)) {
350
- return file.path + subpath;
351
- }
352
357
  const newLink = generateMarkdownLink({
353
358
  alias,
354
359
  app,
355
- isWikilink: forceMarkdownLinks ? false : void 0,
360
+ isWikilink: shouldForceMarkdownLinks ? false : void 0,
356
361
  originalLink: link.original,
357
- pathOrFile: file,
358
- sourcePathOrFile,
359
- subpath
362
+ sourcePathOrFile: newSourcePathOrFile,
363
+ subpath,
364
+ targetPathOrFile: targetFile
360
365
  });
361
366
  return newLink;
362
367
  }
363
368
  async function updateLinksInFile(options) {
364
369
  const {
365
370
  app,
366
- embedOnlyLinks,
367
- forceMarkdownLinks,
368
- oldPathOrFile,
369
- pathOrFile,
370
- renameMap,
371
+ newSourcePathOrFile,
372
+ oldSourcePathOrFile,
373
+ shouldForceMarkdownLinks,
374
+ shouldUpdateEmbedOnlyLinks,
371
375
  shouldUpdateFilenameAlias
372
376
  } = options;
373
- if ((0, import_FileSystem.isCanvasFile)(pathOrFile) && !app.internalPlugins.getEnabledPluginById("canvas")) {
377
+ if ((0, import_FileSystem.isCanvasFile)(newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById("canvas")) {
374
378
  return;
375
379
  }
376
- await editLinks(app, pathOrFile, (link) => {
380
+ await editLinks(app, newSourcePathOrFile, (link) => {
377
381
  const isEmbedLink = testEmbed(link.original);
378
- if (embedOnlyLinks !== void 0 && embedOnlyLinks !== isEmbedLink) {
382
+ if (shouldUpdateEmbedOnlyLinks !== void 0 && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {
379
383
  return;
380
384
  }
381
385
  return convertLink({
382
386
  app,
383
- forceMarkdownLinks,
384
387
  link,
385
- oldPathOrFile,
386
- renameMap,
387
- shouldUpdateFilenameAlias,
388
- sourcePathOrFile: pathOrFile
388
+ newSourcePathOrFile,
389
+ oldSourcePathOrFile,
390
+ shouldForceMarkdownLinks,
391
+ shouldUpdateFilenameAlias
389
392
  });
390
393
  });
391
394
  }
@@ -406,4 +409,4 @@ async function updateLinksInFile(options) {
406
409
  updateLink,
407
410
  updateLinksInFile
408
411
  });
409
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/Link.ts"],
  "sourcesContent": ["var __process = globalThis['process'] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation Link\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n * The functions integrate with Obsidian's API to ensure that links are managed correctly within the vault.\n **/\n\nimport type {\n  Link,\n  Text\n} from 'mdast';\nimport type {\n  App,\n  Reference,\n  TFile\n} from 'obsidian';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\n\nimport type {\n  MaybePromise,\n  RetryOptions\n} from '../Async.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\n\nimport { toJson } from '../Object.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  trimStart\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport { applyFileChanges } from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  trimMarkdownExtension\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  tempRegisterFileAndRun\n} from './MetadataCache.ts';\nimport {\n  shouldUseRelativeLinks,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport { referenceToFileChange } from './Reference.ts';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Options for converting a link.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Whether to force markdown links.\n   */\n  forceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * The reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * The old path of the link.\n   */\n  oldPathOrFile?: PathOrFile | undefined;\n\n  /**\n   * A map of old and new file paths.\n   */\n  renameMap?: Map<string, string> | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n\n  /**\n   * The source file containing the link.\n   */\n  sourcePathOrFile: PathOrFile;\n}\n\n/**\n * Wrapper for default options for generating markdown links.\n */\nexport interface GenerateMarkdownLinkDefaultOptionsWrapper {\n  /**\n   * The default options for generating markdown links.\n   */\n  defaultOptionsFn: () => Partial<GenerateMarkdownLinkOptions>;\n}\n\n/**\n * Options for generating a markdown link.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * The alias for the link.\n   */\n  alias?: string | undefined;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   */\n  allowEmptyEmbedAlias?: boolean | undefined;\n\n  /**\n   * Whether to allow non-existing files. If `false` and `pathOrFile` is a non-existing file, an error will be thrown. Defaults to `false`.\n   */\n  allowNonExistingFile?: boolean | undefined;\n\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be relative. If not provided or `false`, it will be inferred based on the Obsidian settings.\n   */\n  forceRelativePath?: boolean | undefined;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Has no effect if `allowEmptyEmbedAlias` is `true`. Defaults to `false`.\n   */\n  includeAttachmentExtensionToEmbedAlias?: boolean | undefined;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   */\n  isEmbed?: boolean | undefined;\n\n  /**\n   * Indicates if the link should be a wikilink. If not provided, it will be inferred based on the Obsidian settings.\n   */\n  isWikilink?: boolean | undefined;\n\n  /**\n    * The original link text. If provided, it will be used to infer the values of `isEmbed`, `isWikilink`, `useLeadingDot`, and `useAngleBrackets`.\n    * These inferred values will be overridden by corresponding settings if specified.\n    */\n  originalLink?: string | undefined;\n\n  /**\n   * The file to link to.\n   */\n  pathOrFile: PathOrFile;\n\n  /**\n   * The source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * The subpath of the link.\n   */\n  subpath?: string | undefined;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`. Has no effect if `isWikilink` is `true`\n   */\n  useAngleBrackets?: boolean | undefined;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`. Has no effect if `isWikilink` is `true` or `isRelative` is `false`.\n   */\n  useLeadingDot?: boolean | undefined;\n}\n\n/**\n * The result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * The alias of the link.\n   */\n  alias?: string | undefined;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal?: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * The title of the link.\n   */\n  title?: string | undefined;\n\n  /**\n   * The URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for determining if the alias of a link should be reset.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * The display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean | undefined;\n\n  /**\n   * Other paths associated with the link.\n   */\n  otherPathOrFiles: (PathOrFile | undefined)[];\n\n  /**\n   * The path or file of the link.\n   */\n  pathOrFile: PathOrFile;\n\n  /**\n   * The source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * The link path.\n   */\n  linkPath: string;\n\n  /**\n   * The subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for updating a link.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Whether to force markdown links.\n   */\n  forceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * The reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * The old path of the file.\n   */\n  oldPathOrFile?: PathOrFile | undefined;\n\n  /**\n   * The file associated with the link.\n   */\n  pathOrFile: null | PathOrFile;\n\n  /**\n   * A map of old and new file paths.\n   */\n  renameMap?: Map<string, string> | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n\n  /**\n   * The source file containing the link.\n   */\n  sourcePathOrFile: PathOrFile;\n}\n\n/**\n * Options for updating links in a file.\n */\nexport interface UpdateLinksInFileOptions {\n  /**\n   * The obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  embedOnlyLinks?: boolean | undefined;\n\n  /**\n   * Whether to force the links to be in Markdown format.\n   */\n  forceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * The old path of the file.\n   */\n  oldPathOrFile?: PathOrFile | undefined;\n\n  /**\n   * The file to update the links in.\n   */\n  pathOrFile: PathOrFile;\n\n  /**\n   * A map of old and new paths for renaming links.\n   */\n  renameMap?: Map<string, string> | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n}\n\ninterface WikiLinkNode {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const pathOrFile = options.oldPathOrFile ? extractLinkFile(options.app, options.link, options.oldPathOrFile) : null;\n  return updateLink({\n    app: options.app,\n    forceMarkdownLinks: options.forceMarkdownLinks,\n    link: options.link,\n    oldPathOrFile: pathOrFile ?? undefined,\n    pathOrFile,\n    renameMap: options.renameMap,\n    shouldUpdateFilenameAlias: options.shouldUpdateFilenameAlias,\n    sourcePathOrFile: options.sourcePathOrFile\n  });\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param retryOptions - Optional options for retrying the operation.\n * @returns A promise that resolves when the backlinks have been edited.\n */\n// eslint-disable-next-line @typescript-eslint/no-invalid-void-type\nexport async function editBacklinks(app: App, pathOrFile: PathOrFile, linkConverter: (link: Reference) => MaybePromise<string | void>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, retryOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, retryOptions);\n  }\n}\n\n/**\n * Edits the links in the specified file or path using the provided link converter function.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param retryOptions - Optional options for retrying the operation.\n * @returns A promise that resolves when the links have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n  linkConverter: (link: Reference) => MaybePromise<string | void>,\n  retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async () => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    if (!cache) {\n      return [];\n    }\n\n    const changes: FileChange[] = [];\n\n    for (const link of getAllLinks(cache)) {\n      const newContent = await linkConverter(link);\n      if (newContent === undefined) {\n        continue;\n      }\n\n      changes.push(referenceToFileChange(link, newContent));\n    }\n\n    return changes;\n  }, retryOptions);\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param notePathOrFile - The path or file of the note containing the link.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, notePathOrFile: PathOrFile): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  return app.metadataCache.getFirstLinkpathDest(linkPath, getPath(notePathOrFile));\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const configurableDefaultOptionsFn = (app.fileManager.generateMarkdownLink as Partial<GenerateMarkdownLinkDefaultOptionsWrapper>).defaultOptionsFn ?? ((): Partial<GenerateMarkdownLinkOptions> => ({}));\n  const configurableDefaultOptions = configurableDefaultOptionsFn();\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    allowEmptyEmbedAlias: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };\n\n  const file = getFile(app, options.pathOrFile, options.allowNonExistingFile);\n\n  return tempRegisterFileAndRun(app, file, () => {\n    const sourcePath = getPath(options.sourcePathOrFile);\n    const subpath = options.subpath ?? '';\n    let alias = options.alias ?? '';\n    const isEmbed = options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(file);\n    const isWikilink = options.isWikilink ?? (options.originalLink ? testWikilink(options.originalLink) : undefined) ?? shouldUseWikilinks(app);\n    const forceRelativePath = options.forceRelativePath ?? shouldUseRelativeLinks(app);\n    const useLeadingDot = options.useLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false;\n    const useAngleBrackets = options.useAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false;\n\n    let linkText = file.path === sourcePath && subpath\n      ? subpath\n      : forceRelativePath\n        ? relative(dirname(sourcePath), isWikilink ? trimMarkdownExtension(file) : file.path) + subpath\n        : app.metadataCache.fileToLinktext(file, sourcePath, isWikilink) + subpath;\n\n    if (forceRelativePath && useLeadingDot && !linkText.startsWith('.') && !linkText.startsWith('#')) {\n      linkText = './' + linkText;\n    }\n\n    const embedPrefix = isEmbed ? '!' : '';\n\n    if (!isWikilink) {\n      if (useAngleBrackets) {\n        linkText = `<${linkText}>`;\n      } else {\n        linkText = linkText.replace(SPECIAL_LINK_SYMBOLS_REGEXP, function (specialLinkSymbol) {\n          return encodeURIComponent(specialLinkSymbol);\n        });\n      }\n\n      if (!alias && (!isEmbed || !options.allowEmptyEmbedAlias)) {\n        alias = !options.includeAttachmentExtensionToEmbedAlias || isMarkdownFile(file) ? file.basename : file.name;\n      }\n\n      alias = alias.replace(SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&');\n\n      return `${embedPrefix}[${alias}](${linkText})`;\n    } else {\n      if (alias && alias.toLowerCase() === linkText.toLowerCase()) {\n        linkText = alias;\n        alias = '';\n      }\n\n      const aliasPart = alias ? `|${alias}` : '';\n      return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n    }\n  });\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  if (isUrl(str)) {\n    return {\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      url: str\n    };\n  }\n\n  const EMBED_PREFIX = '!';\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const WIKILINK_DIVIDER = '|';\n\n  const isEmbed = str.startsWith(EMBED_PREFIX);\n  if (isEmbed) {\n    str = trimStart(str, EMBED_PREFIX);\n  }\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(str);\n\n  if (root.children.length !== 1) {\n    return null;\n  }\n\n  const paragraph = root.children[0];\n\n  if (paragraph?.type !== 'paragraph') {\n    return null;\n  }\n\n  if (paragraph.children.length !== 1) {\n    return null;\n  }\n\n  const node = paragraph.children[0];\n\n  if (node?.position?.start.offset !== 0) {\n    return null;\n  }\n\n  if (node.position.end.offset !== str.length) {\n    return null;\n  }\n\n  switch (node.type as string) {\n    case 'link': {\n      const linkNode = node as Link;\n      const aliasNode = linkNode.children[0] as Text | undefined;\n      const rawUrl = str.slice((aliasNode?.position?.end.offset ?? 1) + LINK_ALIAS_SUFFIX.length, (linkNode.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n      const hasAngleBrackets = str.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n      const isExternal = isUrl(linkNode.url);\n      let url = linkNode.url;\n      if (!isExternal) {\n        if (!hasAngleBrackets) {\n          try {\n            url = decodeURIComponent(url);\n          } catch (error) {\n            console.error(`Failed to decode URL ${url}`, error);\n          }\n        }\n      }\n      return {\n        alias: aliasNode?.value,\n        hasAngleBrackets,\n        isEmbed,\n        isExternal,\n        isWikilink: false,\n        title: linkNode.title ?? undefined,\n        url\n      };\n    }\n    case 'wikiLink': {\n      const wikiLinkNode = node as unknown as WikiLinkNode;\n      return {\n        alias: str.includes(WIKILINK_DIVIDER) ? wikiLinkNode.data.alias : undefined,\n        isEmbed,\n        isWikilink: true,\n        url: wikiLinkNode.value\n      };\n    }\n    default:\n      return null;\n  }\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    otherPathOrFiles,\n    pathOrFile,\n    sourcePathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  const file = getFile(app, pathOrFile);\n\n  if (!displayText) {\n    return true;\n  }\n\n  const sourcePath = getPath(sourcePathOrFile);\n  const sourceDir = dirname(sourcePath);\n\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [file.path, ...otherPathOrFiles]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(sourceDir, path));\n  }\n\n  aliasesToReset.add(app.metadataCache.fileToLinktext(file, sourcePath, false));\n\n  const cleanDisplayText = normalizePath(displayText.split(' > ')[0] ?? '').replace(/^\\.\\//, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const dir = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(dir, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    forceMarkdownLinks,\n    link,\n    oldPathOrFile,\n    pathOrFile,\n    renameMap,\n    shouldUpdateFilenameAlias,\n    sourcePathOrFile\n  } = options;\n  if (!pathOrFile) {\n    return link.original;\n  }\n  let file = getFile(app, pathOrFile);\n  const oldPath = getPath(oldPathOrFile ?? sourcePathOrFile);\n  const isWikilink = testWikilink(link.original) && forceMarkdownLinks !== true;\n  const { subpath } = splitSubpath(link.link);\n\n  const newPath = renameMap?.get(file.path);\n  let alias = shouldResetAlias({\n    app,\n    displayText: link.displayText,\n    isWikilink,\n    otherPathOrFiles: [oldPath, newPath],\n    pathOrFile,\n    sourcePathOrFile\n  })\n    ? undefined\n    : link.displayText;\n\n  if (shouldUpdateFilenameAlias ?? true) {\n    if (alias?.toLowerCase() === basename(oldPath, extname(oldPath)).toLowerCase()) {\n      alias = file.basename;\n    } else if (alias?.toLowerCase() === basename(oldPath).toLowerCase()) {\n      alias = file.name;\n    }\n  }\n\n  if (newPath) {\n    file = getFile(app, newPath, true);\n  }\n\n  if (isCanvasFile(sourcePathOrFile)) {\n    return file.path + subpath;\n  }\n\n  const newLink = generateMarkdownLink({\n    alias,\n    app,\n    isWikilink: forceMarkdownLinks ? false : undefined,\n    originalLink: link.original,\n    pathOrFile: file,\n    sourcePathOrFile,\n    subpath\n  });\n  return newLink;\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A promise that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    embedOnlyLinks,\n    forceMarkdownLinks,\n    oldPathOrFile,\n    pathOrFile,\n    renameMap,\n    shouldUpdateFilenameAlias\n  } = options;\n\n  if (isCanvasFile(pathOrFile) && !app.internalPlugins.getEnabledPluginById('canvas')) {\n    return;\n  }\n\n  await editLinks(app, pathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (embedOnlyLinks !== undefined && embedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink({\n      app,\n      forceMarkdownLinks,\n      link,\n      oldPathOrFile,\n      renameMap,\n      shouldUpdateFilenameAlias,\n      sourcePathOrFile: pathOrFile\n    });\n  });\n}\n"],
  "mappings": "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",
  "names": ["remarkParse", "pathOrFile"]
}

412
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/Link.ts"],
  "sourcesContent": ["var __process = globalThis['process'] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation Link\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n * The functions integrate with Obsidian's API to ensure that links are managed correctly within the vault.\n **/\n\nimport type {\n  Link,\n  Text\n} from 'mdast';\nimport type {\n  App,\n  Reference,\n  TFile\n} from 'obsidian';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\n\nimport type {\n  MaybePromise,\n  RetryOptions\n} from '../Async.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\n\nimport { toJson } from '../Object.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  trimStart\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport { applyFileChanges } from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  trimMarkdownExtension\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  tempRegisterFileAndRun\n} from './MetadataCache.ts';\nimport {\n  shouldUseRelativeLinks,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport { referenceToFileChange } from './Reference.ts';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Options for converting a link.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * The reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * The source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * The old path of the link.\n   */\n  oldSourcePathOrFile?: PathOrFile | undefined;\n\n  /**\n   * Whether to force markdown links.\n   */\n  shouldForceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n}\n\n/**\n * Wrapper for default options for generating markdown links.\n */\nexport interface GenerateMarkdownLinkDefaultOptionsWrapper {\n  /**\n   * The default options for generating markdown links.\n   */\n  defaultOptionsFn: () => Partial<GenerateMarkdownLinkOptions>;\n}\n\n/**\n * Options for generating a markdown link.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * The alias for the link.\n   */\n  alias?: string | undefined;\n\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   */\n  isEmbed?: boolean | undefined;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   */\n  isEmptyEmbedAliasAllowed?: boolean | undefined;\n\n  /**\n   * Whether to allow non-existing files. If `false` and `pathOrFile` is a non-existing file, an error will be thrown. Defaults to `false`.\n   */\n  isNonExistingFileAllowed?: boolean | undefined;\n\n  /**\n   * Indicates if the link should be a wikilink. If not provided, it will be inferred based on the Obsidian settings.\n   */\n  isWikilink?: boolean | undefined;\n\n  /**\n    * The original link text. If provided, it will be used to infer the values of `isEmbed`, `isWikilink`, `useLeadingDot`, and `useAngleBrackets`.\n    * These inferred values will be overridden by corresponding settings if specified.\n    */\n  originalLink?: string | undefined;\n\n  /**\n   * Indicates if the link should be relative. If not provided or `false`, it will be inferred based on the Obsidian settings.\n   */\n  shouldForceRelativePath?: boolean | undefined;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Has no effect if `allowEmptyEmbedAlias` is `true`. Defaults to `false`.\n   */\n  shouldIncludeAttachmentExtensionToEmbedAlias?: boolean | undefined;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`. Has no effect if `isWikilink` is `true`\n   */\n  shouldUseAngleBrackets?: boolean | undefined;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`. Has no effect if `isWikilink` is `true` or `isRelative` is `false`.\n   */\n  shouldUseLeadingDot?: boolean | undefined;\n\n  /**\n   * The source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * The subpath of the link.\n   */\n  subpath?: string | undefined;\n\n  /**\n   * The target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * The result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * The alias of the link.\n   */\n  alias?: string | undefined;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal?: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * The title of the link.\n   */\n  title?: string | undefined;\n\n  /**\n   * The URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for determining if the alias of a link should be reset.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * The display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean | undefined;\n\n  /**\n   * The source path of the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * The old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile | undefined;\n\n  /**\n   * The old target path of the link.\n   */\n  oldTargetPath: PathOrFile;\n\n  /**\n   * The target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * The link path.\n   */\n  linkPath: string;\n\n  /**\n   * The subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for updating a link.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * The reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * The source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * The file associated with the link.\n   */\n  newTargetPathOrFile: PathOrFile;\n\n  /**\n   * The old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile | undefined;\n\n  /**\n   * The old path of the file.\n   */\n  oldTargetPathOrFile?: PathOrFile | undefined;\n\n  /**\n   * Whether to force markdown links.\n   */\n  shouldForceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n}\n\n/**\n * Options for updating links in a file.\n */\nexport interface UpdateLinksInFileOptions {\n  /**\n   * The obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * The file to update the links in.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * The old path of the file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to force the links to be in Markdown format.\n   */\n  shouldForceMarkdownLinks?: boolean | undefined;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean | undefined;\n\n  /**\n   * Whether to update filename alias. Defaults to `true`.\n   */\n  shouldUpdateFilenameAlias?: boolean | undefined;\n}\n\ninterface WikiLinkNode {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);\n  if (!targetFile) {\n    return options.link.original;\n  }\n\n  return updateLink({\n    app: options.app,\n    link: options.link,\n    newSourcePathOrFile: options.newSourcePathOrFile,\n    newTargetPathOrFile: targetFile,\n    oldSourcePathOrFile: options.oldSourcePathOrFile,\n    shouldForceMarkdownLinks: options.shouldForceMarkdownLinks,\n    shouldUpdateFilenameAlias: options.shouldUpdateFilenameAlias\n  });\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param retryOptions - Optional options for retrying the operation.\n * @returns A promise that resolves when the backlinks have been edited.\n */\n// eslint-disable-next-line @typescript-eslint/no-invalid-void-type\nexport async function editBacklinks(app: App, pathOrFile: PathOrFile, linkConverter: (link: Reference) => MaybePromise<string | void>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, retryOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, retryOptions);\n  }\n}\n\n/**\n * Edits the links in the specified file or path using the provided link converter function.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param retryOptions - Optional options for retrying the operation.\n * @returns A promise that resolves when the links have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n  linkConverter: (link: Reference) => MaybePromise<string | void>,\n  retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async () => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    if (!cache) {\n      return [];\n    }\n\n    const changes: FileChange[] = [];\n\n    for (const link of getAllLinks(cache)) {\n      const newContent = await linkConverter(link);\n      if (newContent === undefined) {\n        continue;\n      }\n\n      changes.push(referenceToFileChange(link, newContent));\n    }\n\n    return changes;\n  }, retryOptions);\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param sourcePathOrFile - The source path or file.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, sourcePathOrFile: PathOrFile): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  return app.metadataCache.getFirstLinkpathDest(linkPath, getPath(sourcePathOrFile));\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const configurableDefaultOptionsFn = (app.fileManager.generateMarkdownLink as Partial<GenerateMarkdownLinkDefaultOptionsWrapper>).defaultOptionsFn ?? ((): Partial<GenerateMarkdownLinkOptions> => ({}));\n  const configurableDefaultOptions = configurableDefaultOptionsFn();\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    isEmptyEmbedAliasAllowed: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };\n\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n\n  return tempRegisterFileAndRun(app, targetFile, () => {\n    const sourcePath = getPath(options.sourcePathOrFile);\n    const subpath = options.subpath ?? '';\n    let alias = options.alias ?? '';\n    const isEmbed = options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(targetFile);\n    const isWikilink = options.isWikilink ?? (options.originalLink ? testWikilink(options.originalLink) : undefined) ?? shouldUseWikilinks(app);\n    const shouldForceRelativePath = options.shouldForceRelativePath ?? shouldUseRelativeLinks(app);\n    const shouldUseLeadingDot = options.shouldUseLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false;\n    const shouldUseAngleBrackets = options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false;\n\n    let linkText = targetFile.path === sourcePath && subpath\n      ? subpath\n      : shouldForceRelativePath\n        ? relative(dirname(sourcePath), isWikilink ? trimMarkdownExtension(targetFile) : targetFile.path) + subpath\n        : app.metadataCache.fileToLinktext(targetFile, sourcePath, isWikilink) + subpath;\n\n    if (shouldForceRelativePath && shouldUseLeadingDot && !linkText.startsWith('.') && !linkText.startsWith('#')) {\n      linkText = './' + linkText;\n    }\n\n    const embedPrefix = isEmbed ? '!' : '';\n\n    if (!isWikilink) {\n      if (shouldUseAngleBrackets) {\n        linkText = `<${linkText}>`;\n      } else {\n        linkText = linkText.replace(SPECIAL_LINK_SYMBOLS_REGEXP, function (specialLinkSymbol) {\n          return encodeURIComponent(specialLinkSymbol);\n        });\n      }\n\n      if (!alias && (!isEmbed || !options.isEmptyEmbedAliasAllowed)) {\n        alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || isMarkdownFile(targetFile) ? targetFile.basename : targetFile.name;\n      }\n\n      alias = alias.replace(SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&');\n\n      return `${embedPrefix}[${alias}](${linkText})`;\n    } else {\n      if (alias && alias.toLowerCase() === linkText.toLowerCase()) {\n        linkText = alias;\n        alias = '';\n      }\n\n      const aliasPart = alias ? `|${alias}` : '';\n      return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n    }\n  });\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  if (isUrl(str)) {\n    return {\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      url: str\n    };\n  }\n\n  const EMBED_PREFIX = '!';\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const WIKILINK_DIVIDER = '|';\n\n  const isEmbed = str.startsWith(EMBED_PREFIX);\n  if (isEmbed) {\n    str = trimStart(str, EMBED_PREFIX);\n  }\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(str);\n\n  if (root.children.length !== 1) {\n    return null;\n  }\n\n  const paragraph = root.children[0];\n\n  if (paragraph?.type !== 'paragraph') {\n    return null;\n  }\n\n  if (paragraph.children.length !== 1) {\n    return null;\n  }\n\n  const node = paragraph.children[0];\n\n  if (node?.position?.start.offset !== 0) {\n    return null;\n  }\n\n  if (node.position.end.offset !== str.length) {\n    return null;\n  }\n\n  switch (node.type as string) {\n    case 'link': {\n      const linkNode = node as Link;\n      const aliasNode = linkNode.children[0] as Text | undefined;\n      const rawUrl = str.slice((aliasNode?.position?.end.offset ?? 1) + LINK_ALIAS_SUFFIX.length, (linkNode.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n      const hasAngleBrackets = str.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n      const isExternal = isUrl(linkNode.url);\n      let url = linkNode.url;\n      if (!isExternal) {\n        if (!hasAngleBrackets) {\n          try {\n            url = decodeURIComponent(url);\n          } catch (error) {\n            console.error(`Failed to decode URL ${url}`, error);\n          }\n        }\n      }\n      return {\n        alias: aliasNode?.value,\n        hasAngleBrackets,\n        isEmbed,\n        isExternal,\n        isWikilink: false,\n        title: linkNode.title ?? undefined,\n        url\n      };\n    }\n    case 'wikiLink': {\n      const wikiLinkNode = node as unknown as WikiLinkNode;\n      return {\n        alias: str.includes(WIKILINK_DIVIDER) ? wikiLinkNode.data.alias : undefined,\n        isEmbed,\n        isWikilink: true,\n        url: wikiLinkNode.value\n      };\n    }\n    default:\n      return null;\n  }\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  if (!displayText) {\n    return true;\n  }\n\n  const targetFile = getFile(app, targetPathOrFile);\n  const newSourcePath = getPath(newSourcePathOrFile);\n  const oldSourcePath = getPath(oldSourcePathOrFile ?? newSourcePathOrFile);\n  const newSourceDir = dirname(newSourcePath);\n  const oldSourceDir = dirname(oldSourcePath);\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [targetFile.path, oldTargetPath]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(newSourceDir, path));\n    aliasesToReset.add(relative(oldSourceDir, path));\n  }\n\n  for (const sourcePath of [oldSourcePath, newSourcePath]) {\n    aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));\n  }\n\n  const cleanDisplayText = normalizePath(displayText.split(' > ')[0] ?? '').replace(/^\\.\\//, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const dir = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(dir, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    link,\n    newSourcePathOrFile: newSourcePathOrFile,\n    newTargetPathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPathOrFile,\n    shouldForceMarkdownLinks,\n    shouldUpdateFilenameAlias\n  } = options;\n  if (!newTargetPathOrFile) {\n    return link.original;\n  }\n  const targetFile = getFile(app, newTargetPathOrFile);\n  const oldTargetPath = getPath(oldTargetPathOrFile ?? newTargetPathOrFile);\n  const isWikilink = testWikilink(link.original) && shouldForceMarkdownLinks !== true;\n  const { subpath } = splitSubpath(link.link);\n\n  if (isCanvasFile(newSourcePathOrFile)) {\n    return targetFile.path + subpath;\n  }\n\n  let alias = shouldResetAlias({\n    app,\n    displayText: link.displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile: targetFile\n  })\n    ? undefined\n    : link.displayText;\n\n  if (shouldUpdateFilenameAlias ?? true) {\n    if (alias?.toLowerCase() === basename(oldTargetPath, extname(oldTargetPath)).toLowerCase()) {\n      alias = targetFile.basename;\n    } else if (alias?.toLowerCase() === basename(oldTargetPath).toLowerCase()) {\n      alias = targetFile.name;\n    }\n  }\n\n  const newLink = generateMarkdownLink({\n    alias,\n    app,\n    isWikilink: shouldForceMarkdownLinks ? false : undefined,\n    originalLink: link.original,\n    sourcePathOrFile: newSourcePathOrFile,\n    subpath,\n    targetPathOrFile: targetFile\n  });\n  return newLink;\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A promise that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldForceMarkdownLinks,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFilenameAlias\n  } = options;\n\n  if (isCanvasFile(newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById('canvas')) {\n    return;\n  }\n\n  await editLinks(app, newSourcePathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink({\n      app,\n      link,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldForceMarkdownLinks,\n      shouldUpdateFilenameAlias\n    });\n  });\n}\n"],
  "mappings": "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",
  "names": ["remarkParse"]
}
