obsidian-dev-utils 39.0.1 → 40.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +8 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/AttachmentPath.cjs +52 -23
- package/dist/lib/cjs/obsidian/AttachmentPath.d.cts +103 -26
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +7 -2
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/AttachmentPath.d.mts +103 -26
- package/dist/lib/esm/obsidian/AttachmentPath.mjs +53 -24
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +7 -2
- package/package.json +1 -1
|
@@ -19,48 +19,77 @@ import {
|
|
|
19
19
|
trimStart
|
|
20
20
|
} from "../String.mjs";
|
|
21
21
|
import {
|
|
22
|
-
|
|
22
|
+
getFileOrNull,
|
|
23
23
|
getFolder,
|
|
24
24
|
getFolderOrNull,
|
|
25
25
|
getPath
|
|
26
26
|
} from "./FileSystem.mjs";
|
|
27
27
|
const DUMMY_PATH = "__DUMMY__";
|
|
28
|
-
async function getAttachmentFilePath(
|
|
28
|
+
async function getAttachmentFilePath(options) {
|
|
29
|
+
const {
|
|
30
|
+
app,
|
|
31
|
+
attachmentPathOrFile,
|
|
32
|
+
notePathOrFile,
|
|
33
|
+
shouldSkipDuplicateCheck
|
|
34
|
+
} = options;
|
|
29
35
|
const attachmentPath = getPath(app, attachmentPathOrFile);
|
|
30
|
-
const
|
|
31
|
-
const
|
|
32
|
-
const
|
|
33
|
-
const fileBaseName = basename(attachmentPath, ext);
|
|
36
|
+
const attachmentFileExtension = extname(attachmentPath);
|
|
37
|
+
const attachmentFileBaseName = basename(attachmentPath, attachmentFileExtension);
|
|
38
|
+
const attachmentFile = getFileOrNull(app, attachmentPath);
|
|
34
39
|
const internalFn = app.vault.getAvailablePathForAttachments;
|
|
35
|
-
|
|
36
|
-
|
|
40
|
+
const extendedFn = internalFn.extended;
|
|
41
|
+
if (extendedFn) {
|
|
42
|
+
return extendedFn({
|
|
43
|
+
attachmentFileBaseName,
|
|
44
|
+
attachmentFileContent: attachmentFile ? await app.vault.readBinary(attachmentFile) : void 0,
|
|
45
|
+
attachmentFileExtension: attachmentFileExtension.slice(1),
|
|
46
|
+
attachmentFileStat: attachmentFile?.stat,
|
|
47
|
+
notePathOrFile,
|
|
48
|
+
shouldSkipDuplicateCheck,
|
|
49
|
+
shouldSkipMissingAttachmentFolderCreation: true
|
|
50
|
+
});
|
|
37
51
|
}
|
|
38
|
-
return await getAvailablePathForAttachments(
|
|
52
|
+
return await getAvailablePathForAttachments({
|
|
53
|
+
app,
|
|
54
|
+
attachmentFileBaseName,
|
|
55
|
+
attachmentFileExtension: attachmentFileExtension.slice(1),
|
|
56
|
+
notePathOrFile,
|
|
57
|
+
shouldSkipDuplicateCheck,
|
|
58
|
+
shouldSkipMissingAttachmentFolderCreation: true
|
|
59
|
+
});
|
|
39
60
|
}
|
|
40
61
|
async function getAttachmentFolderPath(app, notePathOrFile) {
|
|
41
|
-
return parentFolderPath(
|
|
62
|
+
return parentFolderPath(
|
|
63
|
+
await getAttachmentFilePath({
|
|
64
|
+
app,
|
|
65
|
+
attachmentPathOrFile: DUMMY_PATH,
|
|
66
|
+
notePathOrFile,
|
|
67
|
+
shouldSkipDuplicateCheck: true
|
|
68
|
+
})
|
|
69
|
+
);
|
|
42
70
|
}
|
|
43
|
-
async function getAvailablePathForAttachments(
|
|
71
|
+
async function getAvailablePathForAttachments(options) {
|
|
72
|
+
const {
|
|
73
|
+
app,
|
|
74
|
+
attachmentFileExtension,
|
|
75
|
+
notePathOrFile,
|
|
76
|
+
shouldSkipDuplicateCheck,
|
|
77
|
+
shouldSkipMissingAttachmentFolderCreation
|
|
78
|
+
} = options;
|
|
44
79
|
let attachmentFolderPath = app.vault.getConfig("attachmentFolderPath");
|
|
45
80
|
const isCurrentFolder = attachmentFolderPath === "." || attachmentFolderPath === "./";
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
relativePath = trimStart(attachmentFolderPath, "./");
|
|
49
|
-
}
|
|
81
|
+
const relativePath = attachmentFolderPath.startsWith("./") ? trimStart(attachmentFolderPath, "./") : null;
|
|
82
|
+
const noteFileOrNull = getFileOrNull(app, notePathOrFile);
|
|
50
83
|
if (isCurrentFolder) {
|
|
51
|
-
attachmentFolderPath =
|
|
84
|
+
attachmentFolderPath = noteFileOrNull ? noteFileOrNull.parent?.path ?? "" : "";
|
|
52
85
|
} else if (relativePath) {
|
|
53
|
-
attachmentFolderPath = (
|
|
86
|
+
attachmentFolderPath = (noteFileOrNull ? noteFileOrNull.parent?.getParentPrefix() ?? "" : "") + relativePath;
|
|
54
87
|
}
|
|
55
88
|
attachmentFolderPath = normalize(normalizeSlashes(attachmentFolderPath));
|
|
56
|
-
attachmentFileBaseName = normalize(normalizeSlashes(attachmentFileBaseName));
|
|
89
|
+
const attachmentFileBaseName = normalize(normalizeSlashes(options.attachmentFileBaseName));
|
|
57
90
|
let folder = getFolderOrNull(app, attachmentFolderPath, true);
|
|
58
91
|
if (!folder && relativePath) {
|
|
59
|
-
|
|
60
|
-
folder = getFolder(app, attachmentFolderPath, true);
|
|
61
|
-
} else {
|
|
62
|
-
folder = await app.vault.createFolder(attachmentFolderPath);
|
|
63
|
-
}
|
|
92
|
+
folder = shouldSkipMissingAttachmentFolderCreation ? getFolder(app, attachmentFolderPath, true) : await app.vault.createFolder(attachmentFolderPath);
|
|
64
93
|
}
|
|
65
94
|
const prefix = folder?.getParentPrefix() ?? "";
|
|
66
95
|
return shouldSkipDuplicateCheck ? makeFileName(prefix + attachmentFileBaseName, attachmentFileExtension) : app.vault.getAvailablePath(prefix + attachmentFileBaseName, attachmentFileExtension);
|
|
@@ -82,4 +111,4 @@ export {
|
|
|
82
111
|
getAvailablePathForAttachments,
|
|
83
112
|
hasOwnAttachmentFolder
|
|
84
113
|
};
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/AttachmentPath.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Provides utility functions for working with attachment paths.\n */\n\nimport type {\n  App,\n  TFile\n} from 'obsidian';\n\nimport { parentFolderPath } from 'obsidian-typings/implementations';\n\nimport type { PathOrFile } from './FileSystem.ts';\n\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  makeFileName\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll,\n  trimStart\n} from '../String.ts';\nimport {\n  getFile,\n  getFolder,\n  getFolderOrNull,\n  getPath\n} from './FileSystem.ts';\n\n/**\n * Is overridden wrapper.\n */\nexport interface ExtendedWrapper {\n  /**\n   * Is extended.\n   */\n  isExtended: true;\n}\n\n/**\n * Get available path for attachments function.\n *\n * @param attachmentFileBaseName - File base name of the attachment.\n * @param attachmentFileExtension - File extension of the attachment.\n * @param noteFile - The note file to attach to.\n * @param shouldSkipMissingAttachmentFolderCreation - Should missing attachment folder creation be skipped.\n * @param shouldSkipDuplicateCheck - Should the duplicate check be skipped.\n * @returns A {@link Promise} that resolves to the available path for attachments.\n */\nexport type GetAvailablePathForAttachmentsExtendedFn = (\n  attachmentFileBaseName: string,\n  attachmentFileExtension: string,\n  noteFile: null | TFile,\n  shouldSkipMissingAttachmentFolderCreation?: boolean,\n  shouldSkipDuplicateCheck?: boolean\n) => Promise<string>;\n\n/**\n * Dummy path.\n */\nexport const DUMMY_PATH = '__DUMMY__';\n\n/**\n * Retrieves the file path for an attachment within a note.\n *\n * @param app - The Obsidian application instance.\n * @param attachmentPathOrFile - The path of the attachment.\n * @param notePathOrFile - The path of the note.\n * @param shouldSkipDuplicateCheck - Should the duplicate check be skipped.\n * @returns A {@link Promise} that resolves to the file path of the attachment.\n */\nexport async function getAttachmentFilePath(\n  app: App,\n  attachmentPathOrFile: PathOrFile,\n  notePathOrFile: PathOrFile,\n  shouldSkipDuplicateCheck: boolean\n): Promise<string> {\n  const attachmentPath = getPath(app, attachmentPathOrFile);\n  const notePath = getPath(app, notePathOrFile);\n  const note = getFile(app, notePath, true);\n  const ext = extname(attachmentPath);\n  const fileBaseName = basename(attachmentPath, ext);\n\n  // eslint-disable-next-line @typescript-eslint/unbound-method\n  const internalFn = app.vault.getAvailablePathForAttachments;\n  if ((internalFn as Partial<ExtendedWrapper>).isExtended) {\n    return (internalFn as GetAvailablePathForAttachmentsExtendedFn)(fileBaseName, ext.slice(1), note, true, shouldSkipDuplicateCheck);\n  }\n\n  return await getAvailablePathForAttachments(app, fileBaseName, ext.slice(1), note, true, shouldSkipDuplicateCheck);\n}\n\n/**\n * Retrieves the attachment folder path for a given note.\n *\n * @param app - The Obsidian application instance.\n * @param notePathOrFile - The path of the note.\n * @returns A {@link Promise} that resolves to the attachment folder path.\n */\nexport async function getAttachmentFolderPath(app: App, notePathOrFile: PathOrFile): Promise<string> {\n  return parentFolderPath(await getAttachmentFilePath(app, DUMMY_PATH, notePathOrFile, true));\n}\n\n/**\n * Retrieves the available path for attachments.\n *\n * @param app - The Obsidian application instance.\n * @param attachmentFileBaseName - File name of the attachment.\n * @param attachmentFileExtension - Extension of the attachment.\n * @param noteFile - The file to attach to.\n * @param shouldSkipMissingAttachmentFolderCreation - Should missing attachment folder creation be skipped.\n * @param shouldSkipDuplicateCheck - Should the duplicate check be skipped.\n * @returns A {@link Promise} that resolves to the available path for attachments.\n */\nexport async function getAvailablePathForAttachments(\n  app: App,\n  attachmentFileBaseName: string,\n  attachmentFileExtension: string,\n  noteFile: null | TFile,\n  shouldSkipMissingAttachmentFolderCreation: boolean,\n  shouldSkipDuplicateCheck: boolean\n): Promise<string> {\n  let attachmentFolderPath = app.vault.getConfig('attachmentFolderPath') as string;\n  const isCurrentFolder = attachmentFolderPath === '.' || attachmentFolderPath === './';\n  let relativePath = null;\n\n  if (attachmentFolderPath.startsWith('./')) {\n    relativePath = trimStart(attachmentFolderPath, './');\n  }\n\n  if (isCurrentFolder) {\n    attachmentFolderPath = noteFile ? noteFile.parent?.path ?? '' : '';\n  } else if (relativePath) {\n    attachmentFolderPath = (noteFile ? noteFile.parent?.getParentPrefix() ?? '' : '') + relativePath;\n  }\n\n  attachmentFolderPath = normalize(normalizeSlashes(attachmentFolderPath));\n  attachmentFileBaseName = normalize(normalizeSlashes(attachmentFileBaseName));\n\n  let folder = getFolderOrNull(app, attachmentFolderPath, true);\n\n  if (!folder && relativePath) {\n    if (shouldSkipMissingAttachmentFolderCreation) {\n      folder = getFolder(app, attachmentFolderPath, true);\n    } else {\n      folder = await app.vault.createFolder(attachmentFolderPath);\n    }\n  }\n\n  const prefix = folder?.getParentPrefix() ?? '';\n  return shouldSkipDuplicateCheck\n    ? makeFileName(prefix + attachmentFileBaseName, attachmentFileExtension)\n    : app.vault.getAvailablePath(prefix + attachmentFileBaseName, attachmentFileExtension);\n}\n\n/**\n * Checks if a note has its own attachment folder.\n *\n * @param app - The Obsidian application instance.\n * @param path - The path of the note.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the note has its own attachment folder.\n */\nexport async function hasOwnAttachmentFolder(app: App, path: string): Promise<boolean> {\n  const attachmentFolderPath = await getAttachmentFolderPath(app, path);\n  const dummyAttachmentFolderPath = await getAttachmentFolderPath(app, join(dirname(path), 'DUMMY_FILE.md'));\n  return attachmentFolderPath !== dummyAttachmentFolderPath;\n}\n\n/**\n * Normalizes a path by combining multiple slashes into a single slash and removing leading and trailing slashes.\n *\n * @param path - Path to normalize.\n * @returns The normalized path.\n */\nfunction normalizeSlashes(path: string): string {\n  path = replaceAll(path, /(?:[\\\\/])+/g, '/');\n  path = replaceAll(path, /^\\/+|\\/+$/g, '');\n  return path || '/';\n}\n"],
  "mappings": ";;;;;;;AAWA,SAAS,wBAAwB;AAIjC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAiCA,MAAM,aAAa;AAW1B,eAAsB,sBACpB,KACA,sBACA,gBACA,0BACiB;AACjB,QAAM,iBAAiB,QAAQ,KAAK,oBAAoB;AACxD,QAAM,WAAW,QAAQ,KAAK,cAAc;AAC5C,QAAM,OAAO,QAAQ,KAAK,UAAU,IAAI;AACxC,QAAM,MAAM,QAAQ,cAAc;AAClC,QAAM,eAAe,SAAS,gBAAgB,GAAG;AAGjD,QAAM,aAAa,IAAI,MAAM;AAC7B,MAAK,WAAwC,YAAY;AACvD,WAAQ,WAAwD,cAAc,IAAI,MAAM,CAAC,GAAG,MAAM,MAAM,wBAAwB;AAAA,EAClI;AAEA,SAAO,MAAM,+BAA+B,KAAK,cAAc,IAAI,MAAM,CAAC,GAAG,MAAM,MAAM,wBAAwB;AACnH;AASA,eAAsB,wBAAwB,KAAU,gBAA6C;AACnG,SAAO,iBAAiB,MAAM,sBAAsB,KAAK,YAAY,gBAAgB,IAAI,CAAC;AAC5F;AAaA,eAAsB,+BACpB,KACA,wBACA,yBACA,UACA,2CACA,0BACiB;AACjB,MAAI,uBAAuB,IAAI,MAAM,UAAU,sBAAsB;AACrE,QAAM,kBAAkB,yBAAyB,OAAO,yBAAyB;AACjF,MAAI,eAAe;AAEnB,MAAI,qBAAqB,WAAW,IAAI,GAAG;AACzC,mBAAe,UAAU,sBAAsB,IAAI;AAAA,EACrD;AAEA,MAAI,iBAAiB;AACnB,2BAAuB,WAAW,SAAS,QAAQ,QAAQ,KAAK;AAAA,EAClE,WAAW,cAAc;AACvB,4BAAwB,WAAW,SAAS,QAAQ,gBAAgB,KAAK,KAAK,MAAM;AAAA,EACtF;AAEA,yBAAuB,UAAU,iBAAiB,oBAAoB,CAAC;AACvE,2BAAyB,UAAU,iBAAiB,sBAAsB,CAAC;AAE3E,MAAI,SAAS,gBAAgB,KAAK,sBAAsB,IAAI;AAE5D,MAAI,CAAC,UAAU,cAAc;AAC3B,QAAI,2CAA2C;AAC7C,eAAS,UAAU,KAAK,sBAAsB,IAAI;AAAA,IACpD,OAAO;AACL,eAAS,MAAM,IAAI,MAAM,aAAa,oBAAoB;AAAA,IAC5D;AAAA,EACF;AAEA,QAAM,SAAS,QAAQ,gBAAgB,KAAK;AAC5C,SAAO,2BACH,aAAa,SAAS,wBAAwB,uBAAuB,IACrE,IAAI,MAAM,iBAAiB,SAAS,wBAAwB,uBAAuB;AACzF;AASA,eAAsB,uBAAuB,KAAU,MAAgC;AACrF,QAAM,uBAAuB,MAAM,wBAAwB,KAAK,IAAI;AACpE,QAAM,4BAA4B,MAAM,wBAAwB,KAAK,KAAK,QAAQ,IAAI,GAAG,eAAe,CAAC;AACzG,SAAO,yBAAyB;AAClC;AAQA,SAAS,iBAAiB,MAAsB;AAC9C,SAAO,WAAW,MAAM,eAAe,GAAG;AAC1C,SAAO,WAAW,MAAM,cAAc,EAAE;AACxC,SAAO,QAAQ;AACjB;",
  "names": []
}

|
|
114
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/AttachmentPath.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Provides utility functions for working with attachment paths.\n */\n\nimport type {\n  App,\n  FileStats,\n  Vault\n} from 'obsidian';\n\nimport { parentFolderPath } from 'obsidian-typings/implementations';\n\nimport type { PathOrFile } from './FileSystem.ts';\n\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  makeFileName\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll,\n  trimStart\n} from '../String.ts';\nimport {\n  getFileOrNull,\n  getFolder,\n  getFolderOrNull,\n  getPath\n} from './FileSystem.ts';\n\n/**\n * Get available path for attachments extended function.\n *\n * @param options - Options for the get available path for attachments extended function.\n * @returns A {@link Promise} that resolves to the available path for attachments.\n */\nexport type GetAvailablePathForAttachmentsExtendedFn = (options: GetAvailablePathForAttachmentsExtendedFnOptions) => Promise<string>;\n\n/**\n * Options for the get available path for attachments extended function.\n */\nexport interface GetAvailablePathForAttachmentsExtendedFnOptions {\n  /**\n   * A base name of the attachment.\n   */\n  attachmentFileBaseName: string;\n\n  /**\n   * A content of the attachment file.\n   */\n  attachmentFileContent?: ArrayBuffer | undefined;\n\n  /**\n   * An extension of the attachment.\n   */\n  attachmentFileExtension: string;\n\n  /**\n   * A stats of the attachment file.\n   */\n  attachmentFileStat?: FileStats | undefined;\n\n  /**\n   * A path or file of the note.\n   */\n  notePathOrFile: null | PathOrFile;\n\n  /**\n   * Should the duplicate check be skipped.\n   */\n  shouldSkipDuplicateCheck?: boolean;\n\n  /**\n   * Should the generated attachment file name be skipped.\n   */\n  shouldSkipGeneratedAttachmentFileName?: boolean;\n\n  /**\n   * Should missing attachment folder creation be skipped.\n   */\n  shouldSkipMissingAttachmentFolderCreation: boolean | undefined;\n}\n\n/**\n * {@link Vault.getAvailablePathForAttachments} extended wrapper.\n */\nexport interface GetAvailablePathForAttachmentsFnExtendedWrapper extends GetAvailablePathForAttachmentsFn {\n  /**\n   * An extended function.\n   */\n  extended: GetAvailablePathForAttachmentsExtendedFn;\n}\n\ntype GetAvailablePathForAttachmentsFn = Vault['getAvailablePathForAttachments'];\n\n/**\n * Dummy path.\n */\nexport const DUMMY_PATH = '__DUMMY__';\n\n/**\n * Options for the getAttachmentFilePath function.\n */\nexport interface GetAttachmentFilePathOptions {\n  /**\n   * An Obsidian application instance.\n   */\n  app: App;\n  /**\n   * A path of the attachment.\n   */\n  attachmentPathOrFile: PathOrFile;\n  /**\n   * A path of the note.\n   */\n  notePathOrFile: PathOrFile;\n  /**\n   * Should the duplicate check be skipped.\n   */\n  shouldSkipDuplicateCheck: boolean;\n}\n\n/**\n * Options for the getAvailablePathForAttachments function.\n */\nexport interface GetAvailablePathForAttachmentsOptions {\n  /**\n   * An Obsidian application instance.\n   */\n  app: App;\n  /**\n   * A base name of the attachment.\n   */\n  attachmentFileBaseName: string;\n  /**\n   * An extension of the attachment.\n   */\n  attachmentFileExtension: string;\n  /**\n   * A file to attach to.\n   */\n  notePathOrFile: null | PathOrFile;\n  /**\n   * Should the duplicate check be skipped.\n   */\n  shouldSkipDuplicateCheck?: boolean;\n  /**\n   * Should missing attachment folder creation be skipped.\n   */\n  shouldSkipMissingAttachmentFolderCreation?: boolean;\n}\n\n/**\n * Retrieves the file path for an attachment within a note.\n *\n * @param options - Options for the get attachment file path function.\n * @returns A {@link Promise} that resolves to the file path of the attachment.\n */\nexport async function getAttachmentFilePath(options: GetAttachmentFilePathOptions): Promise<string> {\n  const {\n    app,\n    attachmentPathOrFile,\n    notePathOrFile,\n    shouldSkipDuplicateCheck\n  } = options;\n  const attachmentPath = getPath(app, attachmentPathOrFile);\n  const attachmentFileExtension = extname(attachmentPath);\n  const attachmentFileBaseName = basename(attachmentPath, attachmentFileExtension);\n  const attachmentFile = getFileOrNull(app, attachmentPath);\n\n  // eslint-disable-next-line @typescript-eslint/unbound-method\n  const internalFn = app.vault.getAvailablePathForAttachments;\n  const extendedFn = (internalFn as Partial<GetAvailablePathForAttachmentsFnExtendedWrapper>).extended;\n  if (extendedFn) {\n    return extendedFn({\n      attachmentFileBaseName,\n      attachmentFileContent: attachmentFile ? await app.vault.readBinary(attachmentFile) : undefined,\n      attachmentFileExtension: attachmentFileExtension.slice(1),\n      attachmentFileStat: attachmentFile?.stat,\n      notePathOrFile,\n      shouldSkipDuplicateCheck,\n      shouldSkipMissingAttachmentFolderCreation: true\n    });\n  }\n\n  return await getAvailablePathForAttachments({\n    app,\n    attachmentFileBaseName,\n    attachmentFileExtension: attachmentFileExtension.slice(1),\n    notePathOrFile,\n    shouldSkipDuplicateCheck,\n    shouldSkipMissingAttachmentFolderCreation: true\n  });\n}\n\n/**\n * Retrieves the attachment folder path for a given note.\n *\n * @param app - The Obsidian application instance.\n * @param notePathOrFile - The path of the note.\n * @returns A {@link Promise} that resolves to the attachment folder path.\n */\nexport async function getAttachmentFolderPath(app: App, notePathOrFile: PathOrFile): Promise<string> {\n  return parentFolderPath(\n    await getAttachmentFilePath({\n      app,\n      attachmentPathOrFile: DUMMY_PATH,\n      notePathOrFile,\n      shouldSkipDuplicateCheck: true\n    })\n  );\n}\n\n/**\n * Retrieves the available path for attachments.\n *\n * @param options - Options for the get available path for attachments function.\n * @returns A {@link Promise} that resolves to the available path for attachments.\n */\nexport async function getAvailablePathForAttachments(options: GetAvailablePathForAttachmentsOptions): Promise<string> {\n  const {\n    app,\n    attachmentFileExtension,\n    notePathOrFile,\n    shouldSkipDuplicateCheck,\n    shouldSkipMissingAttachmentFolderCreation\n  } = options;\n  let attachmentFolderPath = app.vault.getConfig('attachmentFolderPath') as string;\n  const isCurrentFolder = attachmentFolderPath === '.' || attachmentFolderPath === './';\n  const relativePath = attachmentFolderPath.startsWith('./') ? trimStart(attachmentFolderPath, './') : null;\n\n  const noteFileOrNull = getFileOrNull(app, notePathOrFile);\n\n  if (isCurrentFolder) {\n    attachmentFolderPath = noteFileOrNull ? noteFileOrNull.parent?.path ?? '' : '';\n  } else if (relativePath) {\n    attachmentFolderPath = (noteFileOrNull ? noteFileOrNull.parent?.getParentPrefix() ?? '' : '') + relativePath;\n  }\n\n  attachmentFolderPath = normalize(normalizeSlashes(attachmentFolderPath));\n  const attachmentFileBaseName = normalize(normalizeSlashes(options.attachmentFileBaseName));\n\n  let folder = getFolderOrNull(app, attachmentFolderPath, true);\n\n  if (!folder && relativePath) {\n    folder = shouldSkipMissingAttachmentFolderCreation\n      ? getFolder(app, attachmentFolderPath, true)\n      : await app.vault.createFolder(attachmentFolderPath);\n  }\n\n  const prefix = folder?.getParentPrefix() ?? '';\n  return shouldSkipDuplicateCheck\n    ? makeFileName(prefix + attachmentFileBaseName, attachmentFileExtension)\n    : app.vault.getAvailablePath(prefix + attachmentFileBaseName, attachmentFileExtension);\n}\n\n/**\n * Checks if a note has its own attachment folder.\n *\n * @param app - The Obsidian application instance.\n * @param path - The path of the note.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the note has its own attachment folder.\n */\nexport async function hasOwnAttachmentFolder(app: App, path: string): Promise<boolean> {\n  const attachmentFolderPath = await getAttachmentFolderPath(app, path);\n  const dummyAttachmentFolderPath = await getAttachmentFolderPath(app, join(dirname(path), 'DUMMY_FILE.md'));\n  return attachmentFolderPath !== dummyAttachmentFolderPath;\n}\n\n/**\n * Normalizes a path by combining multiple slashes into a single slash and removing leading and trailing slashes.\n *\n * @param path - Path to normalize.\n * @returns The normalized path.\n */\nfunction normalizeSlashes(path: string): string {\n  path = replaceAll(path, /(?:[\\\\/])+/g, '/');\n  path = replaceAll(path, /^\\/+|\\/+$/g, '');\n  return path || '/';\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -173,7 +173,12 @@ async function fillRenameMap(app, oldPath, newPath, renameMap, oldPathLinks, abo
|
|
|
173
173
|
}
|
|
174
174
|
let newAttachmentFilePath;
|
|
175
175
|
if (settings.shouldRenameAttachmentFiles) {
|
|
176
|
-
newAttachmentFilePath = await getAttachmentFilePath(
|
|
176
|
+
newAttachmentFilePath = await getAttachmentFilePath({
|
|
177
|
+
app,
|
|
178
|
+
attachmentPathOrFile: oldAttachmentFile,
|
|
179
|
+
notePathOrFile: newPath,
|
|
180
|
+
shouldSkipDuplicateCheck: true
|
|
181
|
+
});
|
|
177
182
|
abortSignal.throwIfAborted();
|
|
178
183
|
} else {
|
|
179
184
|
const relativePath = isOldAttachmentFolderAtRoot ? oldAttachmentFile.path : relative(oldAttachmentFolderPath, oldAttachmentFile.path);
|
|
@@ -553,4 +558,4 @@ export {
|
|
|
553
558
|
EmptyAttachmentFolderBehavior,
|
|
554
559
|
registerRenameDeleteHandlers
|
|
555
560
|
};
|
|
556
|
-
//# 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\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 { Vault } from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type {\n  UpdateLinkOptions,\n  UpdateLinksInFileOptions\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  getAttachmentFilePath,\n  getAttachmentFolderPath,\n  hasOwnAttachmentFolder\n} from './AttachmentPath.ts';\nimport {\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  tempRegisterFilesAndRun\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  deleteEmptyFolderHierarchy,\n  deleteSafe\n} from './VaultEx.ts';\n\nconst deletedMetadataCacheMap = new Map<string, CachedMetadata>();\nconst handledRenames = new Set<string>();\nconst interruptedRenamesMap = new Map<string, InterruptedRename[]>();\n\n/**\n * A behavior of the rename/delete handler when deleting empty attachment folders.\n */\nexport enum EmptyAttachmentFolderBehavior {\n  /**\n   * Delete the empty attachment folder.\n   */\n  Delete = 'Delete',\n\n  /**\n   * Delete the empty attachment folder and all its empty parents.\n   */\n  DeleteWithEmptyParents = 'DeleteWithEmptyParents',\n\n  /**\n   * Keep the empty attachment 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 attachment folders.\n   */\n  emptyAttachmentFolderBehavior: EmptyAttachmentFolderBehavior;\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 InterruptedRename {\n  combinedBacklinksMap: Map<string, Map<string, string>>;\n  oldPath: string;\n}\n\ntype RunAsyncLinkUpdateFn = FileManager['runAsyncLinkUpdate'];\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  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(plugin.app);\n  const pluginId = plugin.manifest.id;\n\n  renameDeleteHandlersMap.set(pluginId, settingsBuilder);\n  logRegisteredHandlers(plugin.app);\n\n  plugin.register(() => {\n    renameDeleteHandlersMap.delete(pluginId);\n    logRegisteredHandlers(plugin.app);\n  });\n\n  const app = plugin.app;\n  const abortSignal = plugin.abortSignal ?? abortSignalNever();\n\n  plugin.registerEvent(\n    app.vault.on('delete', (file) => {\n      handleDeleteIfEnabled(plugin, file, abortSignal);\n    })\n  );\n\n  plugin.registerEvent(\n    app.vault.on('rename', (file, oldPath) => {\n      handleRenameIfEnabled(plugin, file, oldPath, abortSignal);\n    })\n  );\n\n  plugin.registerEvent(\n    app.metadataCache.on('deleted', (file, prevCache) => {\n      handleMetadataDeletedIfEnabled(plugin, file, prevCache);\n    })\n  );\n\n  registerPatch(plugin, app.fileManager, {\n    runAsyncLinkUpdate: (next: RunAsyncLinkUpdateFn): RunAsyncLinkUpdateFn => {\n      return (linkUpdatesHandler) => runAsyncLinkUpdate(app, next, linkUpdatesHandler);\n    }\n  });\n}\n\nasync function cleanupParentFolders(app: App, parentFolderPaths: string[], notePath: string): Promise<void> {\n  const settings = getSettings(app);\n  if (settings.emptyAttachmentFolderBehavior === EmptyAttachmentFolderBehavior.Keep) {\n    return;\n  }\n  for (const parentFolderPath of parentFolderPaths) {\n    switch (settings.emptyAttachmentFolderBehavior) {\n      case EmptyAttachmentFolderBehavior.Delete:\n        await deleteSafe(app, parentFolderPath, notePath, undefined, true);\n        break;\n      case EmptyAttachmentFolderBehavior.DeleteWithEmptyParents:\n        await deleteEmptyFolderHierarchy(app, parentFolderPath);\n        break;\n      default:\n        break;\n    }\n  }\n}\n\nasync function continueInterruptedRenames(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<void> {\n  const interruptedRenames = interruptedRenamesMap.get(oldPath);\n  if (interruptedRenames) {\n    interruptedRenamesMap.delete(oldPath);\n    for (const interruptedRename of interruptedRenames) {\n      await handleRenameAsync(app, interruptedRename.oldPath, newPath, oldPathBacklinksMap, oldPathLinks, interruptedRename.combinedBacklinksMap);\n    }\n  }\n}\n\nasync function fillRenameMap(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  renameMap: Map<string, string>,\n  oldPathLinks: Reference[],\n  abortSignal: AbortSignal\n): Promise<void> {\n  abortSignal.throwIfAborted();\n  renameMap.set(oldPath, newPath);\n\n  if (!isNoteEx(app, oldPath)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n\n  const oldAttachmentFolderPath = await getAttachmentFolderPath(app, oldPath);\n  const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n    ? await getAttachmentFolderPath(app, newPath)\n    : oldAttachmentFolderPath;\n\n  const isOldAttachmentFolderAtRoot = oldAttachmentFolderPath === '/';\n\n  const oldAttachmentFolder = getFolderOrNull(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(app, oldPath)) {\n    Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {\n      abortSignal.throwIfAborted();\n      if (isFile(oldAttachmentFile)) {\n        oldAttachmentFiles.push(oldAttachmentFile);\n      }\n    });\n  } else {\n    for (const oldPathLink of oldPathLinks) {\n      abortSignal.throwIfAborted();\n      const oldAttachmentFile = extractLinkFile(app, oldPathLink, oldPath);\n      if (!oldAttachmentFile) {\n        continue;\n      }\n\n      if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n        const oldAttachmentBacklinks = await getBacklinksForFileSafe(app, oldAttachmentFile);\n        abortSignal.throwIfAborted();\n        if (oldAttachmentBacklinks.keys().length === 1) {\n          oldAttachmentFiles.push(oldAttachmentFile);\n        }\n      }\n    }\n  }\n\n  for (const oldAttachmentFile of oldAttachmentFiles) {\n    abortSignal.throwIfAborted();\n    if (isNoteEx(app, oldAttachmentFile.path)) {\n      continue;\n    }\n\n    let newAttachmentFilePath: string;\n    if (settings.shouldRenameAttachmentFiles) {\n      newAttachmentFilePath = await getAttachmentFilePath(app, oldAttachmentFile, newPath, true);\n      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(app, newAttachmentFilePath);\n      if (newAttachmentFile) {\n        console.warn(`Removing conflicting attachment ${newAttachmentFile.path}.`);\n        await app.fileManager.trashFile(newAttachmentFile);\n        abortSignal.throwIfAborted();\n      }\n    } else {\n      const dir = dirname(newAttachmentFilePath);\n      const ext = extname(newAttachmentFilePath);\n      const baseName = basename(newAttachmentFilePath, ext);\n      newAttachmentFilePath = app.vault.getAvailablePath(join(dir, baseName), ext.slice(1));\n    }\n    renameMap.set(oldAttachmentFile.path, newAttachmentFilePath);\n  }\n}\n\nfunction getRenameDeleteHandlersMap(app: App): Map<string, () => Partial<RenameDeleteHandlerSettings>> {\n  return getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n}\n\nfunction getSettings(app: App): Partial<RenameDeleteHandlerSettings> {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  const settingsBuilders = Array.from(renameDeleteHandlersMap.values()).reverse();\n\n  const settings: Partial<RenameDeleteHandlerSettings> = {};\n  settings.isNote = (path: string): boolean => isNote(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.emptyAttachmentFolderBehavior) {\n      settings.emptyAttachmentFolderBehavior ??= newSettings.emptyAttachmentFolderBehavior;\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.emptyAttachmentFolderBehavior ??= EmptyAttachmentFolderBehavior.Keep;\n  return settings;\n}\n\nasync function handleCaseCollision(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<boolean> {\n  if (!app.vault.adapter.insensitive || oldPath.toLowerCase() !== newPath.toLowerCase()) {\n    return false;\n  }\n\n  const tempPath = join(dirname(newPath), `__temp__${basename(newPath)}`);\n  await renameHandled(app, newPath, tempPath);\n  await handleRenameAsync(app, oldPath, tempPath, oldPathBacklinksMap, oldPathLinks);\n  await app.vault.rename(getFile(app, tempPath), newPath);\n  return true;\n}\n\nasync function handleDelete(app: App, path: string, abortSignal: AbortSignal): Promise<void> {\n  abortSignal.throwIfAborted();\n  getLibDebugger('RenameDeleteHandler:handleDelete')(`Handle Delete ${path}`);\n  if (!isNoteEx(app, path)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldHandleDeletions) {\n    return;\n  }\n\n  if (settings.isPathIgnored?.(path)) {\n    console.warn(`Skipping delete handler of ${path} as the path is ignored.`);\n    return;\n  }\n\n  const cache = deletedMetadataCacheMap.get(path);\n  deletedMetadataCacheMap.delete(path);\n  const parentFolderPaths = new Set<string>();\n  if (cache) {\n    const links = getAllLinks(cache);\n\n    for (const link of links) {\n      const attachmentFile = extractLinkFile(app, link, path);\n      if (!attachmentFile) {\n        continue;\n      }\n\n      if (isNoteEx(app, attachmentFile.path)) {\n        continue;\n      }\n\n      parentFolderPaths.add(attachmentFile.parent?.path ?? '');\n      await deleteSafe(app, attachmentFile, path, false, settings.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.Keep);\n      abortSignal.throwIfAborted();\n    }\n  }\n\n  await cleanupParentFolders(app, Array.from(parentFolderPaths), path);\n  abortSignal.throwIfAborted();\n\n  const attachmentFolderPath = await getAttachmentFolderPath(app, path);\n  const attachmentFolder = getFolderOrNull(app, attachmentFolderPath);\n\n  if (!attachmentFolder) {\n    return;\n  }\n\n  if (!(await hasOwnAttachmentFolder(app, path))) {\n    return;\n  }\n\n  abortSignal.throwIfAborted();\n\n  await deleteSafe(app, attachmentFolder, path, false, settings.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.Keep);\n  abortSignal.throwIfAborted();\n}\n\nfunction handleDeleteIfEnabled(plugin: AbortablePlugin, file: TAbstractFile, abortSignal: AbortSignal): void {\n  const app = plugin.app;\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  const path = file.path;\n  addToQueue(app, (abortSignal2) => handleDelete(app, path, abortSignal2), abortSignal);\n}\n\nfunction handleMetadataDeleted(app: App, file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  const settings = getSettings(app);\n  if (settings.isPathIgnored?.(file.path)) {\n    console.warn(`Skipping metadata delete handler of ${file.path} as the path is ignored.`);\n    return;\n  }\n\n  if (!settings.shouldHandleDeletions) {\n    return;\n  }\n  if (isMarkdownFile(app, file) && prevCache) {\n    deletedMetadataCacheMap.set(file.path, prevCache);\n  }\n}\n\nfunction handleMetadataDeletedIfEnabled(plugin: Plugin, file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  handleMetadataDeleted(plugin.app, file, prevCache);\n}\n\nfunction handleRename(app: App, oldPath: string, newPath: string, abortSignal: AbortSignal): void {\n  const key = makeKey(oldPath, newPath);\n  getLibDebugger('RenameDeleteHandler:handleRename')(`Handle Rename ${key}`);\n  if (handledRenames.has(key)) {\n    handledRenames.delete(key);\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldHandleRenames) {\n    return;\n  }\n\n  if (settings.isPathIgnored?.(oldPath)) {\n    console.warn(`Skipping rename handler of old path ${oldPath} as the path is ignored.`);\n    return;\n  }\n\n  if (settings.isPathIgnored?.(newPath)) {\n    console.warn(`Skipping rename handler of new path ${newPath} as the path is ignored.`);\n    return;\n  }\n\n  const cache = app.metadataCache.getCache(oldPath) ?? app.metadataCache.getCache(newPath);\n  const oldPathLinks = cache ? getAllLinks(cache) : [];\n  const oldPathBacklinksMap = getBacklinksForFileOrPath(app, oldPath).data;\n  addToQueue(app, (abortSignal2) => handleRenameAsync(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks, undefined, abortSignal2), abortSignal);\n}\n\nasync function handleRenameAsync(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[],\n  interruptedCombinedBacklinksMap?: Map<string, Map<string, string>>,\n  abortSignal?: AbortSignal\n): Promise<void> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  await continueInterruptedRenames(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks);\n  abortSignal.throwIfAborted();\n  await refreshLinks(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks);\n  abortSignal.throwIfAborted();\n  if (await handleCaseCollision(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks)) {\n    return;\n  }\n\n  abortSignal.throwIfAborted();\n\n  try {\n    const renameMap = new Map<string, string>();\n    await fillRenameMap(app, oldPath, newPath, renameMap, oldPathLinks, abortSignal);\n    abortSignal.throwIfAborted();\n\n    const combinedBacklinksMap = new Map<string, Map<string, string>>();\n    initBacklinksMap(oldPathBacklinksMap, renameMap, combinedBacklinksMap, oldPath);\n\n    for (const attachmentOldPath of renameMap.keys()) {\n      if (attachmentOldPath === oldPath) {\n        continue;\n      }\n      const attachmentOldPathBacklinksMap = (await getBacklinksForFileSafe(app, attachmentOldPath)).data;\n      initBacklinksMap(attachmentOldPathBacklinksMap, renameMap, combinedBacklinksMap, attachmentOldPath);\n    }\n\n    const parentFolderPaths = new Set<string>();\n\n    for (const [oldAttachmentPath, newAttachmentPath] of renameMap.entries()) {\n      if (oldAttachmentPath === oldPath) {\n        continue;\n      }\n      const fixedNewAttachmentPath = await renameHandled(app, oldAttachmentPath, newAttachmentPath);\n      renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);\n      parentFolderPaths.add(dirname(oldAttachmentPath));\n    }\n\n    await cleanupParentFolders(app, Array.from(parentFolderPaths), oldPath);\n    abortSignal.throwIfAborted();\n    const settings = getSettings(app);\n\n    for (\n      const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(\n        Array.from(interruptedCombinedBacklinksMap?.entries() ?? [])\n      )\n    ) {\n      await editLinks(app, newBacklinkPath, (link) => {\n        const oldAttachmentPath = linkJsonToPathMap.get(toJson(link));\n        if (!oldAttachmentPath) {\n          return;\n        }\n\n        const newAttachmentPath = renameMap.get(oldAttachmentPath);\n        if (!newAttachmentPath) {\n          return;\n        }\n\n        return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n          app,\n          link,\n          newSourcePathOrFile: newBacklinkPath,\n          newTargetPathOrFile: newAttachmentPath,\n          oldTargetPathOrFile: oldAttachmentPath,\n          shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n        }));\n      }, {\n        shouldFailOnMissingFile: false\n      });\n      abortSignal.throwIfAborted();\n    }\n\n    if (isNoteEx(app, newPath)) {\n      await updateLinksInFile(normalizeOptionalProperties<UpdateLinksInFileOptions>({\n        app,\n        newSourcePathOrFile: newPath,\n        oldSourcePathOrFile: oldPath,\n        shouldFailOnMissingFile: false,\n        shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n      }));\n      abortSignal.throwIfAborted();\n    }\n\n    if (!getFileOrNull(app, newPath)) {\n      let interruptedRenames = interruptedRenamesMap.get(newPath);\n      if (!interruptedRenames) {\n        interruptedRenames = [];\n        interruptedRenamesMap.set(newPath, interruptedRenames);\n      }\n      interruptedRenames.push({\n        combinedBacklinksMap,\n        oldPath\n      });\n    }\n  } finally {\n    const orphanKeys = Array.from(handledRenames);\n    addToQueue(app, () => {\n      for (const key of orphanKeys) {\n        handledRenames.delete(key);\n      }\n    }, abortSignal);\n  }\n}\n\nfunction handleRenameIfEnabled(plugin: Plugin, file: TAbstractFile, oldPath: string, abortSignal: AbortSignal): void {\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  if (!isFile(file)) {\n    return;\n  }\n  const newPath = file.path;\n  handleRename(plugin.app, oldPath, newPath, abortSignal);\n}\n\nfunction initBacklinksMap(\n  singleBacklinksMap: Map<string, Reference[]>,\n  renameMap: Map<string, string>,\n  combinedBacklinksMap: Map<string, Map<string, string>>,\n  path: string\n): void {\n  for (const [backlinkPath, links] of singleBacklinksMap.entries()) {\n    const newBacklinkPath = renameMap.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\nfunction isNoteEx(app: App, path: string): boolean {\n  const settings = getSettings(app);\n  return settings.isNote?.(path) ?? false;\n}\n\nfunction logRegisteredHandlers(app: App): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  getLibDebugger('RenameDeleteHandler:logRegisteredHandlers')(\n    `Plugins with registered rename/delete handlers: ${JSON.stringify(Array.from(renameDeleteHandlersMap.keys()))}`\n  );\n}\n\nfunction makeKey(oldPath: string, newPath: string): string {\n  return `${oldPath} -> ${newPath}`;\n}\n\nasync function refreshLinks(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<void> {\n  const cache = app.metadataCache.getCache(oldPath) ?? app.metadataCache.getCache(newPath);\n  const oldPathLinksRefreshed = cache ? getAllLinks(cache) : [];\n  const fakeOldFile = getFile(app, oldPath, true);\n  let oldPathBacklinksMapRefreshed = new Map<string, Reference[]>();\n  await tempRegisterFilesAndRun(app, [fakeOldFile], async () => {\n    oldPathBacklinksMapRefreshed = (await getBacklinksForFileSafe(app, fakeOldFile)).data;\n  });\n\n  for (const link of oldPathLinksRefreshed) {\n    if (oldPathLinks.includes(link)) {\n      continue;\n    }\n    oldPathLinks.push(link);\n  }\n\n  for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {\n    let oldLinks = oldPathBacklinksMap.get(backlinkPath);\n    if (!oldLinks) {\n      oldLinks = [];\n      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\nasync function renameHandled(app: App, oldPath: string, newPath: string): Promise<string> {\n  newPath = getSafeRenamePath(app, oldPath, newPath);\n  if (oldPath === newPath) {\n    return newPath;\n  }\n  const key = makeKey(oldPath, newPath);\n  handledRenames.add(key);\n  newPath = await renameSafe(app, oldPath, newPath);\n  return newPath;\n}\n\nasync function runAsyncLinkUpdate(app: App, next: RunAsyncLinkUpdateFn, linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n  await next.call(app.fileManager, wrappedHandler);\n\n  async function wrappedHandler(linkUpdates: LinkUpdate[]): Promise<void> {\n    let isRenameCalled = false;\n    const eventRef = app.vault.on('rename', () => {\n      isRenameCalled = true;\n    });\n    try {\n      await linkUpdatesHandler(linkUpdates);\n    } finally {\n      app.vault.offref(eventRef);\n    }\n    const settings = getSettings(app);\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    if (isRenameCalled && settings.shouldHandleRenames) {\n      filterInPlace(\n        linkUpdates,\n        (linkUpdate) => {\n          if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {\n            console.warn(`Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`);\n            return true;\n          }\n\n          if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {\n            console.warn(`Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`);\n            return true;\n          }\n\n          if (!app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n            return false;\n          }\n\n          if (app.plugins.getPlugin('backlink-cache')) {\n            return false;\n          }\n\n          if (linkUpdate.sourceFile.extension === 'canvas') {\n            return true;\n          }\n\n          if (linkUpdate.resolvedFile.extension === 'canvas') {\n            return true;\n          }\n\n          return false;\n        }\n      );\n    }\n  }\n}\n\nfunction shouldInvokeHandler(plugin: Plugin): boolean {\n  const app = plugin.app;\n  const pluginId = plugin.manifest.id;\n\n  const renameDeleteHandlerPluginIds = getRenameDeleteHandlersMap(app);\n  const mainPluginId = Array.from(renameDeleteHandlerPluginIds.keys())[0];\n  if (mainPluginId !== pluginId) {\n    return false;\n  }\n  return true;\n}\n"],
  "mappings": "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",
  "names": ["EmptyAttachmentFolderBehavior"]
}

|
|
561
|
+
//# 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\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 { Vault } from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type {\n  UpdateLinkOptions,\n  UpdateLinksInFileOptions\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  getAttachmentFilePath,\n  getAttachmentFolderPath,\n  hasOwnAttachmentFolder\n} from './AttachmentPath.ts';\nimport {\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  tempRegisterFilesAndRun\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  deleteEmptyFolderHierarchy,\n  deleteSafe\n} from './VaultEx.ts';\n\nconst deletedMetadataCacheMap = new Map<string, CachedMetadata>();\nconst handledRenames = new Set<string>();\nconst interruptedRenamesMap = new Map<string, InterruptedRename[]>();\n\n/**\n * A behavior of the rename/delete handler when deleting empty attachment folders.\n */\nexport enum EmptyAttachmentFolderBehavior {\n  /**\n   * Delete the empty attachment folder.\n   */\n  Delete = 'Delete',\n\n  /**\n   * Delete the empty attachment folder and all its empty parents.\n   */\n  DeleteWithEmptyParents = 'DeleteWithEmptyParents',\n\n  /**\n   * Keep the empty attachment 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 attachment folders.\n   */\n  emptyAttachmentFolderBehavior: EmptyAttachmentFolderBehavior;\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 InterruptedRename {\n  combinedBacklinksMap: Map<string, Map<string, string>>;\n  oldPath: string;\n}\n\ntype RunAsyncLinkUpdateFn = FileManager['runAsyncLinkUpdate'];\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  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(plugin.app);\n  const pluginId = plugin.manifest.id;\n\n  renameDeleteHandlersMap.set(pluginId, settingsBuilder);\n  logRegisteredHandlers(plugin.app);\n\n  plugin.register(() => {\n    renameDeleteHandlersMap.delete(pluginId);\n    logRegisteredHandlers(plugin.app);\n  });\n\n  const app = plugin.app;\n  const abortSignal = plugin.abortSignal ?? abortSignalNever();\n\n  plugin.registerEvent(\n    app.vault.on('delete', (file) => {\n      handleDeleteIfEnabled(plugin, file, abortSignal);\n    })\n  );\n\n  plugin.registerEvent(\n    app.vault.on('rename', (file, oldPath) => {\n      handleRenameIfEnabled(plugin, file, oldPath, abortSignal);\n    })\n  );\n\n  plugin.registerEvent(\n    app.metadataCache.on('deleted', (file, prevCache) => {\n      handleMetadataDeletedIfEnabled(plugin, file, prevCache);\n    })\n  );\n\n  registerPatch(plugin, app.fileManager, {\n    runAsyncLinkUpdate: (next: RunAsyncLinkUpdateFn): RunAsyncLinkUpdateFn => {\n      return (linkUpdatesHandler) => runAsyncLinkUpdate(app, next, linkUpdatesHandler);\n    }\n  });\n}\n\nasync function cleanupParentFolders(app: App, parentFolderPaths: string[], notePath: string): Promise<void> {\n  const settings = getSettings(app);\n  if (settings.emptyAttachmentFolderBehavior === EmptyAttachmentFolderBehavior.Keep) {\n    return;\n  }\n  for (const parentFolderPath of parentFolderPaths) {\n    switch (settings.emptyAttachmentFolderBehavior) {\n      case EmptyAttachmentFolderBehavior.Delete:\n        await deleteSafe(app, parentFolderPath, notePath, undefined, true);\n        break;\n      case EmptyAttachmentFolderBehavior.DeleteWithEmptyParents:\n        await deleteEmptyFolderHierarchy(app, parentFolderPath);\n        break;\n      default:\n        break;\n    }\n  }\n}\n\nasync function continueInterruptedRenames(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<void> {\n  const interruptedRenames = interruptedRenamesMap.get(oldPath);\n  if (interruptedRenames) {\n    interruptedRenamesMap.delete(oldPath);\n    for (const interruptedRename of interruptedRenames) {\n      await handleRenameAsync(app, interruptedRename.oldPath, newPath, oldPathBacklinksMap, oldPathLinks, interruptedRename.combinedBacklinksMap);\n    }\n  }\n}\n\nasync function fillRenameMap(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  renameMap: Map<string, string>,\n  oldPathLinks: Reference[],\n  abortSignal: AbortSignal\n): Promise<void> {\n  abortSignal.throwIfAborted();\n  renameMap.set(oldPath, newPath);\n\n  if (!isNoteEx(app, oldPath)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n\n  const oldAttachmentFolderPath = await getAttachmentFolderPath(app, oldPath);\n  const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n    ? await getAttachmentFolderPath(app, newPath)\n    : oldAttachmentFolderPath;\n\n  const isOldAttachmentFolderAtRoot = oldAttachmentFolderPath === '/';\n\n  const oldAttachmentFolder = getFolderOrNull(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(app, oldPath)) {\n    Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {\n      abortSignal.throwIfAborted();\n      if (isFile(oldAttachmentFile)) {\n        oldAttachmentFiles.push(oldAttachmentFile);\n      }\n    });\n  } else {\n    for (const oldPathLink of oldPathLinks) {\n      abortSignal.throwIfAborted();\n      const oldAttachmentFile = extractLinkFile(app, oldPathLink, oldPath);\n      if (!oldAttachmentFile) {\n        continue;\n      }\n\n      if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n        const oldAttachmentBacklinks = await getBacklinksForFileSafe(app, oldAttachmentFile);\n        abortSignal.throwIfAborted();\n        if (oldAttachmentBacklinks.keys().length === 1) {\n          oldAttachmentFiles.push(oldAttachmentFile);\n        }\n      }\n    }\n  }\n\n  for (const oldAttachmentFile of oldAttachmentFiles) {\n    abortSignal.throwIfAborted();\n    if (isNoteEx(app, oldAttachmentFile.path)) {\n      continue;\n    }\n\n    let newAttachmentFilePath: string;\n    if (settings.shouldRenameAttachmentFiles) {\n      newAttachmentFilePath = await getAttachmentFilePath({\n        app,\n        attachmentPathOrFile: oldAttachmentFile,\n        notePathOrFile: newPath,\n        shouldSkipDuplicateCheck: true\n      });\n      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(app, newAttachmentFilePath);\n      if (newAttachmentFile) {\n        console.warn(`Removing conflicting attachment ${newAttachmentFile.path}.`);\n        await app.fileManager.trashFile(newAttachmentFile);\n        abortSignal.throwIfAborted();\n      }\n    } else {\n      const dir = dirname(newAttachmentFilePath);\n      const ext = extname(newAttachmentFilePath);\n      const baseName = basename(newAttachmentFilePath, ext);\n      newAttachmentFilePath = app.vault.getAvailablePath(join(dir, baseName), ext.slice(1));\n    }\n    renameMap.set(oldAttachmentFile.path, newAttachmentFilePath);\n  }\n}\n\nfunction getRenameDeleteHandlersMap(app: App): Map<string, () => Partial<RenameDeleteHandlerSettings>> {\n  return getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n}\n\nfunction getSettings(app: App): Partial<RenameDeleteHandlerSettings> {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  const settingsBuilders = Array.from(renameDeleteHandlersMap.values()).reverse();\n\n  const settings: Partial<RenameDeleteHandlerSettings> = {};\n  settings.isNote = (path: string): boolean => isNote(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.emptyAttachmentFolderBehavior) {\n      settings.emptyAttachmentFolderBehavior ??= newSettings.emptyAttachmentFolderBehavior;\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.emptyAttachmentFolderBehavior ??= EmptyAttachmentFolderBehavior.Keep;\n  return settings;\n}\n\nasync function handleCaseCollision(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<boolean> {\n  if (!app.vault.adapter.insensitive || oldPath.toLowerCase() !== newPath.toLowerCase()) {\n    return false;\n  }\n\n  const tempPath = join(dirname(newPath), `__temp__${basename(newPath)}`);\n  await renameHandled(app, newPath, tempPath);\n  await handleRenameAsync(app, oldPath, tempPath, oldPathBacklinksMap, oldPathLinks);\n  await app.vault.rename(getFile(app, tempPath), newPath);\n  return true;\n}\n\nasync function handleDelete(app: App, path: string, abortSignal: AbortSignal): Promise<void> {\n  abortSignal.throwIfAborted();\n  getLibDebugger('RenameDeleteHandler:handleDelete')(`Handle Delete ${path}`);\n  if (!isNoteEx(app, path)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldHandleDeletions) {\n    return;\n  }\n\n  if (settings.isPathIgnored?.(path)) {\n    console.warn(`Skipping delete handler of ${path} as the path is ignored.`);\n    return;\n  }\n\n  const cache = deletedMetadataCacheMap.get(path);\n  deletedMetadataCacheMap.delete(path);\n  const parentFolderPaths = new Set<string>();\n  if (cache) {\n    const links = getAllLinks(cache);\n\n    for (const link of links) {\n      const attachmentFile = extractLinkFile(app, link, path);\n      if (!attachmentFile) {\n        continue;\n      }\n\n      if (isNoteEx(app, attachmentFile.path)) {\n        continue;\n      }\n\n      parentFolderPaths.add(attachmentFile.parent?.path ?? '');\n      await deleteSafe(app, attachmentFile, path, false, settings.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.Keep);\n      abortSignal.throwIfAborted();\n    }\n  }\n\n  await cleanupParentFolders(app, Array.from(parentFolderPaths), path);\n  abortSignal.throwIfAborted();\n\n  const attachmentFolderPath = await getAttachmentFolderPath(app, path);\n  const attachmentFolder = getFolderOrNull(app, attachmentFolderPath);\n\n  if (!attachmentFolder) {\n    return;\n  }\n\n  if (!(await hasOwnAttachmentFolder(app, path))) {\n    return;\n  }\n\n  abortSignal.throwIfAborted();\n\n  await deleteSafe(app, attachmentFolder, path, false, settings.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.Keep);\n  abortSignal.throwIfAborted();\n}\n\nfunction handleDeleteIfEnabled(plugin: AbortablePlugin, file: TAbstractFile, abortSignal: AbortSignal): void {\n  const app = plugin.app;\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  const path = file.path;\n  addToQueue(app, (abortSignal2) => handleDelete(app, path, abortSignal2), abortSignal);\n}\n\nfunction handleMetadataDeleted(app: App, file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  const settings = getSettings(app);\n  if (settings.isPathIgnored?.(file.path)) {\n    console.warn(`Skipping metadata delete handler of ${file.path} as the path is ignored.`);\n    return;\n  }\n\n  if (!settings.shouldHandleDeletions) {\n    return;\n  }\n  if (isMarkdownFile(app, file) && prevCache) {\n    deletedMetadataCacheMap.set(file.path, prevCache);\n  }\n}\n\nfunction handleMetadataDeletedIfEnabled(plugin: Plugin, file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  handleMetadataDeleted(plugin.app, file, prevCache);\n}\n\nfunction handleRename(app: App, oldPath: string, newPath: string, abortSignal: AbortSignal): void {\n  const key = makeKey(oldPath, newPath);\n  getLibDebugger('RenameDeleteHandler:handleRename')(`Handle Rename ${key}`);\n  if (handledRenames.has(key)) {\n    handledRenames.delete(key);\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldHandleRenames) {\n    return;\n  }\n\n  if (settings.isPathIgnored?.(oldPath)) {\n    console.warn(`Skipping rename handler of old path ${oldPath} as the path is ignored.`);\n    return;\n  }\n\n  if (settings.isPathIgnored?.(newPath)) {\n    console.warn(`Skipping rename handler of new path ${newPath} as the path is ignored.`);\n    return;\n  }\n\n  const cache = app.metadataCache.getCache(oldPath) ?? app.metadataCache.getCache(newPath);\n  const oldPathLinks = cache ? getAllLinks(cache) : [];\n  const oldPathBacklinksMap = getBacklinksForFileOrPath(app, oldPath).data;\n  addToQueue(app, (abortSignal2) => handleRenameAsync(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks, undefined, abortSignal2), abortSignal);\n}\n\nasync function handleRenameAsync(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[],\n  interruptedCombinedBacklinksMap?: Map<string, Map<string, string>>,\n  abortSignal?: AbortSignal\n): Promise<void> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  await continueInterruptedRenames(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks);\n  abortSignal.throwIfAborted();\n  await refreshLinks(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks);\n  abortSignal.throwIfAborted();\n  if (await handleCaseCollision(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks)) {\n    return;\n  }\n\n  abortSignal.throwIfAborted();\n\n  try {\n    const renameMap = new Map<string, string>();\n    await fillRenameMap(app, oldPath, newPath, renameMap, oldPathLinks, abortSignal);\n    abortSignal.throwIfAborted();\n\n    const combinedBacklinksMap = new Map<string, Map<string, string>>();\n    initBacklinksMap(oldPathBacklinksMap, renameMap, combinedBacklinksMap, oldPath);\n\n    for (const attachmentOldPath of renameMap.keys()) {\n      if (attachmentOldPath === oldPath) {\n        continue;\n      }\n      const attachmentOldPathBacklinksMap = (await getBacklinksForFileSafe(app, attachmentOldPath)).data;\n      initBacklinksMap(attachmentOldPathBacklinksMap, renameMap, combinedBacklinksMap, attachmentOldPath);\n    }\n\n    const parentFolderPaths = new Set<string>();\n\n    for (const [oldAttachmentPath, newAttachmentPath] of renameMap.entries()) {\n      if (oldAttachmentPath === oldPath) {\n        continue;\n      }\n      const fixedNewAttachmentPath = await renameHandled(app, oldAttachmentPath, newAttachmentPath);\n      renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);\n      parentFolderPaths.add(dirname(oldAttachmentPath));\n    }\n\n    await cleanupParentFolders(app, Array.from(parentFolderPaths), oldPath);\n    abortSignal.throwIfAborted();\n    const settings = getSettings(app);\n\n    for (\n      const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(\n        Array.from(interruptedCombinedBacklinksMap?.entries() ?? [])\n      )\n    ) {\n      await editLinks(app, newBacklinkPath, (link) => {\n        const oldAttachmentPath = linkJsonToPathMap.get(toJson(link));\n        if (!oldAttachmentPath) {\n          return;\n        }\n\n        const newAttachmentPath = renameMap.get(oldAttachmentPath);\n        if (!newAttachmentPath) {\n          return;\n        }\n\n        return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n          app,\n          link,\n          newSourcePathOrFile: newBacklinkPath,\n          newTargetPathOrFile: newAttachmentPath,\n          oldTargetPathOrFile: oldAttachmentPath,\n          shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n        }));\n      }, {\n        shouldFailOnMissingFile: false\n      });\n      abortSignal.throwIfAborted();\n    }\n\n    if (isNoteEx(app, newPath)) {\n      await updateLinksInFile(normalizeOptionalProperties<UpdateLinksInFileOptions>({\n        app,\n        newSourcePathOrFile: newPath,\n        oldSourcePathOrFile: oldPath,\n        shouldFailOnMissingFile: false,\n        shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases\n      }));\n      abortSignal.throwIfAborted();\n    }\n\n    if (!getFileOrNull(app, newPath)) {\n      let interruptedRenames = interruptedRenamesMap.get(newPath);\n      if (!interruptedRenames) {\n        interruptedRenames = [];\n        interruptedRenamesMap.set(newPath, interruptedRenames);\n      }\n      interruptedRenames.push({\n        combinedBacklinksMap,\n        oldPath\n      });\n    }\n  } finally {\n    const orphanKeys = Array.from(handledRenames);\n    addToQueue(app, () => {\n      for (const key of orphanKeys) {\n        handledRenames.delete(key);\n      }\n    }, abortSignal);\n  }\n}\n\nfunction handleRenameIfEnabled(plugin: Plugin, file: TAbstractFile, oldPath: string, abortSignal: AbortSignal): void {\n  if (!shouldInvokeHandler(plugin)) {\n    return;\n  }\n  if (!isFile(file)) {\n    return;\n  }\n  const newPath = file.path;\n  handleRename(plugin.app, oldPath, newPath, abortSignal);\n}\n\nfunction initBacklinksMap(\n  singleBacklinksMap: Map<string, Reference[]>,\n  renameMap: Map<string, string>,\n  combinedBacklinksMap: Map<string, Map<string, string>>,\n  path: string\n): void {\n  for (const [backlinkPath, links] of singleBacklinksMap.entries()) {\n    const newBacklinkPath = renameMap.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\nfunction isNoteEx(app: App, path: string): boolean {\n  const settings = getSettings(app);\n  return settings.isNote?.(path) ?? false;\n}\n\nfunction logRegisteredHandlers(app: App): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  getLibDebugger('RenameDeleteHandler:logRegisteredHandlers')(\n    `Plugins with registered rename/delete handlers: ${JSON.stringify(Array.from(renameDeleteHandlersMap.keys()))}`\n  );\n}\n\nfunction makeKey(oldPath: string, newPath: string): string {\n  return `${oldPath} -> ${newPath}`;\n}\n\nasync function refreshLinks(\n  app: App,\n  oldPath: string,\n  newPath: string,\n  oldPathBacklinksMap: Map<string, Reference[]>,\n  oldPathLinks: Reference[]\n): Promise<void> {\n  const cache = app.metadataCache.getCache(oldPath) ?? app.metadataCache.getCache(newPath);\n  const oldPathLinksRefreshed = cache ? getAllLinks(cache) : [];\n  const fakeOldFile = getFile(app, oldPath, true);\n  let oldPathBacklinksMapRefreshed = new Map<string, Reference[]>();\n  await tempRegisterFilesAndRun(app, [fakeOldFile], async () => {\n    oldPathBacklinksMapRefreshed = (await getBacklinksForFileSafe(app, fakeOldFile)).data;\n  });\n\n  for (const link of oldPathLinksRefreshed) {\n    if (oldPathLinks.includes(link)) {\n      continue;\n    }\n    oldPathLinks.push(link);\n  }\n\n  for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {\n    let oldLinks = oldPathBacklinksMap.get(backlinkPath);\n    if (!oldLinks) {\n      oldLinks = [];\n      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\nasync function renameHandled(app: App, oldPath: string, newPath: string): Promise<string> {\n  newPath = getSafeRenamePath(app, oldPath, newPath);\n  if (oldPath === newPath) {\n    return newPath;\n  }\n  const key = makeKey(oldPath, newPath);\n  handledRenames.add(key);\n  newPath = await renameSafe(app, oldPath, newPath);\n  return newPath;\n}\n\nasync function runAsyncLinkUpdate(app: App, next: RunAsyncLinkUpdateFn, linkUpdatesHandler: LinkUpdatesHandler): Promise<void> {\n  await next.call(app.fileManager, wrappedHandler);\n\n  async function wrappedHandler(linkUpdates: LinkUpdate[]): Promise<void> {\n    let isRenameCalled = false;\n    const eventRef = app.vault.on('rename', () => {\n      isRenameCalled = true;\n    });\n    try {\n      await linkUpdatesHandler(linkUpdates);\n    } finally {\n      app.vault.offref(eventRef);\n    }\n    const settings = getSettings(app);\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    if (isRenameCalled && settings.shouldHandleRenames) {\n      filterInPlace(\n        linkUpdates,\n        (linkUpdate) => {\n          if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {\n            console.warn(`Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`);\n            return true;\n          }\n\n          if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {\n            console.warn(`Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`);\n            return true;\n          }\n\n          if (!app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n            return false;\n          }\n\n          if (app.plugins.getPlugin('backlink-cache')) {\n            return false;\n          }\n\n          if (linkUpdate.sourceFile.extension === 'canvas') {\n            return true;\n          }\n\n          if (linkUpdate.resolvedFile.extension === 'canvas') {\n            return true;\n          }\n\n          return false;\n        }\n      );\n    }\n  }\n}\n\nfunction shouldInvokeHandler(plugin: Plugin): boolean {\n  const app = plugin.app;\n  const pluginId = plugin.manifest.id;\n\n  const renameDeleteHandlerPluginIds = getRenameDeleteHandlersMap(app);\n  const mainPluginId = Array.from(renameDeleteHandlerPluginIds.keys())[0];\n  if (mainPluginId !== pluginId) {\n    return false;\n  }\n  return true;\n}\n"],
  "mappings": "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",
  "names": ["EmptyAttachmentFolderBehavior"]
}

|