obsidian-dev-utils 49.0.3 → 50.0.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/CHANGELOG.md +6 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/Markdown.cjs +3 -2
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +7 -7
- package/dist/lib/cjs/obsidian/Vault.cjs +46 -5
- package/dist/lib/cjs/obsidian/Vault.d.cts +24 -0
- package/dist/lib/cjs/obsidian/VaultDelete.cjs +171 -0
- package/dist/lib/cjs/obsidian/VaultDelete.d.cts +18 -0
- package/dist/lib/cjs/obsidian/index.cjs +4 -4
- package/dist/lib/cjs/obsidian/index.d.cts +1 -1
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/Markdown.mjs +3 -2
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +9 -10
- package/dist/lib/esm/obsidian/Vault.d.mts +24 -0
- package/dist/lib/esm/obsidian/Vault.mjs +42 -4
- package/dist/lib/esm/obsidian/VaultDelete.d.mts +18 -0
- package/dist/lib/esm/obsidian/VaultDelete.mjs +73 -0
- package/dist/lib/esm/obsidian/index.d.mts +1 -1
- package/dist/lib/esm/obsidian/index.mjs +3 -3
- package/obsidian/VaultDelete/package.json +6 -0
- package/package.json +1 -1
- package/dist/lib/cjs/obsidian/VaultEx.cjs +0 -198
- package/dist/lib/cjs/obsidian/VaultEx.d.cts +0 -36
- package/dist/lib/esm/obsidian/VaultEx.d.mts +0 -36
- package/dist/lib/esm/obsidian/VaultEx.mjs +0 -98
- package/obsidian/VaultEx/package.json +0 -6
|
@@ -35,6 +35,7 @@ import {
|
|
|
35
35
|
isFolder
|
|
36
36
|
} from "./FileSystem.mjs";
|
|
37
37
|
import { invokeWithPatchAsync } from "./MonkeyAround.mjs";
|
|
38
|
+
import { trashSafe } from "./Vault.mjs";
|
|
38
39
|
let domEventsHandlersConstructor = null;
|
|
39
40
|
class FixedZIndexDomEventsHandlersInfo {
|
|
40
41
|
constructor(app, path, el) {
|
|
@@ -170,7 +171,7 @@ async function getDomEventsHandlersConstructor(app) {
|
|
|
170
171
|
return ctor;
|
|
171
172
|
} finally {
|
|
172
173
|
if (shouldDelete) {
|
|
173
|
-
await app
|
|
174
|
+
await trashSafe(app, mdFile);
|
|
174
175
|
}
|
|
175
176
|
}
|
|
176
177
|
}
|
|
@@ -181,4 +182,4 @@ export {
|
|
|
181
182
|
renderExternalLink,
|
|
182
183
|
renderInternalLink
|
|
183
184
|
};
|
|
184
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Markdown.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing Markdown content in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { App } from 'obsidian';\nimport type {\n  DomEventsHandlersConstructor,\n  DomEventsHandlersInfo,\n  EmbedCreator\n} from 'obsidian-typings';\n\nimport {\n  Component,\n  HoverPopover,\n  MarkdownPreviewRenderer,\n  MarkdownRenderer\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type { PathOrAbstractFile } from './FileSystem.ts';\n\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport { getZIndex } from '../HTMLElement.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport {\n  getAbstractFileOrNull,\n  getPath,\n  isFolder\n} from './FileSystem.ts';\nimport { invokeWithPatchAsync } from './MonkeyAround.ts';\n\nlet domEventsHandlersConstructor: DomEventsHandlersConstructor | null = null;\n\n/**\n * The params for the full render.\n */\nexport interface FullRenderParams {\n  /**\n   * The Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * The Component instance to use for the render.\n   */\n  readonly component?: Component;\n\n  /**\n   * The HTMLElement to render to.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * The Markdown string to render.\n   */\n  readonly markdown: string;\n\n  /**\n   * Whether to register link handlers for the rendered element.\n   */\n  readonly shouldRegisterLinkHandlers?: boolean;\n\n  /**\n   * The source path to resolve relative links.\n   */\n  readonly sourcePath?: string;\n}\n\ntype RegisterDomEventsFn = typeof MarkdownPreviewRenderer.registerDomEvents;\n\nclass FixedZIndexDomEventsHandlersInfo implements DomEventsHandlersInfo {\n  public get hoverPopover(): HoverPopover | null {\n    return this._hoverPopover;\n  }\n\n  public set hoverPopover(hoverPopover: HoverPopover | null) {\n    this._hoverPopover = hoverPopover;\n    if (hoverPopover && this.zIndex !== undefined) {\n      hoverPopover.hoverEl.setCssStyles({\n        zIndex: String(this.zIndex)\n      });\n    }\n  }\n\n  private _hoverPopover: HoverPopover | null = null;\n\n  private zIndex?: number;\n\n  public constructor(public readonly app: App, public readonly path: string, el: HTMLElement) {\n    if (el.isConnected) {\n      this.updateZIndex(el);\n    } else {\n      el.onNodeInserted(() => {\n        this.updateZIndex(el);\n      });\n    }\n  }\n\n  private updateZIndex(el: HTMLElement): void {\n    this.zIndex = getZIndex(el) + 1;\n  }\n}\n\n/**\n * Render the markdown and embeds.\n *\n * @param params - The parameters for the full render.\n * @returns The {@link Promise} that resolves when the full render is complete.\n */\nexport async function fullRender(params: FullRenderParams): Promise<void> {\n  const sourcePath = params.sourcePath ?? '/';\n  let shouldUnloadComponent = false;\n  let component: Component;\n  if (params.component) {\n    component = params.component;\n  } else {\n    component = new Component();\n    component.load();\n    shouldUnloadComponent = true;\n  }\n  await invokeWithPatchAsync(params.app.embedRegistry.embedByExtension, {\n    md: (next: EmbedCreator): EmbedCreator => (context, file, subpath) => {\n      context.displayMode = false;\n      return next(context, file, subpath);\n    }\n  }, async () => {\n    await MarkdownRenderer.render(params.app, params.markdown, params.el, sourcePath, component);\n  });\n\n  if (shouldUnloadComponent) {\n    component.unload();\n  }\n\n  if (params.shouldRegisterLinkHandlers) {\n    await registerLinkHandlers(params.app, params.el, params.sourcePath);\n  }\n}\n\n/**\n * Converts Markdown to HTML.\n *\n * @param app - The Obsidian app instance.\n * @param markdown - The Markdown string to convert.\n * @param sourcePath - (optional) The source path to resolve relative links.\n * @returns The HTML string.\n */\nexport async function markdownToHtml(app: App, markdown: string, sourcePath?: string): Promise<string> {\n  const component = new Component();\n  component.load();\n  const renderDiv = createDiv();\n  await MarkdownRenderer.render(app, markdown, renderDiv, sourcePath ?? '', component);\n  const html = renderDiv.innerHTML;\n  component.unload();\n  return html;\n}\n\n/**\n * Registers link handlers for the given element.\n *\n * @param app - The Obsidian app instance.\n * @param el - The HTMLElement to register link handlers for.\n * @param sourcePath - The source path to resolve relative links from.\n */\nexport async function registerLinkHandlers(app: App, el: HTMLElement, sourcePath?: string): Promise<void> {\n  // eslint-disable-next-line require-atomic-updates -- No race condition.\n  domEventsHandlersConstructor ??= await getDomEventsHandlersConstructor(app);\n  MarkdownPreviewRenderer.registerDomEvents(\n    el,\n    new domEventsHandlersConstructor(new FixedZIndexDomEventsHandlersInfo(app, sourcePath ?? '', el))\n  );\n}\n\n/**\n * Renders an external link.\n *\n * @param app - The Obsidian app instance.\n * @param url - The URL to render the external link for.\n * @param displayText - The text to display for the external link.\n * @returns The HTMLAnchorElement containing the rendered external link.\n */\nexport async function renderExternalLink(app: App, url: string, displayText?: string): Promise<HTMLAnchorElement> {\n  displayText ??= url;\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[${displayText}](${url})`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\n/**\n * Renders an internal link.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrAbstractFile - The path or abstract file to render the internal link for.\n * @param displayText - The text to display for the internal link.\n * @returns The HTMLAnchorElement containing the rendered internal link.\n */\nexport async function renderInternalLink(app: App, pathOrAbstractFile: PathOrAbstractFile, displayText?: string): Promise<HTMLAnchorElement> {\n  const abstractFile = getAbstractFileOrNull(app, pathOrAbstractFile);\n  const path = getPath(app, pathOrAbstractFile);\n  displayText ??= path;\n  if (isFolder(abstractFile)) {\n    return createEl('a', { text: displayText }, (aEl) => {\n      aEl.addEventListener('click', (evt) => {\n        evt.preventDefault();\n        app.internalPlugins.getEnabledPluginById(InternalPluginName.FileExplorer)?.revealInFolder(abstractFile);\n      });\n    });\n  }\n\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[[${path}|${displayText}]]`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\nasync function getDomEventsHandlersConstructor(app: App): Promise<DomEventsHandlersConstructor> {\n  let mdFile = app.vault.getMarkdownFiles()[0];\n  let shouldDelete = false;\n  if (!mdFile) {\n    // eslint-disable-next-line require-atomic-updates -- No race condition.\n    mdFile = await app.vault.create('__temp.md', '');\n    shouldDelete = true;\n  }\n  let ctor: DomEventsHandlersConstructor | null = null;\n  try {\n    await invokeWithPatchAsync(MarkdownPreviewRenderer, {\n      registerDomEvents: (next: RegisterDomEventsFn): RegisterDomEventsFn => {\n        return (el, handlers, childElFn) => {\n          ctor = handlers.constructor as DomEventsHandlersConstructor;\n          next(el, handlers, childElFn);\n        };\n      }\n    }, async () => {\n      const leaf = app.workspace.getLeaf(true);\n      await leaf.openFile(mdFile, {\n        active: true,\n        state: { mode: 'preview' }\n      });\n      await requestAnimationFrameAsync();\n      leaf.detach();\n    });\n\n    assertNonNullable(ctor, 'Failed to get register dom events handlers constructor');\n    return ctor;\n  } finally {\n    if (shouldDelete) {\n      await app.fileManager.trashFile(mdFile);\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["aEl"]
}

|
|
185
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Markdown.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing Markdown content in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { App } from 'obsidian';\nimport type {\n  DomEventsHandlersConstructor,\n  DomEventsHandlersInfo,\n  EmbedCreator\n} from 'obsidian-typings';\n\nimport {\n  Component,\n  HoverPopover,\n  MarkdownPreviewRenderer,\n  MarkdownRenderer\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type { PathOrAbstractFile } from './FileSystem.ts';\n\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport { getZIndex } from '../HTMLElement.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport {\n  getAbstractFileOrNull,\n  getPath,\n  isFolder\n} from './FileSystem.ts';\nimport { invokeWithPatchAsync } from './MonkeyAround.ts';\nimport { trashSafe } from './Vault.ts';\n\nlet domEventsHandlersConstructor: DomEventsHandlersConstructor | null = null;\n\n/**\n * The params for the full render.\n */\nexport interface FullRenderParams {\n  /**\n   * The Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * The Component instance to use for the render.\n   */\n  readonly component?: Component;\n\n  /**\n   * The HTMLElement to render to.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * The Markdown string to render.\n   */\n  readonly markdown: string;\n\n  /**\n   * Whether to register link handlers for the rendered element.\n   */\n  readonly shouldRegisterLinkHandlers?: boolean;\n\n  /**\n   * The source path to resolve relative links.\n   */\n  readonly sourcePath?: string;\n}\n\ntype RegisterDomEventsFn = typeof MarkdownPreviewRenderer.registerDomEvents;\n\nclass FixedZIndexDomEventsHandlersInfo implements DomEventsHandlersInfo {\n  public get hoverPopover(): HoverPopover | null {\n    return this._hoverPopover;\n  }\n\n  public set hoverPopover(hoverPopover: HoverPopover | null) {\n    this._hoverPopover = hoverPopover;\n    if (hoverPopover && this.zIndex !== undefined) {\n      hoverPopover.hoverEl.setCssStyles({\n        zIndex: String(this.zIndex)\n      });\n    }\n  }\n\n  private _hoverPopover: HoverPopover | null = null;\n\n  private zIndex?: number;\n\n  public constructor(public readonly app: App, public readonly path: string, el: HTMLElement) {\n    if (el.isConnected) {\n      this.updateZIndex(el);\n    } else {\n      el.onNodeInserted(() => {\n        this.updateZIndex(el);\n      });\n    }\n  }\n\n  private updateZIndex(el: HTMLElement): void {\n    this.zIndex = getZIndex(el) + 1;\n  }\n}\n\n/**\n * Render the markdown and embeds.\n *\n * @param params - The parameters for the full render.\n * @returns The {@link Promise} that resolves when the full render is complete.\n */\nexport async function fullRender(params: FullRenderParams): Promise<void> {\n  const sourcePath = params.sourcePath ?? '/';\n  let shouldUnloadComponent = false;\n  let component: Component;\n  if (params.component) {\n    component = params.component;\n  } else {\n    component = new Component();\n    component.load();\n    shouldUnloadComponent = true;\n  }\n  await invokeWithPatchAsync(params.app.embedRegistry.embedByExtension, {\n    md: (next: EmbedCreator): EmbedCreator => (context, file, subpath) => {\n      context.displayMode = false;\n      return next(context, file, subpath);\n    }\n  }, async () => {\n    await MarkdownRenderer.render(params.app, params.markdown, params.el, sourcePath, component);\n  });\n\n  if (shouldUnloadComponent) {\n    component.unload();\n  }\n\n  if (params.shouldRegisterLinkHandlers) {\n    await registerLinkHandlers(params.app, params.el, params.sourcePath);\n  }\n}\n\n/**\n * Converts Markdown to HTML.\n *\n * @param app - The Obsidian app instance.\n * @param markdown - The Markdown string to convert.\n * @param sourcePath - (optional) The source path to resolve relative links.\n * @returns The HTML string.\n */\nexport async function markdownToHtml(app: App, markdown: string, sourcePath?: string): Promise<string> {\n  const component = new Component();\n  component.load();\n  const renderDiv = createDiv();\n  await MarkdownRenderer.render(app, markdown, renderDiv, sourcePath ?? '', component);\n  const html = renderDiv.innerHTML;\n  component.unload();\n  return html;\n}\n\n/**\n * Registers link handlers for the given element.\n *\n * @param app - The Obsidian app instance.\n * @param el - The HTMLElement to register link handlers for.\n * @param sourcePath - The source path to resolve relative links from.\n */\nexport async function registerLinkHandlers(app: App, el: HTMLElement, sourcePath?: string): Promise<void> {\n  // eslint-disable-next-line require-atomic-updates -- No race condition.\n  domEventsHandlersConstructor ??= await getDomEventsHandlersConstructor(app);\n  MarkdownPreviewRenderer.registerDomEvents(\n    el,\n    new domEventsHandlersConstructor(new FixedZIndexDomEventsHandlersInfo(app, sourcePath ?? '', el))\n  );\n}\n\n/**\n * Renders an external link.\n *\n * @param app - The Obsidian app instance.\n * @param url - The URL to render the external link for.\n * @param displayText - The text to display for the external link.\n * @returns The HTMLAnchorElement containing the rendered external link.\n */\nexport async function renderExternalLink(app: App, url: string, displayText?: string): Promise<HTMLAnchorElement> {\n  displayText ??= url;\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[${displayText}](${url})`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\n/**\n * Renders an internal link.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrAbstractFile - The path or abstract file to render the internal link for.\n * @param displayText - The text to display for the internal link.\n * @returns The HTMLAnchorElement containing the rendered internal link.\n */\nexport async function renderInternalLink(app: App, pathOrAbstractFile: PathOrAbstractFile, displayText?: string): Promise<HTMLAnchorElement> {\n  const abstractFile = getAbstractFileOrNull(app, pathOrAbstractFile);\n  const path = getPath(app, pathOrAbstractFile);\n  displayText ??= path;\n  if (isFolder(abstractFile)) {\n    return createEl('a', { text: displayText }, (aEl) => {\n      aEl.addEventListener('click', (evt) => {\n        evt.preventDefault();\n        app.internalPlugins.getEnabledPluginById(InternalPluginName.FileExplorer)?.revealInFolder(abstractFile);\n      });\n    });\n  }\n\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[[${path}|${displayText}]]`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\nasync function getDomEventsHandlersConstructor(app: App): Promise<DomEventsHandlersConstructor> {\n  let mdFile = app.vault.getMarkdownFiles()[0];\n  let shouldDelete = false;\n  if (!mdFile) {\n    // eslint-disable-next-line require-atomic-updates -- No race condition.\n    mdFile = await app.vault.create('__temp.md', '');\n    shouldDelete = true;\n  }\n  let ctor: DomEventsHandlersConstructor | null = null;\n  try {\n    await invokeWithPatchAsync(MarkdownPreviewRenderer, {\n      registerDomEvents: (next: RegisterDomEventsFn): RegisterDomEventsFn => {\n        return (el, handlers, childElFn) => {\n          ctor = handlers.constructor as DomEventsHandlersConstructor;\n          next(el, handlers, childElFn);\n        };\n      }\n    }, async () => {\n      const leaf = app.workspace.getLeaf(true);\n      await leaf.openFile(mdFile, {\n        active: true,\n        state: { mode: 'preview' }\n      });\n      await requestAnimationFrameAsync();\n      leaf.detach();\n    });\n\n    assertNonNullable(ctor, 'Failed to get register dom events handlers constructor');\n    return ctor;\n  } finally {\n    if (shouldDelete) {\n      await trashSafe(app, mdFile);\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["aEl"]
}

|
|
@@ -72,15 +72,14 @@ import {
|
|
|
72
72
|
} from "./MetadataCache.mjs";
|
|
73
73
|
import { registerPatch } from "./MonkeyAround.mjs";
|
|
74
74
|
import { addToQueue } from "./Queue.mjs";
|
|
75
|
-
import {
|
|
76
|
-
getSafeRenamePath,
|
|
77
|
-
renameSafe
|
|
78
|
-
} from "./Vault.mjs";
|
|
79
75
|
import {
|
|
80
76
|
deleteEmptyFolder,
|
|
81
77
|
deleteEmptyFolderHierarchy,
|
|
82
|
-
|
|
83
|
-
|
|
78
|
+
getSafeRenamePath,
|
|
79
|
+
renameSafe,
|
|
80
|
+
trashSafe
|
|
81
|
+
} from "./Vault.mjs";
|
|
82
|
+
import { deleteIfNotUsed } from "./VaultDelete.mjs";
|
|
84
83
|
var EmptyFolderBehavior = /* @__PURE__ */ ((EmptyFolderBehavior2) => {
|
|
85
84
|
EmptyFolderBehavior2["Delete"] = "Delete";
|
|
86
85
|
EmptyFolderBehavior2["DeleteWithEmptyParents"] = "DeleteWithEmptyParents";
|
|
@@ -121,7 +120,7 @@ class DeleteHandler {
|
|
|
121
120
|
continue;
|
|
122
121
|
}
|
|
123
122
|
parentFolderPaths.add(attachmentFile.parent?.path ?? "");
|
|
124
|
-
await
|
|
123
|
+
await deleteIfNotUsed(this.app, attachmentFile, this.file.path, false, settings.emptyFolderBehavior !== "Keep" /* Keep */);
|
|
125
124
|
this.abortSignal.throwIfAborted();
|
|
126
125
|
}
|
|
127
126
|
}
|
|
@@ -141,7 +140,7 @@ class DeleteHandler {
|
|
|
141
140
|
return;
|
|
142
141
|
}
|
|
143
142
|
this.abortSignal.throwIfAborted();
|
|
144
|
-
await
|
|
143
|
+
await deleteIfNotUsed(this.app, attachmentFolder, this.file.path, false, settings.emptyFolderBehavior !== "Keep" /* Keep */);
|
|
145
144
|
this.abortSignal.throwIfAborted();
|
|
146
145
|
}
|
|
147
146
|
}
|
|
@@ -705,7 +704,7 @@ class RenameMap {
|
|
|
705
704
|
const newAttachmentFile = getFileOrNull(this.app, newAttachmentFilePath);
|
|
706
705
|
if (newAttachmentFile) {
|
|
707
706
|
getLibDebugger("RenameDeleteHandler:fillRenameMap")(`Removing conflicting attachment ${newAttachmentFile.path}.`);
|
|
708
|
-
await this.app
|
|
707
|
+
await trashSafe(this.app, newAttachmentFile);
|
|
709
708
|
this.abortSignal.throwIfAborted();
|
|
710
709
|
}
|
|
711
710
|
} else {
|
|
@@ -772,4 +771,4 @@ export {
|
|
|
772
771
|
EmptyFolderBehavior,
|
|
773
772
|
registerRenameDeleteHandlers
|
|
774
773
|
};
|
|
775
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/RenameDeleteHandler.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for handling rename and delete events in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type {\n  App,\n  CachedMetadata,\n  FileManager,\n  Plugin,\n  Reference,\n  TAbstractFile,\n  TFile\n} from 'obsidian';\nimport type {\n  LinkUpdate,\n  LinkUpdatesHandler\n} from 'obsidian-typings';\n\nimport { t } from 'i18next';\nimport {\n  Notice,\n  Vault\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type {\n  UpdateLinkParams,\n  UpdateLinksInFileParams\n} from './Link.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport { filterInPlace } from '../Array.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport {\n  AttachmentPathContext,\n  getAttachmentFilePath,\n  getAttachmentFolderPath,\n  hasOwnAttachmentFolder\n} from './AttachmentPath.ts';\nimport {\n  CANVAS_FILE_EXTENSION,\n  getFile,\n  getFileOrNull,\n  getFolderOrNull,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport {\n  editLinks,\n  extractLinkFile,\n  updateLink,\n  updateLinksInFile\n} from './Link.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileOrPath,\n  getBacklinksForFileSafe,\n  registerFileCacheForNonExistingFile,\n  tempRegisterFilesAndRun,\n  tempRegisterFilesAndRunAsync,\n  unregisterFileCacheForNonExistingFile\n} from './MetadataCache.ts';\nimport { registerPatch } from './MonkeyAround.ts';\nimport { addToQueue } from './Queue.ts';\nimport {\n  getSafeRenamePath,\n  renameSafe\n} from './Vault.ts';\nimport {\n  deleteEmptyFolder,\n  deleteEmptyFolderHierarchy,\n  deleteSafe\n} from './VaultEx.ts';\n\n/**\n * A behavior of the rename/delete handler when deleting empty folders.\n */\nexport enum EmptyFolderBehavior {\n  /**\n   * Delete the empty folder.\n   */\n  Delete = 'Delete',\n\n  /**\n   * Delete the empty folder and all its empty parents.\n   */\n  DeleteWithEmptyParents = 'DeleteWithEmptyParents',\n\n  /**\n   * Keep the empty folder.\n   */\n  Keep = 'Keep'\n}\n\n/**\n * Settings for the rename/delete handler.\n */\nexport interface RenameDeleteHandlerSettings {\n  /**\n   * A behavior of the rename/delete handler when deleting empty folders.\n   */\n  emptyFolderBehavior: EmptyFolderBehavior;\n\n  /**\n   * Whether the path is a note.\n   */\n  isNote(path: string): boolean;\n\n  /**\n   * Whether to ignore the path.\n   */\n  isPathIgnored(path: string): boolean;\n\n  /**\n   * Whether to delete conflicting attachments.\n   */\n  shouldDeleteConflictingAttachments: boolean;\n\n  /**\n   * Whether to handle deletions.\n   */\n  shouldHandleDeletions: boolean;\n\n  /**\n   * Whether to handle renames.\n   */\n  shouldHandleRenames: boolean;\n\n  /**\n   * Whether to rename attachment files when a note is renamed.\n   */\n  shouldRenameAttachmentFiles: boolean;\n\n  /**\n   * Whether to rename attachment folder when a note is renamed.\n   */\n  shouldRenameAttachmentFolder: boolean;\n\n  /**\n   * Whether to update file name aliases when a note is renamed.\n   */\n  shouldUpdateFileNameAliases: boolean;\n}\n\ninterface AbortablePlugin extends Plugin {\n  abortSignal?: AbortSignal;\n}\n\ninterface HandledRenameKey {\n  newPath: string;\n  oldPath: string;\n}\n\ninterface InterruptedRename {\n  combinedBacklinksMap: Map<string, Map<string, string>>;\n  oldPath: string;\n}\n\ninterface RenameHandlerParams {\n  readonly abortSignal: AbortSignal;\n  readonly app: App;\n  readonly handledRenames: HandledRenames;\n  readonly interruptedCombinedBacklinksMap?: Map<string, Map<string, string>>;\n  readonly interruptedRenamesMap: Map<string, InterruptedRename[]>;\n  readonly newPath: string;\n  readonly oldCache: CachedMetadata | null;\n  readonly oldPath: string;\n  readonly oldPathBacklinksMap: Map<string, Reference[]>;\n  readonly settingsManager: SettingsManager;\n}\n\ninterface RenameMapParams {\n  readonly abortSignal: AbortSignal;\n  readonly app: App;\n  readonly newPath: string;\n  readonly oldCache: CachedMetadata | null;\n  readonly oldPath: string;\n  readonly settingsManager: SettingsManager;\n}\n\ntype RunAsyncLinkUpdateFn = { renameDeleteHandlerPatched?: boolean } & FileManager['runAsyncLinkUpdate'];\n\nclass DeleteHandler {\n  public constructor(\n    private readonly app: App,\n    private readonly file: TAbstractFile,\n    private readonly abortSignal: AbortSignal,\n    private readonly settingsManager: SettingsManager,\n    private readonly deletedMetadataCacheMap: Map<string, CachedMetadata>\n  ) {\n  }\n\n  public async handle(): Promise<void> {\n    this.abortSignal.throwIfAborted();\n    getLibDebugger('RenameDeleteHandler:handleDelete')(`Handle Delete ${this.file.path}`);\n    if (!isNote(this.app, this.file)) {\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n\n    if (settings.isPathIgnored?.(this.file.path)) {\n      getLibDebugger('RenameDeleteHandler:handleDelete')(`Skipping delete handler of ${this.file.path} as the path is ignored.`);\n      return;\n    }\n\n    const parentFolderPaths = new Set<string>([dirname(this.file.path)]);\n\n    if (settings.shouldHandleDeletions) {\n      const cache = this.deletedMetadataCacheMap.get(this.file.path);\n      this.deletedMetadataCacheMap.delete(this.file.path);\n      if (cache) {\n        const links = getAllLinks(cache);\n\n        for (const link of links) {\n          const attachmentFile = extractLinkFile(this.app, link, this.file.path);\n          if (!attachmentFile) {\n            continue;\n          }\n\n          if (this.settingsManager.isNoteEx(attachmentFile.path)) {\n            continue;\n          }\n\n          parentFolderPaths.add(attachmentFile.parent?.path ?? '');\n          await deleteSafe(this.app, attachmentFile, this.file.path, false, settings.emptyFolderBehavior !== EmptyFolderBehavior.Keep);\n          this.abortSignal.throwIfAborted();\n        }\n      }\n    }\n\n    parentFolderPaths.delete('');\n    await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths));\n    this.abortSignal.throwIfAborted();\n\n    if (!settings.shouldHandleDeletions) {\n      return;\n    }\n\n    const attachmentFolderPath = await getAttachmentFolderPath(this.app, this.file.path, AttachmentPathContext.DeleteNote);\n    const attachmentFolder = getFolderOrNull(this.app, attachmentFolderPath);\n\n    if (!attachmentFolder) {\n      return;\n    }\n\n    if (!await hasOwnAttachmentFolder(this.app, this.file.path, AttachmentPathContext.DeleteNote)) {\n      return;\n    }\n\n    this.abortSignal.throwIfAborted();\n\n    await deleteSafe(this.app, attachmentFolder, this.file.path, false, settings.emptyFolderBehavior !== EmptyFolderBehavior.Keep);\n    this.abortSignal.throwIfAborted();\n  }\n}\n\nclass HandledRenames {\n  private readonly map = new Map<string, HandledRenameKey>();\n\n  public add(oldPath: string, newPath: string): void {\n    this.map.set(this.keyToString(oldPath, newPath), { newPath, oldPath });\n  }\n\n  public delete(oldPath: string, newPath: string): void {\n    this.map.delete(this.keyToString(oldPath, newPath));\n  }\n\n  public has(oldPath: string, newPath: string): boolean {\n    return this.map.has(this.keyToString(oldPath, newPath));\n  }\n\n  public keys(): IterableIterator<HandledRenameKey> {\n    return this.map.values();\n  }\n\n  private keyToString(oldPath: string, newPath: string): string {\n    return `${oldPath} -> ${newPath}`;\n  }\n}\n\nclass MetadataDeletedHandler {\n  public constructor(\n    private readonly app: App,\n    private readonly file: TAbstractFile,\n    private readonly prevCache: CachedMetadata | null,\n    private readonly settingsManager: SettingsManager,\n    private readonly deletedMetadataCacheMap: Map<string, CachedMetadata>\n  ) {\n  }\n\n  public handle(): void {\n    const settings = this.settingsManager.getSettings();\n\n    if (!settings.shouldHandleDeletions) {\n      return;\n    }\n\n    if (settings.isPathIgnored?.(this.file.path)) {\n      getLibDebugger('RenameDeleteHandler:handleMetadataDeleted')(`Skipping metadata delete handler of ${this.file.path} as the path is ignored.`);\n      return;\n    }\n\n    if (isMarkdownFile(this.app, this.file) && this.prevCache) {\n      this.deletedMetadataCacheMap.set(this.file.path, this.prevCache);\n    }\n  }\n}\n\nclass Registry {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly deletedMetadataCacheMap = new Map<string, CachedMetadata>();\n  private readonly handledRenames = new HandledRenames();\n  private readonly interruptedRenamesMap = new Map<string, InterruptedRename[]>();\n  private readonly pluginId: string;\n\n  public constructor(\n    private readonly plugin: AbortablePlugin,\n    private readonly settingsBuilder: () => Partial<RenameDeleteHandlerSettings>,\n    private readonly settingsManager: SettingsManager\n  ) {\n    this.app = plugin.app;\n    this.pluginId = plugin.manifest.id;\n    this.abortSignal = plugin.abortSignal ?? abortSignalNever();\n  }\n\n  public register(): void {\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n\n    renameDeleteHandlersMap.set(this.pluginId, this.settingsBuilder);\n    this.logRegisteredHandlers();\n\n    this.plugin.register(() => {\n      renameDeleteHandlersMap.delete(this.pluginId);\n      this.logRegisteredHandlers();\n    });\n\n    this.plugin.registerEvent(this.app.vault.on('delete', this.handleDelete.bind(this)));\n    this.plugin.registerEvent(this.app.vault.on('rename', this.handleRename.bind(this)));\n    this.plugin.registerEvent(this.app.metadataCache.on('deleted', this.handleMetadataDeleted.bind(this)));\n\n    registerPatch(this.plugin, this.app.fileManager, {\n      runAsyncLinkUpdate: (next: RunAsyncLinkUpdateFn): RunAsyncLinkUpdateFn => {\n        return Object.assign((linkUpdatesHandler) => this.runAsyncLinkUpdate(next, linkUpdatesHandler), { renameDeleteHandlerPatched: true });\n      }\n    });\n  }\n\n  private handleDelete(file: TAbstractFile): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n    addToQueue({\n      app: this.app,\n      operationFn: (abortSignal) => new DeleteHandler(this.app, file, abortSignal, this.settingsManager, this.deletedMetadataCacheMap).handle(),\n      operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleDelete, { filePath: file.path })\n    });\n  }\n\n  private handleMetadataDeleted(file: TAbstractFile, prevCache: CachedMetadata | null): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n    new MetadataDeletedHandler(this.app, file, prevCache, this.settingsManager, this.deletedMetadataCacheMap).handle();\n  }\n\n  private handleRename(file: TAbstractFile, oldPath: string): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n\n    if (!isFile(file)) {\n      return;\n    }\n\n    const newPath = file.path;\n\n    getLibDebugger('RenameDeleteHandler:handleRename')(`Handle Rename ${oldPath} -> ${newPath}`);\n    if (this.handledRenames.has(oldPath, newPath)) {\n      this.handledRenames.delete(oldPath, newPath);\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n    if (!settings.shouldHandleRenames) {\n      return;\n    }\n\n    if (settings.isPathIgnored?.(oldPath)) {\n      getLibDebugger('RenameDeleteHandler:handleRename')(`Skipping rename handler of old path ${oldPath} as the path is ignored.`);\n      return;\n    }\n\n    if (settings.isPathIgnored?.(newPath)) {\n      getLibDebugger('RenameDeleteHandler:handleRename')(`Skipping rename handler of new path ${newPath} as the path is ignored.`);\n      return;\n    }\n\n    const oldCache = this.app.metadataCache.getCache(oldPath) ?? this.app.metadataCache.getCache(newPath);\n    const oldPathBacklinksMap = getBacklinksForFileOrPath(this.app, oldPath).data;\n    addToQueue({\n      abortSignal: this.abortSignal,\n      app: this.app,\n      operationFn: (abortSignal) =>\n        new RenameHandler({\n          abortSignal,\n          app: this.app,\n          handledRenames: this.handledRenames,\n          interruptedRenamesMap: this.interruptedRenamesMap,\n          newPath,\n          oldCache,\n          oldPath,\n          oldPathBacklinksMap,\n          settingsManager: this.settingsManager\n        }).handle(),\n      operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleRename, { newPath, oldPath })\n    });\n  }\n\n  private logRegisteredHandlers(): void {\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n    getLibDebugger('RenameDeleteHandler:logRegisteredHandlers')(\n      `Plugins with registered rename/delete handlers: ${JSON.stringify(Array.from(renameDeleteHandlersMap.keys()))}`\n    );\n  }\n\n  private async runAsyncLinkUpdate(next: RunAsyncLinkUpdateFn, linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n    if (next.renameDeleteHandlerPatched) {\n      await next.call(this.app.fileManager, linkUpdatesHandler);\n      return;\n    }\n    await next.call(this.app.fileManager, (linkUpdates) => this.wrapLinkUpdatesHandler(linkUpdates, linkUpdatesHandler));\n  }\n\n  private shouldInvokeHandler(): boolean {\n    const pluginId = this.plugin.manifest.id;\n\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n    const mainPluginId = Array.from(renameDeleteHandlersMap.keys())[0];\n    return mainPluginId === pluginId;\n  }\n\n  private async wrapLinkUpdatesHandler(linkUpdates: LinkUpdate[], linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n    let isRenameCalled = false;\n    const eventRef = this.app.vault.on('rename', () => {\n      isRenameCalled = true;\n    });\n    try {\n      await linkUpdatesHandler(linkUpdates);\n    } finally {\n      this.app.vault.offref(eventRef);\n    }\n    const settings = this.settingsManager.getSettings();\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- It might changed in `rename` event handler. ESLint mistakenly does not recognize it.\n    if (!isRenameCalled || !settings.shouldHandleRenames) {\n      return;\n    }\n\n    filterInPlace(\n      linkUpdates,\n      (linkUpdate) => {\n        if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {\n          getLibDebugger('RenameDeleteHandler:runAsyncLinkUpdate')(\n            `Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`\n          );\n          return true;\n        }\n\n        if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {\n          getLibDebugger('RenameDeleteHandler:runAsyncLinkUpdate')(\n            `Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`\n          );\n          return true;\n        }\n\n        if (!this.app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n          return false;\n        }\n\n        if (this.app.plugins.getPlugin('backlink-cache')) {\n          return false;\n        }\n\n        if (linkUpdate.sourceFile.extension === CANVAS_FILE_EXTENSION) {\n          return true;\n        }\n\n        if (linkUpdate.resolvedFile.extension === CANVAS_FILE_EXTENSION) {\n          return true;\n        }\n\n        return false;\n      }\n    );\n  }\n}\n\nclass SettingsManager {\n  public readonly renameDeleteHandlersMap: Map<string, () => Partial<RenameDeleteHandlerSettings>>;\n\n  public constructor(private readonly app: App) {\n    this.renameDeleteHandlersMap =\n      getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n  }\n\n  public getSettings(): Partial<RenameDeleteHandlerSettings> {\n    const settingsBuilders = Array.from(this.renameDeleteHandlersMap.values()).reverse();\n\n    const settings: Partial<RenameDeleteHandlerSettings> = {};\n    settings.isNote = (path: string): boolean => isNote(this.app, path);\n    settings.isPathIgnored = (): boolean => false;\n\n    for (const settingsBuilder of settingsBuilders) {\n      const newSettings = settingsBuilder();\n      settings.shouldDeleteConflictingAttachments ||= newSettings.shouldDeleteConflictingAttachments ?? false;\n      if (newSettings.emptyFolderBehavior) {\n        settings.emptyFolderBehavior ??= newSettings.emptyFolderBehavior;\n      }\n      settings.shouldHandleDeletions ||= newSettings.shouldHandleDeletions ?? false;\n      settings.shouldHandleRenames ||= newSettings.shouldHandleRenames ?? false;\n      settings.shouldRenameAttachmentFiles ||= newSettings.shouldRenameAttachmentFiles ?? false;\n      settings.shouldRenameAttachmentFolder ||= newSettings.shouldRenameAttachmentFolder ?? false;\n      settings.shouldUpdateFileNameAliases ||= newSettings.shouldUpdateFileNameAliases ?? false;\n      const isPathIgnored = settings.isPathIgnored;\n      settings.isPathIgnored = (path: string): boolean => isPathIgnored(path) || (newSettings.isPathIgnored?.(path) ?? false);\n      const currentIsNote = settings.isNote;\n      settings.isNote = (path: string): boolean => currentIsNote(path) && (newSettings.isNote?.(path) ?? true);\n    }\n\n    settings.emptyFolderBehavior ??= EmptyFolderBehavior.Keep;\n    return settings;\n  }\n\n  public isNoteEx(path: string): boolean {\n    const settings = this.getSettings();\n    return settings.isNote?.(path) ?? false;\n  }\n}\n\nclass RenameHandler {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly handledRenames: HandledRenames;\n  private readonly interruptedCombinedBacklinksMap: Map<string, Map<string, string>>;\n  private readonly interruptedRenamesMap: Map<string, InterruptedRename[]>;\n  private readonly newPath: string;\n  private readonly oldCache: CachedMetadata | null;\n  private readonly oldPath: string;\n  private readonly oldPathBacklinksMap: Map<string, Reference[]>;\n  private readonly oldPathLinks: Reference[];\n  private readonly settingsManager: SettingsManager;\n\n  public constructor(params: RenameHandlerParams) {\n    this.app = params.app;\n    this.oldPath = params.oldPath;\n    this.newPath = params.newPath;\n    this.oldPathBacklinksMap = params.oldPathBacklinksMap;\n    this.oldCache = params.oldCache;\n    this.abortSignal = params.abortSignal;\n    this.settingsManager = params.settingsManager;\n    this.interruptedRenamesMap = params.interruptedRenamesMap;\n    this.oldPathLinks = this.oldCache ? getAllLinks(this.oldCache) : [];\n    this.handledRenames = params.handledRenames;\n    this.interruptedCombinedBacklinksMap = params.interruptedCombinedBacklinksMap ?? new Map<string, Map<string, string>>();\n  }\n\n  public async handle(): Promise<void> {\n    if (this.oldPath === this.newPath) {\n      return;\n    }\n    this.abortSignal.throwIfAborted();\n    await this.continueInterruptedRenames();\n    this.abortSignal.throwIfAborted();\n    await this.refreshLinks();\n    this.abortSignal.throwIfAborted();\n    if (await this.handleCaseCollision()) {\n      return;\n    }\n\n    this.abortSignal.throwIfAborted();\n\n    const renamedFilePaths = getObsidianDevUtilsState(this.app, 'renamedFilePaths', new Set<string>()).value;\n    const renamedLinks = getObsidianDevUtilsState(this.app, 'renamedLinkPaths', new Set<string>()).value;\n\n    try {\n      const renameMap = new RenameMap({\n        abortSignal: this.abortSignal,\n        app: this.app,\n        newPath: this.newPath,\n        oldCache: this.oldCache,\n        oldPath: this.oldPath,\n        settingsManager: this.settingsManager\n      });\n      await renameMap.fill();\n      this.abortSignal.throwIfAborted();\n\n      const combinedBacklinksMap = new Map<string, Map<string, string>>();\n      renameMap.initOriginalLinksMap(combinedBacklinksMap);\n      renameMap.initBacklinksMap(this.oldPathBacklinksMap, combinedBacklinksMap, this.oldPath);\n\n      for (const attachmentOldPath of renameMap.keys()) {\n        if (attachmentOldPath === this.oldPath) {\n          continue;\n        }\n        const attachmentOldPathBacklinksMap = (await getBacklinksForFileSafe(this.app, attachmentOldPath)).data;\n        this.abortSignal.throwIfAborted();\n        renameMap.initBacklinksMap(attachmentOldPathBacklinksMap, combinedBacklinksMap, attachmentOldPath);\n      }\n\n      const parentFolderPaths = new Set<string>();\n\n      for (const [oldAttachmentPath, newAttachmentPath] of renameMap.entries()) {\n        if (oldAttachmentPath !== this.oldPath) {\n          const fixedNewAttachmentPath = await this.renameHandled(oldAttachmentPath, newAttachmentPath);\n          this.abortSignal.throwIfAborted();\n          renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);\n        }\n        if (!this.settingsManager.isNoteEx(oldAttachmentPath)) {\n          parentFolderPaths.add(dirname(oldAttachmentPath));\n        }\n      }\n\n      await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths));\n      this.abortSignal.throwIfAborted();\n      const settings = this.settingsManager.getSettings();\n\n      for (\n        const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(\n          Array.from(this.interruptedCombinedBacklinksMap.entries())\n        )\n      ) {\n        let linkIndex = 0;\n        await editLinks(this.app, newBacklinkPath, (link) => {\n          linkIndex++;\n          const oldAttachmentPath = linkJsonToPathMap.get(toJson(link));\n          if (!oldAttachmentPath) {\n            return;\n          }\n\n          const newAttachmentPath = renameMap.get(oldAttachmentPath) ?? oldAttachmentPath;\n\n          renamedFilePaths.add(newBacklinkPath);\n          renamedLinks.add(`${newBacklinkPath}//${String(linkIndex)}`);\n\n          return updateLink(normalizeOptionalProperties<UpdateLinkParams>({\n            app: this.app,\n            link,\n            newSourcePathOrFile: newBacklinkPath,\n            newTargetPathOrFile: newAttachmentPath,\n            oldTargetPathOrFile: oldAttachmentPath,\n            shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n          }));\n        }, {\n          shouldFailOnMissingFile: false\n        });\n        this.abortSignal.throwIfAborted();\n      }\n\n      if (isNote(this.app, this.newPath)) {\n        await updateLinksInFile(normalizeOptionalProperties<UpdateLinksInFileParams>({\n          app: this.app,\n          newSourcePathOrFile: this.newPath,\n          oldSourcePathOrFile: this.oldPath,\n          shouldFailOnMissingFile: false,\n          shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n        }));\n        this.abortSignal.throwIfAborted();\n      }\n\n      if (!getFileOrNull(this.app, this.newPath)) {\n        let interruptedRenames = this.interruptedRenamesMap.get(this.newPath);\n        if (!interruptedRenames) {\n          interruptedRenames = [];\n          this.interruptedRenamesMap.set(this.newPath, interruptedRenames);\n        }\n        interruptedRenames.push({\n          combinedBacklinksMap,\n          oldPath: this.oldPath\n        });\n      }\n    } finally {\n      const orphanKeys = Array.from(this.handledRenames.keys());\n      addToQueue({\n        abortSignal: this.abortSignal,\n        app: this.app,\n        operationFn: () => {\n          for (const orphanKey of orphanKeys) {\n            this.handledRenames.delete(orphanKey.oldPath, orphanKey.newPath);\n          }\n\n          if (renamedLinks.size === 0) {\n            return;\n          }\n          new Notice(t(($) => $.obsidianDevUtils.renameDeleteHandler.updatedLinks, { filesCount: renamedFilePaths.size, linksCount: renamedLinks.size }));\n          renamedFilePaths.clear();\n          renamedLinks.clear();\n        },\n        operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleOrphanedRenames)\n      });\n    }\n  }\n\n  private async continueInterruptedRenames(): Promise<void> {\n    const interruptedRenames = this.interruptedRenamesMap.get(this.oldPath);\n    if (interruptedRenames) {\n      this.interruptedRenamesMap.delete(this.oldPath);\n      for (const interruptedRename of interruptedRenames) {\n        await new RenameHandler({\n          abortSignal: this.abortSignal,\n          app: this.app,\n          handledRenames: this.handledRenames,\n          interruptedCombinedBacklinksMap: interruptedRename.combinedBacklinksMap,\n          interruptedRenamesMap: this.interruptedRenamesMap,\n          newPath: this.newPath,\n          oldCache: this.oldCache,\n          oldPath: interruptedRename.oldPath,\n          oldPathBacklinksMap: this.oldPathBacklinksMap,\n          settingsManager: this.settingsManager\n        }).handle();\n      }\n    }\n  }\n\n  private async handleCaseCollision(): Promise<boolean> {\n    if (!this.app.vault.adapter.insensitive || this.oldPath.toLowerCase() !== this.newPath.toLowerCase()) {\n      return false;\n    }\n\n    const tempPath = join(dirname(this.newPath), `__temp__${basename(this.newPath)}`);\n    await this.renameHandled(this.newPath, tempPath);\n\n    await new RenameHandler({\n      abortSignal: this.abortSignal,\n      app: this.app,\n      handledRenames: this.handledRenames,\n      interruptedRenamesMap: this.interruptedRenamesMap,\n      newPath: tempPath,\n      oldCache: this.oldCache,\n      oldPath: this.oldPath,\n      oldPathBacklinksMap: this.oldPathBacklinksMap,\n      settingsManager: this.settingsManager\n    }).handle();\n\n    await this.app.fileManager.renameFile(getFile(this.app, tempPath), this.newPath);\n    return true;\n  }\n\n  private async refreshLinks(): Promise<void> {\n    const cache = this.app.metadataCache.getCache(this.oldPath) ?? this.app.metadataCache.getCache(this.newPath);\n    const oldPathLinksRefreshed = cache ? getAllLinks(cache) : [];\n    const fakeOldFile = getFile(this.app, this.oldPath, true);\n    let oldPathBacklinksMapRefreshed = new Map<string, Reference[]>();\n    await tempRegisterFilesAndRun(this.app, [fakeOldFile], async () => {\n      oldPathBacklinksMapRefreshed = (await getBacklinksForFileSafe(this.app, fakeOldFile)).data;\n    });\n\n    for (const link of oldPathLinksRefreshed) {\n      if (this.oldPathLinks.includes(link)) {\n        continue;\n      }\n      this.oldPathLinks.push(link);\n    }\n\n    for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {\n      let oldLinks = this.oldPathBacklinksMap.get(backlinkPath);\n      if (!oldLinks) {\n        oldLinks = [];\n        this.oldPathBacklinksMap.set(backlinkPath, oldLinks);\n      }\n\n      for (const link of refreshedLinks) {\n        if (oldLinks.includes(link)) {\n          continue;\n        }\n        oldLinks.push(link);\n      }\n    }\n  }\n\n  private async renameHandled(oldPath: string, newPath: string): Promise<string> {\n    newPath = getSafeRenamePath(this.app, oldPath, newPath);\n    if (oldPath === newPath) {\n      return newPath;\n    }\n    this.handledRenames.add(oldPath, newPath);\n    newPath = await renameSafe(this.app, oldPath, newPath);\n    return newPath;\n  }\n}\n\nclass RenameMap {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly map = new Map<string, string>();\n  private readonly newPath: string;\n  private readonly oldCache: CachedMetadata | null;\n  private readonly oldPath: string;\n  private readonly oldPathLinks: Reference[];\n  private readonly settingsManager: SettingsManager;\n\n  public constructor(params: RenameMapParams) {\n    this.abortSignal = params.abortSignal;\n    this.app = params.app;\n    this.settingsManager = params.settingsManager;\n    this.oldCache = params.oldCache;\n    this.oldPath = params.oldPath;\n    this.newPath = params.newPath;\n    this.oldPathLinks = this.oldCache ? getAllLinks(this.oldCache) : [];\n  }\n\n  public entries(): IterableIterator<[string, string]> {\n    return this.map.entries();\n  }\n\n  public async fill(): Promise<void> {\n    this.abortSignal.throwIfAborted();\n    this.map.set(this.oldPath, this.newPath);\n\n    if (!isNote(this.app, this.oldPath)) {\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n\n    const oldFile = getFile(this.app, this.oldPath, true);\n    let oldAttachmentFolderPath = '';\n    await tempRegisterFilesAndRunAsync(this.app, [oldFile], async () => {\n      const shouldFakeOldPathCache = this.oldCache && oldFile.deleted;\n      if (shouldFakeOldPathCache) {\n        registerFileCacheForNonExistingFile(this.app, oldFile, this.oldCache);\n      }\n\n      try {\n        oldAttachmentFolderPath = await getAttachmentFolderPath(this.app, this.oldPath, AttachmentPathContext.RenameNote);\n      } finally {\n        if (shouldFakeOldPathCache) {\n          unregisterFileCacheForNonExistingFile(this.app, oldFile);\n        }\n      }\n    });\n\n    const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n      ? await getAttachmentFolderPath(this.app, this.newPath, AttachmentPathContext.RenameNote)\n      : oldAttachmentFolderPath;\n\n    const isOldAttachmentFolderAtRoot = oldAttachmentFolderPath === '/';\n\n    const oldAttachmentFolder = getFolderOrNull(this.app, oldAttachmentFolderPath);\n\n    if (!oldAttachmentFolder) {\n      return;\n    }\n\n    if (oldAttachmentFolderPath === newAttachmentFolderPath && !settings.shouldRenameAttachmentFiles) {\n      return;\n    }\n\n    const oldAttachmentFiles: TFile[] = [];\n\n    if (await hasOwnAttachmentFolder(this.app, this.oldPath, AttachmentPathContext.RenameNote)) {\n      Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {\n        this.abortSignal.throwIfAborted();\n        if (isFile(oldAttachmentFile)) {\n          oldAttachmentFiles.push(oldAttachmentFile);\n        }\n      });\n    } else {\n      for (const oldPathLink of this.oldPathLinks) {\n        this.abortSignal.throwIfAborted();\n        const oldAttachmentFile = extractLinkFile(this.app, oldPathLink, this.oldPath);\n        if (!oldAttachmentFile) {\n          continue;\n        }\n\n        if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n          const oldAttachmentBacklinks = await getBacklinksForFileSafe(this.app, oldAttachmentFile);\n          this.abortSignal.throwIfAborted();\n          const keys = new Set<string>(oldAttachmentBacklinks.keys());\n          keys.delete(this.oldPath);\n          keys.delete(this.newPath);\n          if (keys.size === 0) {\n            oldAttachmentFiles.push(oldAttachmentFile);\n          }\n        }\n      }\n    }\n\n    for (const oldAttachmentFile of oldAttachmentFiles) {\n      this.abortSignal.throwIfAborted();\n      if (this.settingsManager.isNoteEx(oldAttachmentFile.path)) {\n        continue;\n      }\n\n      let newAttachmentFilePath: string;\n      if (settings.shouldRenameAttachmentFiles) {\n        newAttachmentFilePath = await getAttachmentFilePath({\n          app: this.app,\n          context: AttachmentPathContext.RenameNote,\n          notePathOrFile: this.newPath,\n          oldAttachmentPathOrFile: oldAttachmentFile,\n          oldNotePathOrFile: this.oldPath,\n          shouldSkipDuplicateCheck: true\n        });\n        this.abortSignal.throwIfAborted();\n      } else {\n        const relativePath = isOldAttachmentFolderAtRoot ? oldAttachmentFile.path : relative(oldAttachmentFolderPath, oldAttachmentFile.path);\n        const newFolder = join(newAttachmentFolderPath, dirname(relativePath));\n        newAttachmentFilePath = join(newFolder, oldAttachmentFile.name);\n      }\n\n      if (oldAttachmentFile.path === newAttachmentFilePath) {\n        continue;\n      }\n      if (settings.shouldDeleteConflictingAttachments) {\n        const newAttachmentFile = getFileOrNull(this.app, newAttachmentFilePath);\n        if (newAttachmentFile) {\n          getLibDebugger('RenameDeleteHandler:fillRenameMap')(`Removing conflicting attachment ${newAttachmentFile.path}.`);\n          await this.app.fileManager.trashFile(newAttachmentFile);\n          this.abortSignal.throwIfAborted();\n        }\n      } else {\n        const dir = dirname(newAttachmentFilePath);\n        const ext = extname(newAttachmentFilePath);\n        const baseName = basename(newAttachmentFilePath, ext);\n        newAttachmentFilePath = this.app.vault.getAvailablePath(join(dir, baseName), ext.slice(1));\n      }\n      this.map.set(oldAttachmentFile.path, newAttachmentFilePath);\n    }\n  }\n\n  public get(oldPath: string): string | undefined {\n    return this.map.get(oldPath);\n  }\n\n  public initBacklinksMap(\n    singleBacklinksMap: Map<string, Reference[]>,\n    combinedBacklinksMap: Map<string, Map<string, string>>,\n    path: string\n  ): void {\n    for (const [backlinkPath, links] of singleBacklinksMap.entries()) {\n      const newBacklinkPath = this.map.get(backlinkPath) ?? backlinkPath;\n      const linkJsonToPathMap = combinedBacklinksMap.get(newBacklinkPath) ?? new Map<string, string>();\n      combinedBacklinksMap.set(newBacklinkPath, linkJsonToPathMap);\n      for (const link of links) {\n        linkJsonToPathMap.set(toJson(link), path);\n      }\n    }\n  }\n\n  public initOriginalLinksMap(combinedBacklinksMap: Map<string, Map<string, string>>): void {\n    for (const oldPathLink of this.oldPathLinks) {\n      const oldAttachmentFile = extractLinkFile(this.app, oldPathLink, this.oldPath);\n      if (!oldAttachmentFile) {\n        continue;\n      }\n      const backlinksMap = new Map<string, Reference[]>();\n      backlinksMap.set(this.newPath, [oldPathLink]);\n      this.initBacklinksMap(backlinksMap, combinedBacklinksMap, oldAttachmentFile.path);\n    }\n  }\n\n  public keys(): IterableIterator<string> {\n    return this.map.keys();\n  }\n\n  public set(oldPath: string, newPath: string): void {\n    this.map.set(oldPath, newPath);\n  }\n}\n\n/**\n * Registers the rename/delete handlers.\n *\n * @param plugin - The plugin instance.\n * @param settingsBuilder - A function that returns the settings for the rename delete handler.\n */\nexport function registerRenameDeleteHandlers(plugin: AbortablePlugin, settingsBuilder: () => Partial<RenameDeleteHandlerSettings>): void {\n  new Registry(plugin, settingsBuilder, new SettingsManager(plugin.app)).register();\n}\n\nasync function cleanupParentFolders(app: App, settings: Partial<RenameDeleteHandlerSettings>, parentFolderPaths: string[]): Promise<void> {\n  if (settings.emptyFolderBehavior === EmptyFolderBehavior.Keep) {\n    return;\n  }\n  for (const parentFolderPath of parentFolderPaths) {\n    switch (settings.emptyFolderBehavior) {\n      case EmptyFolderBehavior.Delete:\n        await deleteEmptyFolder(app, parentFolderPath);\n        break;\n      case EmptyFolderBehavior.DeleteWithEmptyParents:\n        await deleteEmptyFolderHierarchy(app, parentFolderPath);\n        break;\n      default:\n        break;\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["EmptyFolderBehavior"]
}

|
|
774
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/RenameDeleteHandler.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for handling rename and delete events in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type {\n  App,\n  CachedMetadata,\n  FileManager,\n  Plugin,\n  Reference,\n  TAbstractFile,\n  TFile\n} from 'obsidian';\nimport type {\n  LinkUpdate,\n  LinkUpdatesHandler\n} from 'obsidian-typings';\n\nimport { t } from 'i18next';\nimport {\n  Notice,\n  Vault\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type {\n  UpdateLinkParams,\n  UpdateLinksInFileParams\n} from './Link.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport { filterInPlace } from '../Array.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport {\n  AttachmentPathContext,\n  getAttachmentFilePath,\n  getAttachmentFolderPath,\n  hasOwnAttachmentFolder\n} from './AttachmentPath.ts';\nimport {\n  CANVAS_FILE_EXTENSION,\n  getFile,\n  getFileOrNull,\n  getFolderOrNull,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport {\n  editLinks,\n  extractLinkFile,\n  updateLink,\n  updateLinksInFile\n} from './Link.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileOrPath,\n  getBacklinksForFileSafe,\n  registerFileCacheForNonExistingFile,\n  tempRegisterFilesAndRun,\n  tempRegisterFilesAndRunAsync,\n  unregisterFileCacheForNonExistingFile\n} from './MetadataCache.ts';\nimport { registerPatch } from './MonkeyAround.ts';\nimport { addToQueue } from './Queue.ts';\nimport {\n  deleteEmptyFolder,\n  deleteEmptyFolderHierarchy,\n  getSafeRenamePath,\n  renameSafe,\n  trashSafe\n} from './Vault.ts';\nimport { deleteIfNotUsed } from './VaultDelete.ts';\n\n/**\n * A behavior of the rename/delete handler when deleting empty folders.\n */\nexport enum EmptyFolderBehavior {\n  /**\n   * Delete the empty folder.\n   */\n  Delete = 'Delete',\n\n  /**\n   * Delete the empty folder and all its empty parents.\n   */\n  DeleteWithEmptyParents = 'DeleteWithEmptyParents',\n\n  /**\n   * Keep the empty folder.\n   */\n  Keep = 'Keep'\n}\n\n/**\n * Settings for the rename/delete handler.\n */\nexport interface RenameDeleteHandlerSettings {\n  /**\n   * A behavior of the rename/delete handler when deleting empty folders.\n   */\n  emptyFolderBehavior: EmptyFolderBehavior;\n\n  /**\n   * Whether the path is a note.\n   */\n  isNote(path: string): boolean;\n\n  /**\n   * Whether to ignore the path.\n   */\n  isPathIgnored(path: string): boolean;\n\n  /**\n   * Whether to delete conflicting attachments.\n   */\n  shouldDeleteConflictingAttachments: boolean;\n\n  /**\n   * Whether to handle deletions.\n   */\n  shouldHandleDeletions: boolean;\n\n  /**\n   * Whether to handle renames.\n   */\n  shouldHandleRenames: boolean;\n\n  /**\n   * Whether to rename attachment files when a note is renamed.\n   */\n  shouldRenameAttachmentFiles: boolean;\n\n  /**\n   * Whether to rename attachment folder when a note is renamed.\n   */\n  shouldRenameAttachmentFolder: boolean;\n\n  /**\n   * Whether to update file name aliases when a note is renamed.\n   */\n  shouldUpdateFileNameAliases: boolean;\n}\n\ninterface AbortablePlugin extends Plugin {\n  abortSignal?: AbortSignal;\n}\n\ninterface HandledRenameKey {\n  newPath: string;\n  oldPath: string;\n}\n\ninterface InterruptedRename {\n  combinedBacklinksMap: Map<string, Map<string, string>>;\n  oldPath: string;\n}\n\ninterface RenameHandlerParams {\n  readonly abortSignal: AbortSignal;\n  readonly app: App;\n  readonly handledRenames: HandledRenames;\n  readonly interruptedCombinedBacklinksMap?: Map<string, Map<string, string>>;\n  readonly interruptedRenamesMap: Map<string, InterruptedRename[]>;\n  readonly newPath: string;\n  readonly oldCache: CachedMetadata | null;\n  readonly oldPath: string;\n  readonly oldPathBacklinksMap: Map<string, Reference[]>;\n  readonly settingsManager: SettingsManager;\n}\n\ninterface RenameMapParams {\n  readonly abortSignal: AbortSignal;\n  readonly app: App;\n  readonly newPath: string;\n  readonly oldCache: CachedMetadata | null;\n  readonly oldPath: string;\n  readonly settingsManager: SettingsManager;\n}\n\ntype RunAsyncLinkUpdateFn = { renameDeleteHandlerPatched?: boolean } & FileManager['runAsyncLinkUpdate'];\n\nclass DeleteHandler {\n  public constructor(\n    private readonly app: App,\n    private readonly file: TAbstractFile,\n    private readonly abortSignal: AbortSignal,\n    private readonly settingsManager: SettingsManager,\n    private readonly deletedMetadataCacheMap: Map<string, CachedMetadata>\n  ) {\n  }\n\n  public async handle(): Promise<void> {\n    this.abortSignal.throwIfAborted();\n    getLibDebugger('RenameDeleteHandler:handleDelete')(`Handle Delete ${this.file.path}`);\n    if (!isNote(this.app, this.file)) {\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n\n    if (settings.isPathIgnored?.(this.file.path)) {\n      getLibDebugger('RenameDeleteHandler:handleDelete')(`Skipping delete handler of ${this.file.path} as the path is ignored.`);\n      return;\n    }\n\n    const parentFolderPaths = new Set<string>([dirname(this.file.path)]);\n\n    if (settings.shouldHandleDeletions) {\n      const cache = this.deletedMetadataCacheMap.get(this.file.path);\n      this.deletedMetadataCacheMap.delete(this.file.path);\n      if (cache) {\n        const links = getAllLinks(cache);\n\n        for (const link of links) {\n          const attachmentFile = extractLinkFile(this.app, link, this.file.path);\n          if (!attachmentFile) {\n            continue;\n          }\n\n          if (this.settingsManager.isNoteEx(attachmentFile.path)) {\n            continue;\n          }\n\n          parentFolderPaths.add(attachmentFile.parent?.path ?? '');\n          await deleteIfNotUsed(this.app, attachmentFile, this.file.path, false, settings.emptyFolderBehavior !== EmptyFolderBehavior.Keep);\n          this.abortSignal.throwIfAborted();\n        }\n      }\n    }\n\n    parentFolderPaths.delete('');\n    await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths));\n    this.abortSignal.throwIfAborted();\n\n    if (!settings.shouldHandleDeletions) {\n      return;\n    }\n\n    const attachmentFolderPath = await getAttachmentFolderPath(this.app, this.file.path, AttachmentPathContext.DeleteNote);\n    const attachmentFolder = getFolderOrNull(this.app, attachmentFolderPath);\n\n    if (!attachmentFolder) {\n      return;\n    }\n\n    if (!await hasOwnAttachmentFolder(this.app, this.file.path, AttachmentPathContext.DeleteNote)) {\n      return;\n    }\n\n    this.abortSignal.throwIfAborted();\n\n    await deleteIfNotUsed(this.app, attachmentFolder, this.file.path, false, settings.emptyFolderBehavior !== EmptyFolderBehavior.Keep);\n    this.abortSignal.throwIfAborted();\n  }\n}\n\nclass HandledRenames {\n  private readonly map = new Map<string, HandledRenameKey>();\n\n  public add(oldPath: string, newPath: string): void {\n    this.map.set(this.keyToString(oldPath, newPath), { newPath, oldPath });\n  }\n\n  public delete(oldPath: string, newPath: string): void {\n    this.map.delete(this.keyToString(oldPath, newPath));\n  }\n\n  public has(oldPath: string, newPath: string): boolean {\n    return this.map.has(this.keyToString(oldPath, newPath));\n  }\n\n  public keys(): IterableIterator<HandledRenameKey> {\n    return this.map.values();\n  }\n\n  private keyToString(oldPath: string, newPath: string): string {\n    return `${oldPath} -> ${newPath}`;\n  }\n}\n\nclass MetadataDeletedHandler {\n  public constructor(\n    private readonly app: App,\n    private readonly file: TAbstractFile,\n    private readonly prevCache: CachedMetadata | null,\n    private readonly settingsManager: SettingsManager,\n    private readonly deletedMetadataCacheMap: Map<string, CachedMetadata>\n  ) {\n  }\n\n  public handle(): void {\n    const settings = this.settingsManager.getSettings();\n\n    if (!settings.shouldHandleDeletions) {\n      return;\n    }\n\n    if (settings.isPathIgnored?.(this.file.path)) {\n      getLibDebugger('RenameDeleteHandler:handleMetadataDeleted')(`Skipping metadata delete handler of ${this.file.path} as the path is ignored.`);\n      return;\n    }\n\n    if (isMarkdownFile(this.app, this.file) && this.prevCache) {\n      this.deletedMetadataCacheMap.set(this.file.path, this.prevCache);\n    }\n  }\n}\n\nclass Registry {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly deletedMetadataCacheMap = new Map<string, CachedMetadata>();\n  private readonly handledRenames = new HandledRenames();\n  private readonly interruptedRenamesMap = new Map<string, InterruptedRename[]>();\n  private readonly pluginId: string;\n\n  public constructor(\n    private readonly plugin: AbortablePlugin,\n    private readonly settingsBuilder: () => Partial<RenameDeleteHandlerSettings>,\n    private readonly settingsManager: SettingsManager\n  ) {\n    this.app = plugin.app;\n    this.pluginId = plugin.manifest.id;\n    this.abortSignal = plugin.abortSignal ?? abortSignalNever();\n  }\n\n  public register(): void {\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n\n    renameDeleteHandlersMap.set(this.pluginId, this.settingsBuilder);\n    this.logRegisteredHandlers();\n\n    this.plugin.register(() => {\n      renameDeleteHandlersMap.delete(this.pluginId);\n      this.logRegisteredHandlers();\n    });\n\n    this.plugin.registerEvent(this.app.vault.on('delete', this.handleDelete.bind(this)));\n    this.plugin.registerEvent(this.app.vault.on('rename', this.handleRename.bind(this)));\n    this.plugin.registerEvent(this.app.metadataCache.on('deleted', this.handleMetadataDeleted.bind(this)));\n\n    registerPatch(this.plugin, this.app.fileManager, {\n      runAsyncLinkUpdate: (next: RunAsyncLinkUpdateFn): RunAsyncLinkUpdateFn => {\n        return Object.assign((linkUpdatesHandler) => this.runAsyncLinkUpdate(next, linkUpdatesHandler), { renameDeleteHandlerPatched: true });\n      }\n    });\n  }\n\n  private handleDelete(file: TAbstractFile): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n    addToQueue({\n      app: this.app,\n      operationFn: (abortSignal) => new DeleteHandler(this.app, file, abortSignal, this.settingsManager, this.deletedMetadataCacheMap).handle(),\n      operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleDelete, { filePath: file.path })\n    });\n  }\n\n  private handleMetadataDeleted(file: TAbstractFile, prevCache: CachedMetadata | null): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n    new MetadataDeletedHandler(this.app, file, prevCache, this.settingsManager, this.deletedMetadataCacheMap).handle();\n  }\n\n  private handleRename(file: TAbstractFile, oldPath: string): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n\n    if (!isFile(file)) {\n      return;\n    }\n\n    const newPath = file.path;\n\n    getLibDebugger('RenameDeleteHandler:handleRename')(`Handle Rename ${oldPath} -> ${newPath}`);\n    if (this.handledRenames.has(oldPath, newPath)) {\n      this.handledRenames.delete(oldPath, newPath);\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n    if (!settings.shouldHandleRenames) {\n      return;\n    }\n\n    if (settings.isPathIgnored?.(oldPath)) {\n      getLibDebugger('RenameDeleteHandler:handleRename')(`Skipping rename handler of old path ${oldPath} as the path is ignored.`);\n      return;\n    }\n\n    if (settings.isPathIgnored?.(newPath)) {\n      getLibDebugger('RenameDeleteHandler:handleRename')(`Skipping rename handler of new path ${newPath} as the path is ignored.`);\n      return;\n    }\n\n    const oldCache = this.app.metadataCache.getCache(oldPath) ?? this.app.metadataCache.getCache(newPath);\n    const oldPathBacklinksMap = getBacklinksForFileOrPath(this.app, oldPath).data;\n    addToQueue({\n      abortSignal: this.abortSignal,\n      app: this.app,\n      operationFn: (abortSignal) =>\n        new RenameHandler({\n          abortSignal,\n          app: this.app,\n          handledRenames: this.handledRenames,\n          interruptedRenamesMap: this.interruptedRenamesMap,\n          newPath,\n          oldCache,\n          oldPath,\n          oldPathBacklinksMap,\n          settingsManager: this.settingsManager\n        }).handle(),\n      operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleRename, { newPath, oldPath })\n    });\n  }\n\n  private logRegisteredHandlers(): void {\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n    getLibDebugger('RenameDeleteHandler:logRegisteredHandlers')(\n      `Plugins with registered rename/delete handlers: ${JSON.stringify(Array.from(renameDeleteHandlersMap.keys()))}`\n    );\n  }\n\n  private async runAsyncLinkUpdate(next: RunAsyncLinkUpdateFn, linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n    if (next.renameDeleteHandlerPatched) {\n      await next.call(this.app.fileManager, linkUpdatesHandler);\n      return;\n    }\n    await next.call(this.app.fileManager, (linkUpdates) => this.wrapLinkUpdatesHandler(linkUpdates, linkUpdatesHandler));\n  }\n\n  private shouldInvokeHandler(): boolean {\n    const pluginId = this.plugin.manifest.id;\n\n    const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;\n    const mainPluginId = Array.from(renameDeleteHandlersMap.keys())[0];\n    return mainPluginId === pluginId;\n  }\n\n  private async wrapLinkUpdatesHandler(linkUpdates: LinkUpdate[], linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n    let isRenameCalled = false;\n    const eventRef = this.app.vault.on('rename', () => {\n      isRenameCalled = true;\n    });\n    try {\n      await linkUpdatesHandler(linkUpdates);\n    } finally {\n      this.app.vault.offref(eventRef);\n    }\n    const settings = this.settingsManager.getSettings();\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- It might changed in `rename` event handler. ESLint mistakenly does not recognize it.\n    if (!isRenameCalled || !settings.shouldHandleRenames) {\n      return;\n    }\n\n    filterInPlace(\n      linkUpdates,\n      (linkUpdate) => {\n        if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {\n          getLibDebugger('RenameDeleteHandler:runAsyncLinkUpdate')(\n            `Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`\n          );\n          return true;\n        }\n\n        if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {\n          getLibDebugger('RenameDeleteHandler:runAsyncLinkUpdate')(\n            `Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`\n          );\n          return true;\n        }\n\n        if (!this.app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n          return false;\n        }\n\n        if (this.app.plugins.getPlugin('backlink-cache')) {\n          return false;\n        }\n\n        if (linkUpdate.sourceFile.extension === CANVAS_FILE_EXTENSION) {\n          return true;\n        }\n\n        if (linkUpdate.resolvedFile.extension === CANVAS_FILE_EXTENSION) {\n          return true;\n        }\n\n        return false;\n      }\n    );\n  }\n}\n\nclass SettingsManager {\n  public readonly renameDeleteHandlersMap: Map<string, () => Partial<RenameDeleteHandlerSettings>>;\n\n  public constructor(private readonly app: App) {\n    this.renameDeleteHandlersMap =\n      getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n  }\n\n  public getSettings(): Partial<RenameDeleteHandlerSettings> {\n    const settingsBuilders = Array.from(this.renameDeleteHandlersMap.values()).reverse();\n\n    const settings: Partial<RenameDeleteHandlerSettings> = {};\n    settings.isNote = (path: string): boolean => isNote(this.app, path);\n    settings.isPathIgnored = (): boolean => false;\n\n    for (const settingsBuilder of settingsBuilders) {\n      const newSettings = settingsBuilder();\n      settings.shouldDeleteConflictingAttachments ||= newSettings.shouldDeleteConflictingAttachments ?? false;\n      if (newSettings.emptyFolderBehavior) {\n        settings.emptyFolderBehavior ??= newSettings.emptyFolderBehavior;\n      }\n      settings.shouldHandleDeletions ||= newSettings.shouldHandleDeletions ?? false;\n      settings.shouldHandleRenames ||= newSettings.shouldHandleRenames ?? false;\n      settings.shouldRenameAttachmentFiles ||= newSettings.shouldRenameAttachmentFiles ?? false;\n      settings.shouldRenameAttachmentFolder ||= newSettings.shouldRenameAttachmentFolder ?? false;\n      settings.shouldUpdateFileNameAliases ||= newSettings.shouldUpdateFileNameAliases ?? false;\n      const isPathIgnored = settings.isPathIgnored;\n      settings.isPathIgnored = (path: string): boolean => isPathIgnored(path) || (newSettings.isPathIgnored?.(path) ?? false);\n      const currentIsNote = settings.isNote;\n      settings.isNote = (path: string): boolean => currentIsNote(path) && (newSettings.isNote?.(path) ?? true);\n    }\n\n    settings.emptyFolderBehavior ??= EmptyFolderBehavior.Keep;\n    return settings;\n  }\n\n  public isNoteEx(path: string): boolean {\n    const settings = this.getSettings();\n    return settings.isNote?.(path) ?? false;\n  }\n}\n\nclass RenameHandler {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly handledRenames: HandledRenames;\n  private readonly interruptedCombinedBacklinksMap: Map<string, Map<string, string>>;\n  private readonly interruptedRenamesMap: Map<string, InterruptedRename[]>;\n  private readonly newPath: string;\n  private readonly oldCache: CachedMetadata | null;\n  private readonly oldPath: string;\n  private readonly oldPathBacklinksMap: Map<string, Reference[]>;\n  private readonly oldPathLinks: Reference[];\n  private readonly settingsManager: SettingsManager;\n\n  public constructor(params: RenameHandlerParams) {\n    this.app = params.app;\n    this.oldPath = params.oldPath;\n    this.newPath = params.newPath;\n    this.oldPathBacklinksMap = params.oldPathBacklinksMap;\n    this.oldCache = params.oldCache;\n    this.abortSignal = params.abortSignal;\n    this.settingsManager = params.settingsManager;\n    this.interruptedRenamesMap = params.interruptedRenamesMap;\n    this.oldPathLinks = this.oldCache ? getAllLinks(this.oldCache) : [];\n    this.handledRenames = params.handledRenames;\n    this.interruptedCombinedBacklinksMap = params.interruptedCombinedBacklinksMap ?? new Map<string, Map<string, string>>();\n  }\n\n  public async handle(): Promise<void> {\n    if (this.oldPath === this.newPath) {\n      return;\n    }\n    this.abortSignal.throwIfAborted();\n    await this.continueInterruptedRenames();\n    this.abortSignal.throwIfAborted();\n    await this.refreshLinks();\n    this.abortSignal.throwIfAborted();\n    if (await this.handleCaseCollision()) {\n      return;\n    }\n\n    this.abortSignal.throwIfAborted();\n\n    const renamedFilePaths = getObsidianDevUtilsState(this.app, 'renamedFilePaths', new Set<string>()).value;\n    const renamedLinks = getObsidianDevUtilsState(this.app, 'renamedLinkPaths', new Set<string>()).value;\n\n    try {\n      const renameMap = new RenameMap({\n        abortSignal: this.abortSignal,\n        app: this.app,\n        newPath: this.newPath,\n        oldCache: this.oldCache,\n        oldPath: this.oldPath,\n        settingsManager: this.settingsManager\n      });\n      await renameMap.fill();\n      this.abortSignal.throwIfAborted();\n\n      const combinedBacklinksMap = new Map<string, Map<string, string>>();\n      renameMap.initOriginalLinksMap(combinedBacklinksMap);\n      renameMap.initBacklinksMap(this.oldPathBacklinksMap, combinedBacklinksMap, this.oldPath);\n\n      for (const attachmentOldPath of renameMap.keys()) {\n        if (attachmentOldPath === this.oldPath) {\n          continue;\n        }\n        const attachmentOldPathBacklinksMap = (await getBacklinksForFileSafe(this.app, attachmentOldPath)).data;\n        this.abortSignal.throwIfAborted();\n        renameMap.initBacklinksMap(attachmentOldPathBacklinksMap, combinedBacklinksMap, attachmentOldPath);\n      }\n\n      const parentFolderPaths = new Set<string>();\n\n      for (const [oldAttachmentPath, newAttachmentPath] of renameMap.entries()) {\n        if (oldAttachmentPath !== this.oldPath) {\n          const fixedNewAttachmentPath = await this.renameHandled(oldAttachmentPath, newAttachmentPath);\n          this.abortSignal.throwIfAborted();\n          renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);\n        }\n        if (!this.settingsManager.isNoteEx(oldAttachmentPath)) {\n          parentFolderPaths.add(dirname(oldAttachmentPath));\n        }\n      }\n\n      await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths));\n      this.abortSignal.throwIfAborted();\n      const settings = this.settingsManager.getSettings();\n\n      for (\n        const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(\n          Array.from(this.interruptedCombinedBacklinksMap.entries())\n        )\n      ) {\n        let linkIndex = 0;\n        await editLinks(this.app, newBacklinkPath, (link) => {\n          linkIndex++;\n          const oldAttachmentPath = linkJsonToPathMap.get(toJson(link));\n          if (!oldAttachmentPath) {\n            return;\n          }\n\n          const newAttachmentPath = renameMap.get(oldAttachmentPath) ?? oldAttachmentPath;\n\n          renamedFilePaths.add(newBacklinkPath);\n          renamedLinks.add(`${newBacklinkPath}//${String(linkIndex)}`);\n\n          return updateLink(normalizeOptionalProperties<UpdateLinkParams>({\n            app: this.app,\n            link,\n            newSourcePathOrFile: newBacklinkPath,\n            newTargetPathOrFile: newAttachmentPath,\n            oldTargetPathOrFile: oldAttachmentPath,\n            shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n          }));\n        }, {\n          shouldFailOnMissingFile: false\n        });\n        this.abortSignal.throwIfAborted();\n      }\n\n      if (isNote(this.app, this.newPath)) {\n        await updateLinksInFile(normalizeOptionalProperties<UpdateLinksInFileParams>({\n          app: this.app,\n          newSourcePathOrFile: this.newPath,\n          oldSourcePathOrFile: this.oldPath,\n          shouldFailOnMissingFile: false,\n          shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n        }));\n        this.abortSignal.throwIfAborted();\n      }\n\n      if (!getFileOrNull(this.app, this.newPath)) {\n        let interruptedRenames = this.interruptedRenamesMap.get(this.newPath);\n        if (!interruptedRenames) {\n          interruptedRenames = [];\n          this.interruptedRenamesMap.set(this.newPath, interruptedRenames);\n        }\n        interruptedRenames.push({\n          combinedBacklinksMap,\n          oldPath: this.oldPath\n        });\n      }\n    } finally {\n      const orphanKeys = Array.from(this.handledRenames.keys());\n      addToQueue({\n        abortSignal: this.abortSignal,\n        app: this.app,\n        operationFn: () => {\n          for (const orphanKey of orphanKeys) {\n            this.handledRenames.delete(orphanKey.oldPath, orphanKey.newPath);\n          }\n\n          if (renamedLinks.size === 0) {\n            return;\n          }\n          new Notice(t(($) => $.obsidianDevUtils.renameDeleteHandler.updatedLinks, { filesCount: renamedFilePaths.size, linksCount: renamedLinks.size }));\n          renamedFilePaths.clear();\n          renamedLinks.clear();\n        },\n        operationName: t(($) => $.obsidianDevUtils.renameDeleteHandler.handleOrphanedRenames)\n      });\n    }\n  }\n\n  private async continueInterruptedRenames(): Promise<void> {\n    const interruptedRenames = this.interruptedRenamesMap.get(this.oldPath);\n    if (interruptedRenames) {\n      this.interruptedRenamesMap.delete(this.oldPath);\n      for (const interruptedRename of interruptedRenames) {\n        await new RenameHandler({\n          abortSignal: this.abortSignal,\n          app: this.app,\n          handledRenames: this.handledRenames,\n          interruptedCombinedBacklinksMap: interruptedRename.combinedBacklinksMap,\n          interruptedRenamesMap: this.interruptedRenamesMap,\n          newPath: this.newPath,\n          oldCache: this.oldCache,\n          oldPath: interruptedRename.oldPath,\n          oldPathBacklinksMap: this.oldPathBacklinksMap,\n          settingsManager: this.settingsManager\n        }).handle();\n      }\n    }\n  }\n\n  private async handleCaseCollision(): Promise<boolean> {\n    if (!this.app.vault.adapter.insensitive || this.oldPath.toLowerCase() !== this.newPath.toLowerCase()) {\n      return false;\n    }\n\n    const tempPath = join(dirname(this.newPath), `__temp__${basename(this.newPath)}`);\n    await this.renameHandled(this.newPath, tempPath);\n\n    await new RenameHandler({\n      abortSignal: this.abortSignal,\n      app: this.app,\n      handledRenames: this.handledRenames,\n      interruptedRenamesMap: this.interruptedRenamesMap,\n      newPath: tempPath,\n      oldCache: this.oldCache,\n      oldPath: this.oldPath,\n      oldPathBacklinksMap: this.oldPathBacklinksMap,\n      settingsManager: this.settingsManager\n    }).handle();\n\n    await this.app.fileManager.renameFile(getFile(this.app, tempPath), this.newPath);\n    return true;\n  }\n\n  private async refreshLinks(): Promise<void> {\n    const cache = this.app.metadataCache.getCache(this.oldPath) ?? this.app.metadataCache.getCache(this.newPath);\n    const oldPathLinksRefreshed = cache ? getAllLinks(cache) : [];\n    const fakeOldFile = getFile(this.app, this.oldPath, true);\n    let oldPathBacklinksMapRefreshed = new Map<string, Reference[]>();\n    await tempRegisterFilesAndRun(this.app, [fakeOldFile], async () => {\n      oldPathBacklinksMapRefreshed = (await getBacklinksForFileSafe(this.app, fakeOldFile)).data;\n    });\n\n    for (const link of oldPathLinksRefreshed) {\n      if (this.oldPathLinks.includes(link)) {\n        continue;\n      }\n      this.oldPathLinks.push(link);\n    }\n\n    for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {\n      let oldLinks = this.oldPathBacklinksMap.get(backlinkPath);\n      if (!oldLinks) {\n        oldLinks = [];\n        this.oldPathBacklinksMap.set(backlinkPath, oldLinks);\n      }\n\n      for (const link of refreshedLinks) {\n        if (oldLinks.includes(link)) {\n          continue;\n        }\n        oldLinks.push(link);\n      }\n    }\n  }\n\n  private async renameHandled(oldPath: string, newPath: string): Promise<string> {\n    newPath = getSafeRenamePath(this.app, oldPath, newPath);\n    if (oldPath === newPath) {\n      return newPath;\n    }\n    this.handledRenames.add(oldPath, newPath);\n    newPath = await renameSafe(this.app, oldPath, newPath);\n    return newPath;\n  }\n}\n\nclass RenameMap {\n  private readonly abortSignal: AbortSignal;\n  private readonly app: App;\n  private readonly map = new Map<string, string>();\n  private readonly newPath: string;\n  private readonly oldCache: CachedMetadata | null;\n  private readonly oldPath: string;\n  private readonly oldPathLinks: Reference[];\n  private readonly settingsManager: SettingsManager;\n\n  public constructor(params: RenameMapParams) {\n    this.abortSignal = params.abortSignal;\n    this.app = params.app;\n    this.settingsManager = params.settingsManager;\n    this.oldCache = params.oldCache;\n    this.oldPath = params.oldPath;\n    this.newPath = params.newPath;\n    this.oldPathLinks = this.oldCache ? getAllLinks(this.oldCache) : [];\n  }\n\n  public entries(): IterableIterator<[string, string]> {\n    return this.map.entries();\n  }\n\n  public async fill(): Promise<void> {\n    this.abortSignal.throwIfAborted();\n    this.map.set(this.oldPath, this.newPath);\n\n    if (!isNote(this.app, this.oldPath)) {\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n\n    const oldFile = getFile(this.app, this.oldPath, true);\n    let oldAttachmentFolderPath = '';\n    await tempRegisterFilesAndRunAsync(this.app, [oldFile], async () => {\n      const shouldFakeOldPathCache = this.oldCache && oldFile.deleted;\n      if (shouldFakeOldPathCache) {\n        registerFileCacheForNonExistingFile(this.app, oldFile, this.oldCache);\n      }\n\n      try {\n        oldAttachmentFolderPath = await getAttachmentFolderPath(this.app, this.oldPath, AttachmentPathContext.RenameNote);\n      } finally {\n        if (shouldFakeOldPathCache) {\n          unregisterFileCacheForNonExistingFile(this.app, oldFile);\n        }\n      }\n    });\n\n    const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n      ? await getAttachmentFolderPath(this.app, this.newPath, AttachmentPathContext.RenameNote)\n      : oldAttachmentFolderPath;\n\n    const isOldAttachmentFolderAtRoot = oldAttachmentFolderPath === '/';\n\n    const oldAttachmentFolder = getFolderOrNull(this.app, oldAttachmentFolderPath);\n\n    if (!oldAttachmentFolder) {\n      return;\n    }\n\n    if (oldAttachmentFolderPath === newAttachmentFolderPath && !settings.shouldRenameAttachmentFiles) {\n      return;\n    }\n\n    const oldAttachmentFiles: TFile[] = [];\n\n    if (await hasOwnAttachmentFolder(this.app, this.oldPath, AttachmentPathContext.RenameNote)) {\n      Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {\n        this.abortSignal.throwIfAborted();\n        if (isFile(oldAttachmentFile)) {\n          oldAttachmentFiles.push(oldAttachmentFile);\n        }\n      });\n    } else {\n      for (const oldPathLink of this.oldPathLinks) {\n        this.abortSignal.throwIfAborted();\n        const oldAttachmentFile = extractLinkFile(this.app, oldPathLink, this.oldPath);\n        if (!oldAttachmentFile) {\n          continue;\n        }\n\n        if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n          const oldAttachmentBacklinks = await getBacklinksForFileSafe(this.app, oldAttachmentFile);\n          this.abortSignal.throwIfAborted();\n          const keys = new Set<string>(oldAttachmentBacklinks.keys());\n          keys.delete(this.oldPath);\n          keys.delete(this.newPath);\n          if (keys.size === 0) {\n            oldAttachmentFiles.push(oldAttachmentFile);\n          }\n        }\n      }\n    }\n\n    for (const oldAttachmentFile of oldAttachmentFiles) {\n      this.abortSignal.throwIfAborted();\n      if (this.settingsManager.isNoteEx(oldAttachmentFile.path)) {\n        continue;\n      }\n\n      let newAttachmentFilePath: string;\n      if (settings.shouldRenameAttachmentFiles) {\n        newAttachmentFilePath = await getAttachmentFilePath({\n          app: this.app,\n          context: AttachmentPathContext.RenameNote,\n          notePathOrFile: this.newPath,\n          oldAttachmentPathOrFile: oldAttachmentFile,\n          oldNotePathOrFile: this.oldPath,\n          shouldSkipDuplicateCheck: true\n        });\n        this.abortSignal.throwIfAborted();\n      } else {\n        const relativePath = isOldAttachmentFolderAtRoot ? oldAttachmentFile.path : relative(oldAttachmentFolderPath, oldAttachmentFile.path);\n        const newFolder = join(newAttachmentFolderPath, dirname(relativePath));\n        newAttachmentFilePath = join(newFolder, oldAttachmentFile.name);\n      }\n\n      if (oldAttachmentFile.path === newAttachmentFilePath) {\n        continue;\n      }\n      if (settings.shouldDeleteConflictingAttachments) {\n        const newAttachmentFile = getFileOrNull(this.app, newAttachmentFilePath);\n        if (newAttachmentFile) {\n          getLibDebugger('RenameDeleteHandler:fillRenameMap')(`Removing conflicting attachment ${newAttachmentFile.path}.`);\n          await trashSafe(this.app, newAttachmentFile);\n          this.abortSignal.throwIfAborted();\n        }\n      } else {\n        const dir = dirname(newAttachmentFilePath);\n        const ext = extname(newAttachmentFilePath);\n        const baseName = basename(newAttachmentFilePath, ext);\n        newAttachmentFilePath = this.app.vault.getAvailablePath(join(dir, baseName), ext.slice(1));\n      }\n      this.map.set(oldAttachmentFile.path, newAttachmentFilePath);\n    }\n  }\n\n  public get(oldPath: string): string | undefined {\n    return this.map.get(oldPath);\n  }\n\n  public initBacklinksMap(\n    singleBacklinksMap: Map<string, Reference[]>,\n    combinedBacklinksMap: Map<string, Map<string, string>>,\n    path: string\n  ): void {\n    for (const [backlinkPath, links] of singleBacklinksMap.entries()) {\n      const newBacklinkPath = this.map.get(backlinkPath) ?? backlinkPath;\n      const linkJsonToPathMap = combinedBacklinksMap.get(newBacklinkPath) ?? new Map<string, string>();\n      combinedBacklinksMap.set(newBacklinkPath, linkJsonToPathMap);\n      for (const link of links) {\n        linkJsonToPathMap.set(toJson(link), path);\n      }\n    }\n  }\n\n  public initOriginalLinksMap(combinedBacklinksMap: Map<string, Map<string, string>>): void {\n    for (const oldPathLink of this.oldPathLinks) {\n      const oldAttachmentFile = extractLinkFile(this.app, oldPathLink, this.oldPath);\n      if (!oldAttachmentFile) {\n        continue;\n      }\n      const backlinksMap = new Map<string, Reference[]>();\n      backlinksMap.set(this.newPath, [oldPathLink]);\n      this.initBacklinksMap(backlinksMap, combinedBacklinksMap, oldAttachmentFile.path);\n    }\n  }\n\n  public keys(): IterableIterator<string> {\n    return this.map.keys();\n  }\n\n  public set(oldPath: string, newPath: string): void {\n    this.map.set(oldPath, newPath);\n  }\n}\n\n/**\n * Registers the rename/delete handlers.\n *\n * @param plugin - The plugin instance.\n * @param settingsBuilder - A function that returns the settings for the rename delete handler.\n */\nexport function registerRenameDeleteHandlers(plugin: AbortablePlugin, settingsBuilder: () => Partial<RenameDeleteHandlerSettings>): void {\n  new Registry(plugin, settingsBuilder, new SettingsManager(plugin.app)).register();\n}\n\nasync function cleanupParentFolders(app: App, settings: Partial<RenameDeleteHandlerSettings>, parentFolderPaths: string[]): Promise<void> {\n  if (settings.emptyFolderBehavior === EmptyFolderBehavior.Keep) {\n    return;\n  }\n  for (const parentFolderPath of parentFolderPaths) {\n    switch (settings.emptyFolderBehavior) {\n      case EmptyFolderBehavior.Delete:\n        await deleteEmptyFolder(app, parentFolderPath);\n        break;\n      case EmptyFolderBehavior.DeleteWithEmptyParents:\n        await deleteEmptyFolderHierarchy(app, parentFolderPath);\n        break;\n      default:\n        break;\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["EmptyFolderBehavior"]
}

|
|
@@ -60,6 +60,22 @@ export declare function createTempFile(app: App, path: string): Promise<() => Pr
|
|
|
60
60
|
* @returns A {@link Promise} that resolves to a function that can be called to delete the temporary folder and all its created parents.
|
|
61
61
|
*/
|
|
62
62
|
export declare function createTempFolder(app: App, path: string): Promise<() => Promise<void>>;
|
|
63
|
+
/**
|
|
64
|
+
* Deletes an empty folder.
|
|
65
|
+
*
|
|
66
|
+
* @param app - The application instance.
|
|
67
|
+
* @param pathOrFolder - The folder to delete.
|
|
68
|
+
* @returns A {@link Promise} that resolves when the folder is deleted.
|
|
69
|
+
*/
|
|
70
|
+
export declare function deleteEmptyFolder(app: App, pathOrFolder: null | PathOrFolder): Promise<void>;
|
|
71
|
+
/**
|
|
72
|
+
* Removes empty folder hierarchy starting from the given folder.
|
|
73
|
+
*
|
|
74
|
+
* @param app - The application instance.
|
|
75
|
+
* @param pathOrFolder - The folder to start removing empty hierarchy from.
|
|
76
|
+
* @returns A {@link Promise} that resolves when the empty hierarchy is deleted.
|
|
77
|
+
*/
|
|
78
|
+
export declare function deleteEmptyFolderHierarchy(app: App, pathOrFolder: null | PathOrFolder): Promise<void>;
|
|
63
79
|
/**
|
|
64
80
|
* Gets a safe path for a file or folder.
|
|
65
81
|
*
|
|
@@ -235,3 +251,11 @@ export declare function renameSafe(app: App, oldPathOrAbstractFile: PathOrAbstra
|
|
|
235
251
|
* @returns A {@link Promise} that resolves when the note is saved.
|
|
236
252
|
*/
|
|
237
253
|
export declare function saveNote(app: App, pathOrFile: PathOrFile): Promise<void>;
|
|
254
|
+
/**
|
|
255
|
+
* Trashes an abstract file safely from the vault.
|
|
256
|
+
*
|
|
257
|
+
* @param app - The Obsidian application instance.
|
|
258
|
+
* @param pathOrFile - The path or abstract file to trash.
|
|
259
|
+
* @returns A {@link Promise} that resolves when the file is trashed.
|
|
260
|
+
*/
|
|
261
|
+
export declare function trashSafe(app: App, pathOrFile: PathOrAbstractFile): Promise<void>;
|