obsidian-dev-utils 40.9.1 → 40.10.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/MetadataCache.cjs +29 -2
- package/dist/lib/cjs/obsidian/MetadataCache.d.cts +15 -0
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +595 -451
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/MetadataCache.d.mts +15 -0
- package/dist/lib/esm/obsidian/MetadataCache.mjs +27 -2
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +600 -452
- package/package.json +1 -1
|
@@ -45,512 +45,656 @@ var import_MonkeyAround = require('./MonkeyAround.cjs');
|
|
|
45
45
|
var import_Queue = require('./Queue.cjs');
|
|
46
46
|
var import_Vault = require('./Vault.cjs');
|
|
47
47
|
var import_VaultEx = require('./VaultEx.cjs');
|
|
48
|
-
const deletedMetadataCacheMap = /* @__PURE__ */ new Map();
|
|
49
|
-
const handledRenames = /* @__PURE__ */ new Set();
|
|
50
|
-
const interruptedRenamesMap = /* @__PURE__ */ new Map();
|
|
51
48
|
var EmptyAttachmentFolderBehavior = /* @__PURE__ */ ((EmptyAttachmentFolderBehavior2) => {
|
|
52
49
|
EmptyAttachmentFolderBehavior2["Delete"] = "Delete";
|
|
53
50
|
EmptyAttachmentFolderBehavior2["DeleteWithEmptyParents"] = "DeleteWithEmptyParents";
|
|
54
51
|
EmptyAttachmentFolderBehavior2["Keep"] = "Keep";
|
|
55
52
|
return EmptyAttachmentFolderBehavior2;
|
|
56
53
|
})(EmptyAttachmentFolderBehavior || {});
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
handleDeleteIfEnabled(plugin, file, abortSignal);
|
|
71
|
-
})
|
|
72
|
-
);
|
|
73
|
-
plugin.registerEvent(
|
|
74
|
-
app.vault.on("rename", (file, oldPath) => {
|
|
75
|
-
handleRenameIfEnabled(plugin, file, oldPath, abortSignal);
|
|
76
|
-
})
|
|
77
|
-
);
|
|
78
|
-
plugin.registerEvent(
|
|
79
|
-
app.metadataCache.on("deleted", (file, prevCache) => {
|
|
80
|
-
handleMetadataDeletedIfEnabled(plugin, file, prevCache);
|
|
81
|
-
})
|
|
82
|
-
);
|
|
83
|
-
(0, import_MonkeyAround.registerPatch)(plugin, app.fileManager, {
|
|
84
|
-
runAsyncLinkUpdate: (next) => {
|
|
85
|
-
return (linkUpdatesHandler) => runAsyncLinkUpdate(app, next, linkUpdatesHandler);
|
|
86
|
-
}
|
|
87
|
-
});
|
|
88
|
-
}
|
|
89
|
-
async function cleanupParentFolders(app, parentFolderPaths, notePath) {
|
|
90
|
-
const settings = getSettings(app);
|
|
91
|
-
if (settings.emptyAttachmentFolderBehavior === "Keep" /* Keep */) {
|
|
92
|
-
return;
|
|
93
|
-
}
|
|
94
|
-
for (const parentFolderPath of parentFolderPaths) {
|
|
95
|
-
switch (settings.emptyAttachmentFolderBehavior) {
|
|
96
|
-
case "Delete" /* Delete */:
|
|
97
|
-
await (0, import_VaultEx.deleteSafe)(app, parentFolderPath, notePath, void 0, true);
|
|
98
|
-
break;
|
|
99
|
-
case "DeleteWithEmptyParents" /* DeleteWithEmptyParents */:
|
|
100
|
-
await (0, import_VaultEx.deleteEmptyFolderHierarchy)(app, parentFolderPath);
|
|
101
|
-
break;
|
|
102
|
-
default:
|
|
103
|
-
break;
|
|
54
|
+
class DeleteHandler {
|
|
55
|
+
constructor(app, file, abortSignal, settingsManager, deletedMetadataCacheMap) {
|
|
56
|
+
this.app = app;
|
|
57
|
+
this.file = file;
|
|
58
|
+
this.abortSignal = abortSignal;
|
|
59
|
+
this.settingsManager = settingsManager;
|
|
60
|
+
this.deletedMetadataCacheMap = deletedMetadataCacheMap;
|
|
61
|
+
}
|
|
62
|
+
async handle() {
|
|
63
|
+
this.abortSignal.throwIfAborted();
|
|
64
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleDelete")(`Handle Delete ${this.file.path}`);
|
|
65
|
+
if (!this.settingsManager.isNoteEx(this.file.path)) {
|
|
66
|
+
return;
|
|
104
67
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
const interruptedRenames = interruptedRenamesMap.get(oldPath);
|
|
109
|
-
if (interruptedRenames) {
|
|
110
|
-
interruptedRenamesMap.delete(oldPath);
|
|
111
|
-
for (const interruptedRename of interruptedRenames) {
|
|
112
|
-
await handleRenameAsync(app, interruptedRename.oldPath, newPath, oldPathBacklinksMap, oldPathLinks, interruptedRename.combinedBacklinksMap);
|
|
68
|
+
const settings = this.settingsManager.getSettings();
|
|
69
|
+
if (!settings.shouldHandleDeletions) {
|
|
70
|
+
return;
|
|
113
71
|
}
|
|
114
|
-
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
if (oldAttachmentFolderPath === newAttachmentFolderPath && !settings.shouldRenameAttachmentFiles) {
|
|
131
|
-
return;
|
|
132
|
-
}
|
|
133
|
-
const oldAttachmentFiles = [];
|
|
134
|
-
if (await (0, import_AttachmentPath.hasOwnAttachmentFolder)(app, oldPath, import_AttachmentPath.AttachmentPathContext.RenameNote)) {
|
|
135
|
-
import_obsidian.Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {
|
|
136
|
-
abortSignal.throwIfAborted();
|
|
137
|
-
if ((0, import_FileSystem.isFile)(oldAttachmentFile)) {
|
|
138
|
-
oldAttachmentFiles.push(oldAttachmentFile);
|
|
139
|
-
}
|
|
140
|
-
});
|
|
141
|
-
} else {
|
|
142
|
-
for (const oldPathLink of oldPathLinks) {
|
|
143
|
-
abortSignal.throwIfAborted();
|
|
144
|
-
const oldAttachmentFile = (0, import_Link.extractLinkFile)(app, oldPathLink, oldPath);
|
|
145
|
-
if (!oldAttachmentFile) {
|
|
146
|
-
continue;
|
|
147
|
-
}
|
|
148
|
-
if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {
|
|
149
|
-
const oldAttachmentBacklinks = await (0, import_MetadataCache.getBacklinksForFileSafe)(app, oldAttachmentFile);
|
|
150
|
-
abortSignal.throwIfAborted();
|
|
151
|
-
if (oldAttachmentBacklinks.keys().length === 1) {
|
|
152
|
-
oldAttachmentFiles.push(oldAttachmentFile);
|
|
72
|
+
if (settings.isPathIgnored?.(this.file.path)) {
|
|
73
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleDelete")(`Skipping delete handler of ${this.file.path} as the path is ignored.`);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
const cache = this.deletedMetadataCacheMap.get(this.file.path);
|
|
77
|
+
this.deletedMetadataCacheMap.delete(this.file.path);
|
|
78
|
+
const parentFolderPaths = /* @__PURE__ */ new Set();
|
|
79
|
+
if (cache) {
|
|
80
|
+
const links = (0, import_MetadataCache.getAllLinks)(cache);
|
|
81
|
+
for (const link of links) {
|
|
82
|
+
const attachmentFile = (0, import_Link.extractLinkFile)(this.app, link, this.file.path);
|
|
83
|
+
if (!attachmentFile) {
|
|
84
|
+
continue;
|
|
85
|
+
}
|
|
86
|
+
if (this.settingsManager.isNoteEx(attachmentFile.path)) {
|
|
87
|
+
continue;
|
|
153
88
|
}
|
|
89
|
+
parentFolderPaths.add(attachmentFile.parent?.path ?? "");
|
|
90
|
+
await (0, import_VaultEx.deleteSafe)(this.app, attachmentFile, this.file.path, false, settings.emptyAttachmentFolderBehavior !== "Keep" /* Keep */);
|
|
154
91
|
}
|
|
155
92
|
}
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
if (
|
|
164
|
-
|
|
165
|
-
app,
|
|
166
|
-
attachmentPathOrFile: oldAttachmentFile,
|
|
167
|
-
context: import_AttachmentPath.AttachmentPathContext.RenameNote,
|
|
168
|
-
notePathOrFile: newPath,
|
|
169
|
-
oldNotePathOrFile: oldPath,
|
|
170
|
-
shouldSkipDuplicateCheck: true
|
|
171
|
-
});
|
|
172
|
-
abortSignal.throwIfAborted();
|
|
173
|
-
} else {
|
|
174
|
-
const relativePath = isOldAttachmentFolderAtRoot ? oldAttachmentFile.path : (0, import_Path.relative)(oldAttachmentFolderPath, oldAttachmentFile.path);
|
|
175
|
-
const newFolder = (0, import_Path.join)(newAttachmentFolderPath, (0, import_Path.dirname)(relativePath));
|
|
176
|
-
newAttachmentFilePath = (0, import_Path.join)(newFolder, oldAttachmentFile.name);
|
|
177
|
-
}
|
|
178
|
-
if (oldAttachmentFile.path === newAttachmentFilePath) {
|
|
179
|
-
continue;
|
|
180
|
-
}
|
|
181
|
-
if (settings.shouldDeleteConflictingAttachments) {
|
|
182
|
-
const newAttachmentFile = (0, import_FileSystem.getFileOrNull)(app, newAttachmentFilePath);
|
|
183
|
-
if (newAttachmentFile) {
|
|
184
|
-
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:fillRenameMap")(`Removing conflicting attachment ${newAttachmentFile.path}.`);
|
|
185
|
-
await app.fileManager.trashFile(newAttachmentFile);
|
|
186
|
-
abortSignal.throwIfAborted();
|
|
187
|
-
}
|
|
188
|
-
} else {
|
|
189
|
-
const dir = (0, import_Path.dirname)(newAttachmentFilePath);
|
|
190
|
-
const ext = (0, import_Path.extname)(newAttachmentFilePath);
|
|
191
|
-
const baseName = (0, import_Path.basename)(newAttachmentFilePath, ext);
|
|
192
|
-
newAttachmentFilePath = app.vault.getAvailablePath((0, import_Path.join)(dir, baseName), ext.slice(1));
|
|
93
|
+
await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths), this.file.path);
|
|
94
|
+
this.abortSignal.throwIfAborted();
|
|
95
|
+
const attachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(this.app, this.file.path, import_AttachmentPath.AttachmentPathContext.DeleteNote);
|
|
96
|
+
const attachmentFolder = (0, import_FileSystem.getFolderOrNull)(this.app, attachmentFolderPath);
|
|
97
|
+
if (!attachmentFolder) {
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
if (!await (0, import_AttachmentPath.hasOwnAttachmentFolder)(this.app, this.file.path, import_AttachmentPath.AttachmentPathContext.DeleteNote)) {
|
|
101
|
+
return;
|
|
193
102
|
}
|
|
194
|
-
|
|
103
|
+
this.abortSignal.throwIfAborted();
|
|
104
|
+
await (0, import_VaultEx.deleteSafe)(this.app, attachmentFolder, this.file.path, false, settings.emptyAttachmentFolderBehavior !== "Keep" /* Keep */);
|
|
105
|
+
this.abortSignal.throwIfAborted();
|
|
195
106
|
}
|
|
196
107
|
}
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);
|
|
202
|
-
const settingsBuilders = Array.from(renameDeleteHandlersMap.values()).reverse();
|
|
203
|
-
const settings = {};
|
|
204
|
-
settings.isNote = (path) => (0, import_FileSystem.isNote)(app, path);
|
|
205
|
-
settings.isPathIgnored = () => false;
|
|
206
|
-
for (const settingsBuilder of settingsBuilders) {
|
|
207
|
-
const newSettings = settingsBuilder();
|
|
208
|
-
settings.shouldDeleteConflictingAttachments ||= newSettings.shouldDeleteConflictingAttachments ?? false;
|
|
209
|
-
if (newSettings.emptyAttachmentFolderBehavior) {
|
|
210
|
-
settings.emptyAttachmentFolderBehavior ??= newSettings.emptyAttachmentFolderBehavior;
|
|
211
|
-
}
|
|
212
|
-
settings.shouldHandleDeletions ||= newSettings.shouldHandleDeletions ?? false;
|
|
213
|
-
settings.shouldHandleRenames ||= newSettings.shouldHandleRenames ?? false;
|
|
214
|
-
settings.shouldRenameAttachmentFiles ||= newSettings.shouldRenameAttachmentFiles ?? false;
|
|
215
|
-
settings.shouldRenameAttachmentFolder ||= newSettings.shouldRenameAttachmentFolder ?? false;
|
|
216
|
-
settings.shouldUpdateFileNameAliases ||= newSettings.shouldUpdateFileNameAliases ?? false;
|
|
217
|
-
const isPathIgnored = settings.isPathIgnored;
|
|
218
|
-
settings.isPathIgnored = (path) => isPathIgnored(path) || (newSettings.isPathIgnored?.(path) ?? false);
|
|
219
|
-
const currentIsNote = settings.isNote;
|
|
220
|
-
settings.isNote = (path) => currentIsNote(path) && (newSettings.isNote?.(path) ?? true);
|
|
221
|
-
}
|
|
222
|
-
settings.emptyAttachmentFolderBehavior ??= "Keep" /* Keep */;
|
|
223
|
-
return settings;
|
|
224
|
-
}
|
|
225
|
-
async function handleCaseCollision(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks) {
|
|
226
|
-
if (!app.vault.adapter.insensitive || oldPath.toLowerCase() !== newPath.toLowerCase()) {
|
|
227
|
-
return false;
|
|
228
|
-
}
|
|
229
|
-
const tempPath = (0, import_Path.join)((0, import_Path.dirname)(newPath), `__temp__${(0, import_Path.basename)(newPath)}`);
|
|
230
|
-
await renameHandled(app, newPath, tempPath);
|
|
231
|
-
await handleRenameAsync(app, oldPath, tempPath, oldPathBacklinksMap, oldPathLinks);
|
|
232
|
-
await app.vault.rename((0, import_FileSystem.getFile)(app, tempPath), newPath);
|
|
233
|
-
return true;
|
|
234
|
-
}
|
|
235
|
-
async function handleDelete(app, path, abortSignal) {
|
|
236
|
-
abortSignal.throwIfAborted();
|
|
237
|
-
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleDelete")(`Handle Delete ${path}`);
|
|
238
|
-
if (!isNoteEx(app, path)) {
|
|
239
|
-
return;
|
|
108
|
+
class HandledRenames {
|
|
109
|
+
map = /* @__PURE__ */ new Map();
|
|
110
|
+
add(oldPath, newPath) {
|
|
111
|
+
this.map.set(this.keyToString(oldPath, newPath), { newPath, oldPath });
|
|
240
112
|
}
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
return;
|
|
244
|
-
}
|
|
245
|
-
if (settings.isPathIgnored?.(path)) {
|
|
246
|
-
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleDelete")(`Skipping delete handler of ${path} as the path is ignored.`);
|
|
247
|
-
return;
|
|
113
|
+
delete(oldPath, newPath) {
|
|
114
|
+
this.map.delete(this.keyToString(oldPath, newPath));
|
|
248
115
|
}
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
const parentFolderPaths = /* @__PURE__ */ new Set();
|
|
252
|
-
if (cache) {
|
|
253
|
-
const links = (0, import_MetadataCache.getAllLinks)(cache);
|
|
254
|
-
for (const link of links) {
|
|
255
|
-
const attachmentFile = (0, import_Link.extractLinkFile)(app, link, path);
|
|
256
|
-
if (!attachmentFile) {
|
|
257
|
-
continue;
|
|
258
|
-
}
|
|
259
|
-
if (isNoteEx(app, attachmentFile.path)) {
|
|
260
|
-
continue;
|
|
261
|
-
}
|
|
262
|
-
parentFolderPaths.add(attachmentFile.parent?.path ?? "");
|
|
263
|
-
await (0, import_VaultEx.deleteSafe)(app, attachmentFile, path, false, settings.emptyAttachmentFolderBehavior !== "Keep" /* Keep */);
|
|
264
|
-
abortSignal.throwIfAborted();
|
|
265
|
-
}
|
|
116
|
+
has(oldPath, newPath) {
|
|
117
|
+
return this.map.has(this.keyToString(oldPath, newPath));
|
|
266
118
|
}
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
const attachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(app, path, import_AttachmentPath.AttachmentPathContext.DeleteNote);
|
|
270
|
-
const attachmentFolder = (0, import_FileSystem.getFolderOrNull)(app, attachmentFolderPath);
|
|
271
|
-
if (!attachmentFolder) {
|
|
272
|
-
return;
|
|
119
|
+
keys() {
|
|
120
|
+
return this.map.values();
|
|
273
121
|
}
|
|
274
|
-
|
|
275
|
-
return
|
|
122
|
+
keyToString(oldPath, newPath) {
|
|
123
|
+
return `${oldPath} -> ${newPath}`;
|
|
276
124
|
}
|
|
277
|
-
abortSignal.throwIfAborted();
|
|
278
|
-
await (0, import_VaultEx.deleteSafe)(app, attachmentFolder, path, false, settings.emptyAttachmentFolderBehavior !== "Keep" /* Keep */);
|
|
279
|
-
abortSignal.throwIfAborted();
|
|
280
125
|
}
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
126
|
+
class MetadataDeletedHandler {
|
|
127
|
+
constructor(app, file, prevCache, settingsManager, deletedMetadataCacheMap) {
|
|
128
|
+
this.app = app;
|
|
129
|
+
this.file = file;
|
|
130
|
+
this.prevCache = prevCache;
|
|
131
|
+
this.settingsManager = settingsManager;
|
|
132
|
+
this.deletedMetadataCacheMap = deletedMetadataCacheMap;
|
|
133
|
+
}
|
|
134
|
+
handle() {
|
|
135
|
+
const settings = this.settingsManager.getSettings();
|
|
136
|
+
if (!settings.shouldHandleDeletions) {
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
139
|
+
if (settings.isPathIgnored?.(this.file.path)) {
|
|
140
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleMetadataDeleted")(`Skipping metadata delete handler of ${this.file.path} as the path is ignored.`);
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
if ((0, import_FileSystem.isMarkdownFile)(this.app, this.file) && this.prevCache) {
|
|
144
|
+
this.deletedMetadataCacheMap.set(this.file.path, this.prevCache);
|
|
145
|
+
}
|
|
285
146
|
}
|
|
286
|
-
const path = file.path;
|
|
287
|
-
(0, import_Queue.addToQueue)(app, (abortSignal2) => handleDelete(app, path, abortSignal2), abortSignal);
|
|
288
147
|
}
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
148
|
+
class Registry {
|
|
149
|
+
constructor(plugin, settingsBuilder, settingsManager) {
|
|
150
|
+
this.plugin = plugin;
|
|
151
|
+
this.settingsBuilder = settingsBuilder;
|
|
152
|
+
this.settingsManager = settingsManager;
|
|
153
|
+
this.app = plugin.app;
|
|
154
|
+
this.pluginId = plugin.manifest.id;
|
|
155
|
+
this.abortSignal = plugin.abortSignal ?? (0, import_AbortController.abortSignalNever)();
|
|
156
|
+
}
|
|
157
|
+
abortSignal;
|
|
158
|
+
app;
|
|
159
|
+
deletedMetadataCacheMap = /* @__PURE__ */ new Map();
|
|
160
|
+
handledRenames = new HandledRenames();
|
|
161
|
+
interruptedRenamesMap = /* @__PURE__ */ new Map();
|
|
162
|
+
pluginId;
|
|
163
|
+
register() {
|
|
164
|
+
const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;
|
|
165
|
+
renameDeleteHandlersMap.set(this.pluginId, this.settingsBuilder);
|
|
166
|
+
this.logRegisteredHandlers();
|
|
167
|
+
this.plugin.register(() => {
|
|
168
|
+
renameDeleteHandlersMap.delete(this.pluginId);
|
|
169
|
+
this.logRegisteredHandlers();
|
|
170
|
+
});
|
|
171
|
+
this.plugin.registerEvent(this.app.vault.on("delete", this.handleDelete.bind(this)));
|
|
172
|
+
this.plugin.registerEvent(this.app.vault.on("rename", this.handleRename.bind(this)));
|
|
173
|
+
this.plugin.registerEvent(this.app.metadataCache.on("deleted", this.handleMetadataDeleted.bind(this)));
|
|
174
|
+
(0, import_MonkeyAround.registerPatch)(this.plugin, this.app.fileManager, {
|
|
175
|
+
runAsyncLinkUpdate: (next) => {
|
|
176
|
+
return Object.assign((linkUpdatesHandler) => this.runAsyncLinkUpdate(next, linkUpdatesHandler), { renameDeleteHandlerPatched: true });
|
|
177
|
+
}
|
|
178
|
+
});
|
|
294
179
|
}
|
|
295
|
-
|
|
296
|
-
|
|
180
|
+
handleDelete(file) {
|
|
181
|
+
if (!this.shouldInvokeHandler()) {
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
(0, import_Queue.addToQueue)(
|
|
185
|
+
this.app,
|
|
186
|
+
(abortSignal) => new DeleteHandler(this.app, file, abortSignal, this.settingsManager, this.deletedMetadataCacheMap).handle(),
|
|
187
|
+
this.abortSignal
|
|
188
|
+
);
|
|
189
|
+
}
|
|
190
|
+
handleMetadataDeleted(file, prevCache) {
|
|
191
|
+
if (!this.shouldInvokeHandler()) {
|
|
192
|
+
return;
|
|
193
|
+
}
|
|
194
|
+
(0, import_Queue.addToQueue)(this.app, () => {
|
|
195
|
+
new MetadataDeletedHandler(this.app, file, prevCache, this.settingsManager, this.deletedMetadataCacheMap).handle();
|
|
196
|
+
}, this.abortSignal);
|
|
297
197
|
}
|
|
298
|
-
|
|
299
|
-
|
|
198
|
+
handleRename(file, oldPath) {
|
|
199
|
+
if (!this.shouldInvokeHandler()) {
|
|
200
|
+
return;
|
|
201
|
+
}
|
|
202
|
+
if (!(0, import_FileSystem.isFile)(file)) {
|
|
203
|
+
return;
|
|
204
|
+
}
|
|
205
|
+
const newPath = file.path;
|
|
206
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleRename")(`Handle Rename ${oldPath} -> ${newPath}`);
|
|
207
|
+
if (this.handledRenames.has(oldPath, newPath)) {
|
|
208
|
+
this.handledRenames.delete(oldPath, newPath);
|
|
209
|
+
return;
|
|
210
|
+
}
|
|
211
|
+
const settings = this.settingsManager.getSettings();
|
|
212
|
+
if (!settings.shouldHandleRenames) {
|
|
213
|
+
return;
|
|
214
|
+
}
|
|
215
|
+
if (settings.isPathIgnored?.(oldPath)) {
|
|
216
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleRename")(`Skipping rename handler of old path ${oldPath} as the path is ignored.`);
|
|
217
|
+
return;
|
|
218
|
+
}
|
|
219
|
+
if (settings.isPathIgnored?.(newPath)) {
|
|
220
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:handleRename")(`Skipping rename handler of new path ${newPath} as the path is ignored.`);
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
const oldCache = this.app.metadataCache.getCache(oldPath) ?? this.app.metadataCache.getCache(newPath);
|
|
224
|
+
const oldPathBacklinksMap = (0, import_MetadataCache.getBacklinksForFileOrPath)(this.app, oldPath).data;
|
|
225
|
+
(0, import_Queue.addToQueue)(this.app, (abortSignal) => new RenameHandler({
|
|
226
|
+
abortSignal,
|
|
227
|
+
app: this.app,
|
|
228
|
+
handledRenames: this.handledRenames,
|
|
229
|
+
interruptedRenamesMap: this.interruptedRenamesMap,
|
|
230
|
+
newPath,
|
|
231
|
+
oldCache,
|
|
232
|
+
oldPath,
|
|
233
|
+
oldPathBacklinksMap,
|
|
234
|
+
settingsManager: this.settingsManager
|
|
235
|
+
}).handle(), this.abortSignal);
|
|
236
|
+
}
|
|
237
|
+
logRegisteredHandlers() {
|
|
238
|
+
const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;
|
|
239
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:logRegisteredHandlers")(
|
|
240
|
+
`Plugins with registered rename/delete handlers: ${JSON.stringify(Array.from(renameDeleteHandlersMap.keys()))}`
|
|
241
|
+
);
|
|
242
|
+
}
|
|
243
|
+
async runAsyncLinkUpdate(next, linkUpdatesHandler) {
|
|
244
|
+
if (next.renameDeleteHandlerPatched) {
|
|
245
|
+
await next.call(this.app.fileManager, linkUpdatesHandler);
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
await next.call(this.app.fileManager, (linkUpdates) => this.wrapLinkUpdatesHandler(linkUpdates, linkUpdatesHandler));
|
|
300
249
|
}
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
250
|
+
shouldInvokeHandler() {
|
|
251
|
+
const pluginId = this.plugin.manifest.id;
|
|
252
|
+
const renameDeleteHandlersMap = this.settingsManager.renameDeleteHandlersMap;
|
|
253
|
+
const mainPluginId = Array.from(renameDeleteHandlersMap.keys())[0];
|
|
254
|
+
return mainPluginId === pluginId;
|
|
255
|
+
}
|
|
256
|
+
async wrapLinkUpdatesHandler(linkUpdates, linkUpdatesHandler) {
|
|
257
|
+
let isRenameCalled = false;
|
|
258
|
+
const eventRef = this.app.vault.on("rename", () => {
|
|
259
|
+
isRenameCalled = true;
|
|
260
|
+
});
|
|
261
|
+
try {
|
|
262
|
+
await linkUpdatesHandler(linkUpdates);
|
|
263
|
+
} finally {
|
|
264
|
+
this.app.vault.offref(eventRef);
|
|
265
|
+
}
|
|
266
|
+
const settings = this.settingsManager.getSettings();
|
|
267
|
+
if (!isRenameCalled || !settings.shouldHandleRenames) {
|
|
268
|
+
return;
|
|
269
|
+
}
|
|
270
|
+
(0, import_Array.filterInPlace)(
|
|
271
|
+
linkUpdates,
|
|
272
|
+
(linkUpdate) => {
|
|
273
|
+
if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {
|
|
274
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:runAsyncLinkUpdate")(
|
|
275
|
+
`Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`
|
|
276
|
+
);
|
|
277
|
+
return true;
|
|
278
|
+
}
|
|
279
|
+
if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {
|
|
280
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:runAsyncLinkUpdate")(
|
|
281
|
+
`Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`
|
|
282
|
+
);
|
|
283
|
+
return true;
|
|
284
|
+
}
|
|
285
|
+
if (!this.app.internalPlugins.getEnabledPluginById(import_implementations.InternalPluginName.Canvas)) {
|
|
286
|
+
return false;
|
|
287
|
+
}
|
|
288
|
+
if (this.app.plugins.getPlugin("backlink-cache")) {
|
|
289
|
+
return false;
|
|
290
|
+
}
|
|
291
|
+
if (linkUpdate.sourceFile.extension === import_FileSystem.CANVAS_FILE_EXTENSION) {
|
|
292
|
+
return true;
|
|
293
|
+
}
|
|
294
|
+
if (linkUpdate.resolvedFile.extension === import_FileSystem.CANVAS_FILE_EXTENSION) {
|
|
295
|
+
return true;
|
|
296
|
+
}
|
|
297
|
+
return false;
|
|
298
|
+
}
|
|
299
|
+
);
|
|
305
300
|
}
|
|
306
|
-
handleMetadataDeleted(plugin.app, file, prevCache);
|
|
307
301
|
}
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
302
|
+
class RenameHandler {
|
|
303
|
+
abortSignal;
|
|
304
|
+
app;
|
|
305
|
+
handledRenames;
|
|
306
|
+
interruptedCombinedBacklinksMap;
|
|
307
|
+
interruptedRenamesMap;
|
|
308
|
+
newPath;
|
|
309
|
+
oldCache;
|
|
310
|
+
oldPath;
|
|
311
|
+
oldPathBacklinksMap;
|
|
312
|
+
oldPathLinks;
|
|
313
|
+
settingsManager;
|
|
314
|
+
constructor(options) {
|
|
315
|
+
this.app = options.app;
|
|
316
|
+
this.oldPath = options.oldPath;
|
|
317
|
+
this.newPath = options.newPath;
|
|
318
|
+
this.oldPathBacklinksMap = options.oldPathBacklinksMap;
|
|
319
|
+
this.oldCache = options.oldCache;
|
|
320
|
+
this.abortSignal = options.abortSignal;
|
|
321
|
+
this.settingsManager = options.settingsManager;
|
|
322
|
+
this.interruptedRenamesMap = options.interruptedRenamesMap;
|
|
323
|
+
this.oldPathLinks = this.oldCache ? (0, import_MetadataCache.getAllLinks)(this.oldCache) : [];
|
|
324
|
+
this.handledRenames = options.handledRenames;
|
|
325
|
+
this.interruptedCombinedBacklinksMap = options.interruptedCombinedBacklinksMap ?? /* @__PURE__ */ new Map();
|
|
326
|
+
}
|
|
327
|
+
async handle() {
|
|
328
|
+
this.abortSignal.throwIfAborted();
|
|
329
|
+
await this.continueInterruptedRenames();
|
|
330
|
+
this.abortSignal.throwIfAborted();
|
|
331
|
+
await this.refreshLinks();
|
|
332
|
+
this.abortSignal.throwIfAborted();
|
|
333
|
+
if (await this.handleCaseCollision()) {
|
|
334
|
+
return;
|
|
335
|
+
}
|
|
336
|
+
this.abortSignal.throwIfAborted();
|
|
337
|
+
try {
|
|
338
|
+
const renameMap = new RenameMap({
|
|
339
|
+
abortSignal: this.abortSignal,
|
|
340
|
+
app: this.app,
|
|
341
|
+
newPath: this.newPath,
|
|
342
|
+
oldCache: this.oldCache,
|
|
343
|
+
oldPath: this.oldPath,
|
|
344
|
+
settingsManager: this.settingsManager
|
|
345
|
+
});
|
|
346
|
+
await renameMap.fill();
|
|
347
|
+
this.abortSignal.throwIfAborted();
|
|
348
|
+
const combinedBacklinksMap = /* @__PURE__ */ new Map();
|
|
349
|
+
renameMap.initBacklinksMap(this.oldPathBacklinksMap, combinedBacklinksMap, this.oldPath);
|
|
350
|
+
for (const attachmentOldPath of renameMap.keys()) {
|
|
351
|
+
if (attachmentOldPath === this.oldPath) {
|
|
352
|
+
continue;
|
|
353
|
+
}
|
|
354
|
+
const attachmentOldPathBacklinksMap = (await (0, import_MetadataCache.getBacklinksForFileSafe)(this.app, attachmentOldPath)).data;
|
|
355
|
+
this.abortSignal.throwIfAborted();
|
|
356
|
+
renameMap.initBacklinksMap(attachmentOldPathBacklinksMap, combinedBacklinksMap, attachmentOldPath);
|
|
357
|
+
}
|
|
358
|
+
const parentFolderPaths = /* @__PURE__ */ new Set();
|
|
359
|
+
for (const [oldAttachmentPath, newAttachmentPath] of renameMap.entries()) {
|
|
360
|
+
if (oldAttachmentPath === this.oldPath) {
|
|
361
|
+
continue;
|
|
362
|
+
}
|
|
363
|
+
const fixedNewAttachmentPath = await this.renameHandled(oldAttachmentPath, newAttachmentPath);
|
|
364
|
+
this.abortSignal.throwIfAborted();
|
|
365
|
+
renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);
|
|
366
|
+
parentFolderPaths.add((0, import_Path.dirname)(oldAttachmentPath));
|
|
367
|
+
}
|
|
368
|
+
await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths), this.oldPath);
|
|
369
|
+
this.abortSignal.throwIfAborted();
|
|
370
|
+
const settings = this.settingsManager.getSettings();
|
|
371
|
+
for (const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(
|
|
372
|
+
Array.from(this.interruptedCombinedBacklinksMap.entries())
|
|
373
|
+
)) {
|
|
374
|
+
await (0, import_Link.editLinks)(this.app, newBacklinkPath, (link) => {
|
|
375
|
+
const oldAttachmentPath = linkJsonToPathMap.get((0, import_ObjectUtils.toJson)(link));
|
|
376
|
+
if (!oldAttachmentPath) {
|
|
377
|
+
return;
|
|
378
|
+
}
|
|
379
|
+
const newAttachmentPath = renameMap.get(oldAttachmentPath);
|
|
380
|
+
if (!newAttachmentPath) {
|
|
381
|
+
return;
|
|
382
|
+
}
|
|
383
|
+
return (0, import_Link.updateLink)((0, import_ObjectUtils.normalizeOptionalProperties)({
|
|
384
|
+
app: this.app,
|
|
385
|
+
link,
|
|
386
|
+
newSourcePathOrFile: newBacklinkPath,
|
|
387
|
+
newTargetPathOrFile: newAttachmentPath,
|
|
388
|
+
oldTargetPathOrFile: oldAttachmentPath,
|
|
389
|
+
shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases
|
|
390
|
+
}));
|
|
391
|
+
}, {
|
|
392
|
+
shouldFailOnMissingFile: false
|
|
393
|
+
});
|
|
394
|
+
this.abortSignal.throwIfAborted();
|
|
395
|
+
}
|
|
396
|
+
if (this.settingsManager.isNoteEx(this.newPath)) {
|
|
397
|
+
await (0, import_Link.updateLinksInFile)((0, import_ObjectUtils.normalizeOptionalProperties)({
|
|
398
|
+
app: this.app,
|
|
399
|
+
newSourcePathOrFile: this.newPath,
|
|
400
|
+
oldSourcePathOrFile: this.oldPath,
|
|
401
|
+
shouldFailOnMissingFile: false,
|
|
402
|
+
shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases
|
|
403
|
+
}));
|
|
404
|
+
this.abortSignal.throwIfAborted();
|
|
405
|
+
}
|
|
406
|
+
if (!(0, import_FileSystem.getFileOrNull)(this.app, this.newPath)) {
|
|
407
|
+
let interruptedRenames = this.interruptedRenamesMap.get(this.newPath);
|
|
408
|
+
if (!interruptedRenames) {
|
|
409
|
+
interruptedRenames = [];
|
|
410
|
+
this.interruptedRenamesMap.set(this.newPath, interruptedRenames);
|
|
411
|
+
}
|
|
412
|
+
interruptedRenames.push({
|
|
413
|
+
combinedBacklinksMap,
|
|
414
|
+
oldPath: this.oldPath
|
|
415
|
+
});
|
|
416
|
+
}
|
|
417
|
+
} finally {
|
|
418
|
+
const orphanKeys = Array.from(this.handledRenames.keys());
|
|
419
|
+
(0, import_Queue.addToQueue)(this.app, () => {
|
|
420
|
+
for (const orphanKey of orphanKeys) {
|
|
421
|
+
this.handledRenames.delete(orphanKey.oldPath, orphanKey.newPath);
|
|
422
|
+
}
|
|
423
|
+
}, this.abortSignal);
|
|
424
|
+
}
|
|
314
425
|
}
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
426
|
+
async continueInterruptedRenames() {
|
|
427
|
+
const interruptedRenames = this.interruptedRenamesMap.get(this.oldPath);
|
|
428
|
+
if (interruptedRenames) {
|
|
429
|
+
this.interruptedRenamesMap.delete(this.oldPath);
|
|
430
|
+
for (const interruptedRename of interruptedRenames) {
|
|
431
|
+
await new RenameHandler({
|
|
432
|
+
abortSignal: this.abortSignal,
|
|
433
|
+
app: this.app,
|
|
434
|
+
handledRenames: this.handledRenames,
|
|
435
|
+
interruptedCombinedBacklinksMap: interruptedRename.combinedBacklinksMap,
|
|
436
|
+
interruptedRenamesMap: this.interruptedRenamesMap,
|
|
437
|
+
newPath: this.newPath,
|
|
438
|
+
oldCache: this.oldCache,
|
|
439
|
+
oldPath: interruptedRename.oldPath,
|
|
440
|
+
oldPathBacklinksMap: this.oldPathBacklinksMap,
|
|
441
|
+
settingsManager: this.settingsManager
|
|
442
|
+
}).handle();
|
|
443
|
+
}
|
|
444
|
+
}
|
|
318
445
|
}
|
|
319
|
-
|
|
320
|
-
(
|
|
321
|
-
|
|
446
|
+
async handleCaseCollision() {
|
|
447
|
+
if (!this.app.vault.adapter.insensitive || this.oldPath.toLowerCase() !== this.newPath.toLowerCase()) {
|
|
448
|
+
return false;
|
|
449
|
+
}
|
|
450
|
+
const tempPath = (0, import_Path.join)((0, import_Path.dirname)(this.newPath), `__temp__${(0, import_Path.basename)(this.newPath)}`);
|
|
451
|
+
await this.renameHandled(this.newPath, tempPath);
|
|
452
|
+
await new RenameHandler({
|
|
453
|
+
abortSignal: this.abortSignal,
|
|
454
|
+
app: this.app,
|
|
455
|
+
handledRenames: this.handledRenames,
|
|
456
|
+
interruptedRenamesMap: this.interruptedRenamesMap,
|
|
457
|
+
newPath: tempPath,
|
|
458
|
+
oldCache: this.oldCache,
|
|
459
|
+
oldPath: this.oldPath,
|
|
460
|
+
oldPathBacklinksMap: this.oldPathBacklinksMap,
|
|
461
|
+
settingsManager: this.settingsManager
|
|
462
|
+
}).handle();
|
|
463
|
+
await this.app.vault.rename((0, import_FileSystem.getFile)(this.app, tempPath), this.newPath);
|
|
464
|
+
return true;
|
|
465
|
+
}
|
|
466
|
+
async refreshLinks() {
|
|
467
|
+
const cache = this.app.metadataCache.getCache(this.oldPath) ?? this.app.metadataCache.getCache(this.newPath);
|
|
468
|
+
const oldPathLinksRefreshed = cache ? (0, import_MetadataCache.getAllLinks)(cache) : [];
|
|
469
|
+
const fakeOldFile = (0, import_FileSystem.getFile)(this.app, this.oldPath, true);
|
|
470
|
+
let oldPathBacklinksMapRefreshed = /* @__PURE__ */ new Map();
|
|
471
|
+
await (0, import_MetadataCache.tempRegisterFilesAndRun)(this.app, [fakeOldFile], async () => {
|
|
472
|
+
oldPathBacklinksMapRefreshed = (await (0, import_MetadataCache.getBacklinksForFileSafe)(this.app, fakeOldFile)).data;
|
|
473
|
+
});
|
|
474
|
+
for (const link of oldPathLinksRefreshed) {
|
|
475
|
+
if (this.oldPathLinks.includes(link)) {
|
|
476
|
+
continue;
|
|
477
|
+
}
|
|
478
|
+
this.oldPathLinks.push(link);
|
|
479
|
+
}
|
|
480
|
+
for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {
|
|
481
|
+
let oldLinks = this.oldPathBacklinksMap.get(backlinkPath);
|
|
482
|
+
if (!oldLinks) {
|
|
483
|
+
oldLinks = [];
|
|
484
|
+
this.oldPathBacklinksMap.set(backlinkPath, oldLinks);
|
|
485
|
+
}
|
|
486
|
+
for (const link of refreshedLinks) {
|
|
487
|
+
if (oldLinks.includes(link)) {
|
|
488
|
+
continue;
|
|
489
|
+
}
|
|
490
|
+
oldLinks.push(link);
|
|
491
|
+
}
|
|
492
|
+
}
|
|
322
493
|
}
|
|
323
|
-
|
|
324
|
-
(0,
|
|
325
|
-
|
|
494
|
+
async renameHandled(oldPath, newPath) {
|
|
495
|
+
newPath = (0, import_Vault.getSafeRenamePath)(this.app, oldPath, newPath);
|
|
496
|
+
if (oldPath === newPath) {
|
|
497
|
+
return newPath;
|
|
498
|
+
}
|
|
499
|
+
this.handledRenames.add(oldPath, newPath);
|
|
500
|
+
newPath = await (0, import_Vault.renameSafe)(this.app, oldPath, newPath);
|
|
501
|
+
return newPath;
|
|
326
502
|
}
|
|
327
|
-
const cache = app.metadataCache.getCache(oldPath) ?? app.metadataCache.getCache(newPath);
|
|
328
|
-
const oldPathLinks = cache ? (0, import_MetadataCache.getAllLinks)(cache) : [];
|
|
329
|
-
const oldPathBacklinksMap = (0, import_MetadataCache.getBacklinksForFileOrPath)(app, oldPath).data;
|
|
330
|
-
(0, import_Queue.addToQueue)(app, (abortSignal2) => handleRenameAsync(app, oldPath, newPath, oldPathBacklinksMap, oldPathLinks, void 0, abortSignal2), abortSignal);
|
|
331
503
|
}
|
|
332
|
-
|
|
333
|
-
abortSignal
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
504
|
+
class RenameMap {
|
|
505
|
+
abortSignal;
|
|
506
|
+
app;
|
|
507
|
+
map = /* @__PURE__ */ new Map();
|
|
508
|
+
newPath;
|
|
509
|
+
oldCache;
|
|
510
|
+
oldPath;
|
|
511
|
+
oldPathLinks;
|
|
512
|
+
settingsManager;
|
|
513
|
+
constructor(options) {
|
|
514
|
+
this.abortSignal = options.abortSignal;
|
|
515
|
+
this.app = options.app;
|
|
516
|
+
this.settingsManager = options.settingsManager;
|
|
517
|
+
this.oldCache = options.oldCache;
|
|
518
|
+
this.oldPath = options.oldPath;
|
|
519
|
+
this.newPath = options.newPath;
|
|
520
|
+
this.oldPathLinks = this.oldCache ? (0, import_MetadataCache.getAllLinks)(this.oldCache) : [];
|
|
521
|
+
}
|
|
522
|
+
entries() {
|
|
523
|
+
return this.map.entries();
|
|
524
|
+
}
|
|
525
|
+
async fill() {
|
|
526
|
+
this.abortSignal.throwIfAborted();
|
|
527
|
+
this.map.set(this.oldPath, this.newPath);
|
|
528
|
+
if (!this.settingsManager.isNoteEx(this.oldPath)) {
|
|
529
|
+
return;
|
|
355
530
|
}
|
|
356
|
-
const
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
531
|
+
const settings = this.settingsManager.getSettings();
|
|
532
|
+
const oldFile = (0, import_FileSystem.getFile)(this.app, this.oldPath, true);
|
|
533
|
+
let oldAttachmentFolderPath = "";
|
|
534
|
+
await (0, import_MetadataCache.tempRegisterFilesAndRunAsync)(this.app, [oldFile], async () => {
|
|
535
|
+
const shouldFakeOldPathCache = this.oldCache && oldFile.deleted;
|
|
536
|
+
if (shouldFakeOldPathCache) {
|
|
537
|
+
(0, import_MetadataCache.registerFileCacheForNonExistingFile)(this.app, oldFile, this.oldCache);
|
|
360
538
|
}
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
abortSignal.throwIfAborted();
|
|
367
|
-
const settings = getSettings(app);
|
|
368
|
-
for (const [newBacklinkPath, linkJsonToPathMap] of Array.from(combinedBacklinksMap.entries()).concat(
|
|
369
|
-
Array.from(interruptedCombinedBacklinksMap?.entries() ?? [])
|
|
370
|
-
)) {
|
|
371
|
-
await (0, import_Link.editLinks)(app, newBacklinkPath, (link) => {
|
|
372
|
-
const oldAttachmentPath = linkJsonToPathMap.get((0, import_ObjectUtils.toJson)(link));
|
|
373
|
-
if (!oldAttachmentPath) {
|
|
374
|
-
return;
|
|
539
|
+
try {
|
|
540
|
+
oldAttachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(this.app, this.oldPath, import_AttachmentPath.AttachmentPathContext.RenameNote);
|
|
541
|
+
} finally {
|
|
542
|
+
if (shouldFakeOldPathCache) {
|
|
543
|
+
(0, import_MetadataCache.unregisterFileCacheForNonExistingFile)(this.app, oldFile);
|
|
375
544
|
}
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
545
|
+
}
|
|
546
|
+
});
|
|
547
|
+
const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder ? await (0, import_AttachmentPath.getAttachmentFolderPath)(this.app, this.newPath, import_AttachmentPath.AttachmentPathContext.RenameNote) : oldAttachmentFolderPath;
|
|
548
|
+
const isOldAttachmentFolderAtRoot = oldAttachmentFolderPath === "/";
|
|
549
|
+
const oldAttachmentFolder = (0, import_FileSystem.getFolderOrNull)(this.app, oldAttachmentFolderPath);
|
|
550
|
+
if (!oldAttachmentFolder) {
|
|
551
|
+
return;
|
|
552
|
+
}
|
|
553
|
+
if (oldAttachmentFolderPath === newAttachmentFolderPath && !settings.shouldRenameAttachmentFiles) {
|
|
554
|
+
return;
|
|
555
|
+
}
|
|
556
|
+
const oldAttachmentFiles = [];
|
|
557
|
+
if (await (0, import_AttachmentPath.hasOwnAttachmentFolder)(this.app, this.oldPath, import_AttachmentPath.AttachmentPathContext.RenameNote)) {
|
|
558
|
+
import_obsidian.Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {
|
|
559
|
+
this.abortSignal.throwIfAborted();
|
|
560
|
+
if ((0, import_FileSystem.isFile)(oldAttachmentFile)) {
|
|
561
|
+
oldAttachmentFiles.push(oldAttachmentFile);
|
|
379
562
|
}
|
|
380
|
-
return (0, import_Link.updateLink)((0, import_ObjectUtils.normalizeOptionalProperties)({
|
|
381
|
-
app,
|
|
382
|
-
link,
|
|
383
|
-
newSourcePathOrFile: newBacklinkPath,
|
|
384
|
-
newTargetPathOrFile: newAttachmentPath,
|
|
385
|
-
oldTargetPathOrFile: oldAttachmentPath,
|
|
386
|
-
shouldUpdateFileNameAlias: settings.shouldUpdateFileNameAliases
|
|
387
|
-
}));
|
|
388
|
-
}, {
|
|
389
|
-
shouldFailOnMissingFile: false
|
|
390
563
|
});
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
if (!interruptedRenames) {
|
|
406
|
-
interruptedRenames = [];
|
|
407
|
-
interruptedRenamesMap.set(newPath, interruptedRenames);
|
|
564
|
+
} else {
|
|
565
|
+
for (const oldPathLink of this.oldPathLinks) {
|
|
566
|
+
this.abortSignal.throwIfAborted();
|
|
567
|
+
const oldAttachmentFile = (0, import_Link.extractLinkFile)(this.app, oldPathLink, this.oldPath);
|
|
568
|
+
if (!oldAttachmentFile) {
|
|
569
|
+
continue;
|
|
570
|
+
}
|
|
571
|
+
if (isOldAttachmentFolderAtRoot || oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {
|
|
572
|
+
const oldAttachmentBacklinks = await (0, import_MetadataCache.getBacklinksForFileSafe)(this.app, oldAttachmentFile);
|
|
573
|
+
this.abortSignal.throwIfAborted();
|
|
574
|
+
if (oldAttachmentBacklinks.keys().length === 1) {
|
|
575
|
+
oldAttachmentFiles.push(oldAttachmentFile);
|
|
576
|
+
}
|
|
577
|
+
}
|
|
408
578
|
}
|
|
409
|
-
interruptedRenames.push({
|
|
410
|
-
combinedBacklinksMap,
|
|
411
|
-
oldPath
|
|
412
|
-
});
|
|
413
579
|
}
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
handledRenames.delete(key);
|
|
580
|
+
for (const oldAttachmentFile of oldAttachmentFiles) {
|
|
581
|
+
this.abortSignal.throwIfAborted();
|
|
582
|
+
if (this.settingsManager.isNoteEx(oldAttachmentFile.path)) {
|
|
583
|
+
continue;
|
|
419
584
|
}
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
585
|
+
let newAttachmentFilePath;
|
|
586
|
+
if (settings.shouldRenameAttachmentFiles) {
|
|
587
|
+
newAttachmentFilePath = await (0, import_AttachmentPath.getAttachmentFilePath)({
|
|
588
|
+
app: this.app,
|
|
589
|
+
attachmentPathOrFile: oldAttachmentFile,
|
|
590
|
+
context: import_AttachmentPath.AttachmentPathContext.RenameNote,
|
|
591
|
+
notePathOrFile: this.newPath,
|
|
592
|
+
oldNotePathOrFile: this.oldPath,
|
|
593
|
+
shouldSkipDuplicateCheck: true
|
|
594
|
+
});
|
|
595
|
+
this.abortSignal.throwIfAborted();
|
|
596
|
+
} else {
|
|
597
|
+
const relativePath = isOldAttachmentFolderAtRoot ? oldAttachmentFile.path : (0, import_Path.relative)(oldAttachmentFolderPath, oldAttachmentFile.path);
|
|
598
|
+
const newFolder = (0, import_Path.join)(newAttachmentFolderPath, (0, import_Path.dirname)(relativePath));
|
|
599
|
+
newAttachmentFilePath = (0, import_Path.join)(newFolder, oldAttachmentFile.name);
|
|
600
|
+
}
|
|
601
|
+
if (oldAttachmentFile.path === newAttachmentFilePath) {
|
|
602
|
+
continue;
|
|
603
|
+
}
|
|
604
|
+
if (settings.shouldDeleteConflictingAttachments) {
|
|
605
|
+
const newAttachmentFile = (0, import_FileSystem.getFileOrNull)(this.app, newAttachmentFilePath);
|
|
606
|
+
if (newAttachmentFile) {
|
|
607
|
+
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:fillRenameMap")(`Removing conflicting attachment ${newAttachmentFile.path}.`);
|
|
608
|
+
await this.app.fileManager.trashFile(newAttachmentFile);
|
|
609
|
+
this.abortSignal.throwIfAborted();
|
|
610
|
+
}
|
|
611
|
+
} else {
|
|
612
|
+
const dir = (0, import_Path.dirname)(newAttachmentFilePath);
|
|
613
|
+
const ext = (0, import_Path.extname)(newAttachmentFilePath);
|
|
614
|
+
const baseName = (0, import_Path.basename)(newAttachmentFilePath, ext);
|
|
615
|
+
newAttachmentFilePath = this.app.vault.getAvailablePath((0, import_Path.join)(dir, baseName), ext.slice(1));
|
|
616
|
+
}
|
|
617
|
+
this.map.set(oldAttachmentFile.path, newAttachmentFilePath);
|
|
618
|
+
}
|
|
426
619
|
}
|
|
427
|
-
|
|
428
|
-
return;
|
|
620
|
+
get(oldPath) {
|
|
621
|
+
return this.map.get(oldPath);
|
|
429
622
|
}
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
for (const link of links) {
|
|
439
|
-
linkJsonToPathMap.set((0, import_ObjectUtils.toJson)(link), path);
|
|
623
|
+
initBacklinksMap(singleBacklinksMap, combinedBacklinksMap, path) {
|
|
624
|
+
for (const [backlinkPath, links] of singleBacklinksMap.entries()) {
|
|
625
|
+
const newBacklinkPath = this.map.get(backlinkPath) ?? backlinkPath;
|
|
626
|
+
const linkJsonToPathMap = combinedBacklinksMap.get(newBacklinkPath) ?? /* @__PURE__ */ new Map();
|
|
627
|
+
combinedBacklinksMap.set(newBacklinkPath, linkJsonToPathMap);
|
|
628
|
+
for (const link of links) {
|
|
629
|
+
linkJsonToPathMap.set((0, import_ObjectUtils.toJson)(link), path);
|
|
630
|
+
}
|
|
440
631
|
}
|
|
441
632
|
}
|
|
633
|
+
keys() {
|
|
634
|
+
return this.map.keys();
|
|
635
|
+
}
|
|
636
|
+
set(oldPath, newPath) {
|
|
637
|
+
this.map.set(oldPath, newPath);
|
|
638
|
+
}
|
|
442
639
|
}
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
(
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
const fakeOldFile = (0, import_FileSystem.getFile)(app, oldPath, true);
|
|
460
|
-
let oldPathBacklinksMapRefreshed = /* @__PURE__ */ new Map();
|
|
461
|
-
await (0, import_MetadataCache.tempRegisterFilesAndRun)(app, [fakeOldFile], async () => {
|
|
462
|
-
oldPathBacklinksMapRefreshed = (await (0, import_MetadataCache.getBacklinksForFileSafe)(app, fakeOldFile)).data;
|
|
463
|
-
});
|
|
464
|
-
for (const link of oldPathLinksRefreshed) {
|
|
465
|
-
if (oldPathLinks.includes(link)) {
|
|
466
|
-
continue;
|
|
467
|
-
}
|
|
468
|
-
oldPathLinks.push(link);
|
|
469
|
-
}
|
|
470
|
-
for (const [backlinkPath, refreshedLinks] of oldPathBacklinksMapRefreshed.entries()) {
|
|
471
|
-
let oldLinks = oldPathBacklinksMap.get(backlinkPath);
|
|
472
|
-
if (!oldLinks) {
|
|
473
|
-
oldLinks = [];
|
|
474
|
-
oldPathBacklinksMap.set(backlinkPath, oldLinks);
|
|
475
|
-
}
|
|
476
|
-
for (const link of refreshedLinks) {
|
|
477
|
-
if (oldLinks.includes(link)) {
|
|
478
|
-
continue;
|
|
640
|
+
class SettingsManager {
|
|
641
|
+
constructor(app) {
|
|
642
|
+
this.app = app;
|
|
643
|
+
this.renameDeleteHandlersMap = (0, import_App.getObsidianDevUtilsState)(app, "renameDeleteHandlersMap", /* @__PURE__ */ new Map()).value;
|
|
644
|
+
}
|
|
645
|
+
renameDeleteHandlersMap;
|
|
646
|
+
getSettings() {
|
|
647
|
+
const settingsBuilders = Array.from(this.renameDeleteHandlersMap.values()).reverse();
|
|
648
|
+
const settings = {};
|
|
649
|
+
settings.isNote = (path) => (0, import_FileSystem.isNote)(this.app, path);
|
|
650
|
+
settings.isPathIgnored = () => false;
|
|
651
|
+
for (const settingsBuilder of settingsBuilders) {
|
|
652
|
+
const newSettings = settingsBuilder();
|
|
653
|
+
settings.shouldDeleteConflictingAttachments ||= newSettings.shouldDeleteConflictingAttachments ?? false;
|
|
654
|
+
if (newSettings.emptyAttachmentFolderBehavior) {
|
|
655
|
+
settings.emptyAttachmentFolderBehavior ??= newSettings.emptyAttachmentFolderBehavior;
|
|
479
656
|
}
|
|
480
|
-
|
|
657
|
+
settings.shouldHandleDeletions ||= newSettings.shouldHandleDeletions ?? false;
|
|
658
|
+
settings.shouldHandleRenames ||= newSettings.shouldHandleRenames ?? false;
|
|
659
|
+
settings.shouldRenameAttachmentFiles ||= newSettings.shouldRenameAttachmentFiles ?? false;
|
|
660
|
+
settings.shouldRenameAttachmentFolder ||= newSettings.shouldRenameAttachmentFolder ?? false;
|
|
661
|
+
settings.shouldUpdateFileNameAliases ||= newSettings.shouldUpdateFileNameAliases ?? false;
|
|
662
|
+
const isPathIgnored = settings.isPathIgnored;
|
|
663
|
+
settings.isPathIgnored = (path) => isPathIgnored(path) || (newSettings.isPathIgnored?.(path) ?? false);
|
|
664
|
+
const currentIsNote = settings.isNote;
|
|
665
|
+
settings.isNote = (path) => currentIsNote(path) && (newSettings.isNote?.(path) ?? true);
|
|
481
666
|
}
|
|
667
|
+
settings.emptyAttachmentFolderBehavior ??= "Keep" /* Keep */;
|
|
668
|
+
return settings;
|
|
482
669
|
}
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
if (oldPath === newPath) {
|
|
487
|
-
return newPath;
|
|
670
|
+
isNoteEx(path) {
|
|
671
|
+
const settings = this.getSettings();
|
|
672
|
+
return settings.isNote?.(path) ?? false;
|
|
488
673
|
}
|
|
489
|
-
const key = makeKey(oldPath, newPath);
|
|
490
|
-
handledRenames.add(key);
|
|
491
|
-
newPath = await (0, import_Vault.renameSafe)(app, oldPath, newPath);
|
|
492
|
-
return newPath;
|
|
493
674
|
}
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
if (settings.isPathIgnored?.(linkUpdate.sourceFile.path)) {
|
|
512
|
-
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:runAsyncLinkUpdate")(
|
|
513
|
-
`Roll back to default link update of source file ${linkUpdate.sourceFile.path} as the path is ignored.`
|
|
514
|
-
);
|
|
515
|
-
return true;
|
|
516
|
-
}
|
|
517
|
-
if (settings.isPathIgnored?.(linkUpdate.resolvedFile.path)) {
|
|
518
|
-
(0, import_Debug.getLibDebugger)("RenameDeleteHandler:runAsyncLinkUpdate")(
|
|
519
|
-
`Roll back to default link update of resolved file ${linkUpdate.resolvedFile.path} as the path is ignored.`
|
|
520
|
-
);
|
|
521
|
-
return true;
|
|
522
|
-
}
|
|
523
|
-
if (!app.internalPlugins.getEnabledPluginById(import_implementations.InternalPluginName.Canvas)) {
|
|
524
|
-
return false;
|
|
525
|
-
}
|
|
526
|
-
if (app.plugins.getPlugin("backlink-cache")) {
|
|
527
|
-
return false;
|
|
528
|
-
}
|
|
529
|
-
if (linkUpdate.sourceFile.extension === "canvas") {
|
|
530
|
-
return true;
|
|
531
|
-
}
|
|
532
|
-
if (linkUpdate.resolvedFile.extension === "canvas") {
|
|
533
|
-
return true;
|
|
534
|
-
}
|
|
535
|
-
return false;
|
|
536
|
-
}
|
|
537
|
-
);
|
|
675
|
+
function registerRenameDeleteHandlers(plugin, settingsBuilder) {
|
|
676
|
+
new Registry(plugin, settingsBuilder, new SettingsManager(plugin.app)).register();
|
|
677
|
+
}
|
|
678
|
+
async function cleanupParentFolders(app, settings, parentFolderPaths, notePath) {
|
|
679
|
+
if (settings.emptyAttachmentFolderBehavior === "Keep" /* Keep */) {
|
|
680
|
+
return;
|
|
681
|
+
}
|
|
682
|
+
for (const parentFolderPath of parentFolderPaths) {
|
|
683
|
+
switch (settings.emptyAttachmentFolderBehavior) {
|
|
684
|
+
case "Delete" /* Delete */:
|
|
685
|
+
await (0, import_VaultEx.deleteSafe)(app, parentFolderPath, notePath, void 0, true);
|
|
686
|
+
break;
|
|
687
|
+
case "DeleteWithEmptyParents" /* DeleteWithEmptyParents */:
|
|
688
|
+
await (0, import_VaultEx.deleteEmptyFolderHierarchy)(app, parentFolderPath);
|
|
689
|
+
break;
|
|
690
|
+
default:
|
|
691
|
+
break;
|
|
538
692
|
}
|
|
539
693
|
}
|
|
540
694
|
}
|
|
541
|
-
function shouldInvokeHandler(plugin) {
|
|
542
|
-
const app = plugin.app;
|
|
543
|
-
const pluginId = plugin.manifest.id;
|
|
544
|
-
const renameDeleteHandlerPluginIds = getRenameDeleteHandlersMap(app);
|
|
545
|
-
const mainPluginId = Array.from(renameDeleteHandlerPluginIds.keys())[0];
|
|
546
|
-
if (mainPluginId !== pluginId) {
|
|
547
|
-
return false;
|
|
548
|
-
}
|
|
549
|
-
return true;
|
|
550
|
-
}
|
|
551
695
|
// Annotate the CommonJS export names for ESM import in node:
|
|
552
696
|
0 && (module.exports = {
|
|
553
697
|
EmptyAttachmentFolderBehavior,
|
|
554
698
|
registerRenameDeleteHandlers
|
|
555
699
|
});
|
|
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  AttachmentPathContext,\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, AttachmentPathContext.RenameNote);\n  const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n    ? await getAttachmentFolderPath(app, newPath, AttachmentPathContext.RenameNote)\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, AttachmentPathContext.RenameNote)) {\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        context: AttachmentPathContext.RenameNote,\n        notePathOrFile: newPath,\n        oldNotePathOrFile: oldPath,\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        getLibDebugger('RenameDeleteHandler:fillRenameMap')(`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    getLibDebugger('RenameDeleteHandler:handleDelete')(`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, AttachmentPathContext.DeleteNote);\n  const attachmentFolder = getFolderOrNull(app, attachmentFolderPath);\n\n  if (!attachmentFolder) {\n    return;\n  }\n\n  if (!(await hasOwnAttachmentFolder(app, path, AttachmentPathContext.DeleteNote))) {\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    getLibDebugger('RenameDeleteHandler:handleMetadataDeleted')(`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    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 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            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 (!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"]
}

|
|
700
|
+
//# 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  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  deleteEmptyFolderHierarchy,\n  deleteSafe\n} from './VaultEx.ts';\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 HandledRenameKey {\n  newPath: string;\n  oldPath: string;\n}\n\ninterface InterruptedRename {\n  combinedBacklinksMap: Map<string, Map<string, string>>;\n  oldPath: string;\n}\n\ninterface RenameHandlerOptions {\n  abortSignal: AbortSignal;\n  app: App;\n  handledRenames: HandledRenames;\n  interruptedCombinedBacklinksMap?: Map<string, Map<string, string>>;\n  interruptedRenamesMap: Map<string, InterruptedRename[]>;\n  newPath: string;\n  oldCache: CachedMetadata | null;\n  oldPath: string;\n  oldPathBacklinksMap: Map<string, Reference[]>;\n  settingsManager: SettingsManager;\n}\n\ninterface RenameMapOptions {\n  abortSignal: AbortSignal;\n  app: App;\n  newPath: string;\n  oldCache: CachedMetadata | null;\n  oldPath: string;\n  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 (!this.settingsManager.isNoteEx(this.file.path)) {\n      return;\n    }\n\n    const settings = this.settingsManager.getSettings();\n    if (!settings.shouldHandleDeletions) {\n      return;\n    }\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 cache = this.deletedMetadataCacheMap.get(this.file.path);\n    this.deletedMetadataCacheMap.delete(this.file.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(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.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.Keep);\n      }\n    }\n\n    await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths), this.file.path);\n    this.abortSignal.throwIfAborted();\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.emptyAttachmentFolderBehavior !== EmptyAttachmentFolderBehavior.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      this.app,\n      (abortSignal) => new DeleteHandler(this.app, file, abortSignal, this.settingsManager, this.deletedMetadataCacheMap).handle(),\n      this.abortSignal\n    );\n  }\n\n  private handleMetadataDeleted(file: TAbstractFile, prevCache: CachedMetadata | null): void {\n    if (!this.shouldInvokeHandler()) {\n      return;\n    }\n    addToQueue(this.app, () => {\n      new MetadataDeletedHandler(this.app, file, prevCache, this.settingsManager, this.deletedMetadataCacheMap).handle();\n    }, this.abortSignal);\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(this.app, (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(), this.abortSignal);\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\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 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(options: RenameHandlerOptions) {\n    this.app = options.app;\n    this.oldPath = options.oldPath;\n    this.newPath = options.newPath;\n    this.oldPathBacklinksMap = options.oldPathBacklinksMap;\n    this.oldCache = options.oldCache;\n    this.abortSignal = options.abortSignal;\n    this.settingsManager = options.settingsManager;\n    this.interruptedRenamesMap = options.interruptedRenamesMap;\n    this.oldPathLinks = this.oldCache ? getAllLinks(this.oldCache) : [];\n    this.handledRenames = options.handledRenames;\n    this.interruptedCombinedBacklinksMap = options.interruptedCombinedBacklinksMap ?? new Map<string, Map<string, string>>();\n  }\n\n  public async handle(): Promise<void> {\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    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.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          continue;\n        }\n        const fixedNewAttachmentPath = await this.renameHandled(oldAttachmentPath, newAttachmentPath);\n        this.abortSignal.throwIfAborted();\n        renameMap.set(oldAttachmentPath, fixedNewAttachmentPath);\n        parentFolderPaths.add(dirname(oldAttachmentPath));\n      }\n\n      await cleanupParentFolders(this.app, this.settingsManager.getSettings(), Array.from(parentFolderPaths), this.oldPath);\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        await editLinks(this.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: 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 (this.settingsManager.isNoteEx(this.newPath)) {\n        await updateLinksInFile(normalizeOptionalProperties<UpdateLinksInFileOptions>({\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(this.app, () => {\n        for (const orphanKey of orphanKeys) {\n          this.handledRenames.delete(orphanKey.oldPath, orphanKey.newPath);\n        }\n      }, this.abortSignal);\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.vault.rename(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(options: RenameMapOptions) {\n    this.abortSignal = options.abortSignal;\n    this.app = options.app;\n    this.settingsManager = options.settingsManager;\n    this.oldCache = options.oldCache;\n    this.oldPath = options.oldPath;\n    this.newPath = options.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 (!this.settingsManager.isNoteEx(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          if (oldAttachmentBacklinks.keys().length === 1) {\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          attachmentPathOrFile: oldAttachmentFile,\n          context: AttachmentPathContext.RenameNote,\n          notePathOrFile: this.newPath,\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 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\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.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\n  public isNoteEx(path: string): boolean {\n    const settings = this.getSettings();\n    return settings.isNote?.(path) ?? false;\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[], notePath: string): Promise<void> {\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"],
  "mappings": "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",
  "names": ["EmptyAttachmentFolderBehavior"]
}

|