obsidian-dev-utils 3.29.2 → 3.31.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 +15 -0
- package/dist/lib/obsidian/AttachmentPath.cjs +3 -3
- package/dist/lib/obsidian/Callout.cjs +2 -2
- package/dist/lib/obsidian/ChainedPromise.cjs +5 -5
- package/dist/lib/obsidian/ChainedPromise.d.ts +3 -2
- package/dist/lib/obsidian/FileSystem.cjs +3 -3
- package/dist/lib/obsidian/FileSystem.d.ts +5 -4
- package/dist/lib/obsidian/Link.cjs +22 -17
- package/dist/lib/obsidian/Link.d.ts +2 -2
- package/dist/lib/obsidian/Logger.cjs +1 -9
- package/dist/lib/obsidian/Logger.d.ts +2 -9
- package/dist/lib/obsidian/MetadataCache.cjs +6 -2
- package/dist/lib/obsidian/MetadataCache.d.ts +12 -0
- package/dist/lib/obsidian/Plugin/PluginBase.cjs +2 -2
- package/dist/lib/obsidian/RenameDeleteHandler.cjs +154 -152
- package/dist/lib/obsidian/Vault.cjs +5 -2
- package/dist/lib/scripts/version.cjs +4 -1
- package/package.json +1 -1
@@ -29,7 +29,6 @@ __export(RenameDeleteHandler_exports, {
|
|
29
29
|
});
|
30
30
|
module.exports = __toCommonJS(RenameDeleteHandler_exports);
|
31
31
|
var import_obsidian = require('obsidian');
|
32
|
-
var import_implementations = require('obsidian-typings/implementations');
|
33
32
|
var import_Object = require('../Object.cjs');
|
34
33
|
var import_Path = require('../Path.cjs');
|
35
34
|
var import_App = require('./App.cjs');
|
@@ -50,8 +49,8 @@ var __process = globalThis["process"] ?? {
|
|
50
49
|
"env": {},
|
51
50
|
"platform": "android"
|
52
51
|
};
|
53
|
-
const specialRenames = [];
|
54
52
|
const deletedMetadataCacheMap = /* @__PURE__ */ new Map();
|
53
|
+
const handledRenames = /* @__PURE__ */ new Set();
|
55
54
|
function registerRenameDeleteHandlers(plugin, settingsBuilder) {
|
56
55
|
const renameDeleteHandlersMap = getRenameDeleteHandlersMap(plugin.app);
|
57
56
|
const pluginId = plugin.manifest.id;
|
@@ -67,7 +66,8 @@ function registerRenameDeleteHandlers(plugin, settingsBuilder) {
|
|
67
66
|
if (!shouldInvokeHandler(app, pluginId, "Delete")) {
|
68
67
|
return;
|
69
68
|
}
|
70
|
-
|
69
|
+
const path = file.path;
|
70
|
+
(0, import_ChainedPromise.chain)(app, () => handleDelete(app, path));
|
71
71
|
})
|
72
72
|
);
|
73
73
|
plugin.registerEvent(
|
@@ -75,7 +75,8 @@ function registerRenameDeleteHandlers(plugin, settingsBuilder) {
|
|
75
75
|
if (!shouldInvokeHandler(app, pluginId, "Rename")) {
|
76
76
|
return;
|
77
77
|
}
|
78
|
-
|
78
|
+
const newPath = file.path;
|
79
|
+
(0, import_ChainedPromise.chain)(app, () => handleRename(app, oldPath, newPath));
|
79
80
|
})
|
80
81
|
);
|
81
82
|
plugin.registerEvent(
|
@@ -100,83 +101,153 @@ function logPluginSettingsOrder(app) {
|
|
100
101
|
const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);
|
101
102
|
console.debug(`Rename/delete handlers will use plugin settings in the following order: ${Array.from(renameDeleteHandlersMap.keys()).join(", ")}`);
|
102
103
|
}
|
103
|
-
async function handleRename(app,
|
104
|
-
console.debug(`Handle Rename ${oldPath} -> ${
|
105
|
-
|
106
|
-
|
107
|
-
}
|
108
|
-
const specialRename = specialRenames.find((x) => x.oldPath === file.path);
|
109
|
-
if (specialRename) {
|
110
|
-
const newTempPath = await (0, import_Vault.renameSafe)(app, file, specialRename.tempPath);
|
111
|
-
specialRename.tempPath = newTempPath;
|
104
|
+
async function handleRename(app, oldPath, newPath) {
|
105
|
+
console.debug(`Handle Rename ${oldPath} -> ${newPath}`);
|
106
|
+
const newFile = (0, import_FileSystem.getFileOrNull)(app, newPath);
|
107
|
+
if (!newFile) {
|
112
108
|
return;
|
113
109
|
}
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
newPath: file.path,
|
118
|
-
tempPath: (0, import_Path.join)(file.parent?.path ?? "", "__temp__" + file.name)
|
119
|
-
});
|
120
|
-
await (0, import_Vault.renameSafe)(app, file, oldPath);
|
110
|
+
const key = makeKey(oldPath, newPath);
|
111
|
+
if (handledRenames.has(key)) {
|
112
|
+
handledRenames.delete(key);
|
121
113
|
return;
|
122
114
|
}
|
123
115
|
const updateAllLinks = app.fileManager.updateAllLinks;
|
116
|
+
app.fileManager.updateAllLinks = async () => {
|
117
|
+
};
|
124
118
|
try {
|
125
|
-
app
|
126
|
-
|
127
|
-
const renameMap = /* @__PURE__ */ new Map();
|
128
|
-
await fillRenameMap(app, file, oldPath, renameMap);
|
129
|
-
renameMap.set(oldPath, file.path);
|
130
|
-
for (const [oldPath2, newPath2] of renameMap.entries()) {
|
131
|
-
await processRename(app, oldPath2, newPath2, renameMap);
|
132
|
-
}
|
119
|
+
await renameHandled(app, newPath, oldPath);
|
120
|
+
await processAndRename(app, oldPath, newPath);
|
133
121
|
} finally {
|
134
122
|
app.fileManager.updateAllLinks = updateAllLinks;
|
135
|
-
const
|
136
|
-
|
137
|
-
|
138
|
-
|
123
|
+
const orphanKeys = Array.from(handledRenames);
|
124
|
+
(0, import_ChainedPromise.chain)(app, () => {
|
125
|
+
for (const key2 of orphanKeys) {
|
126
|
+
handledRenames.delete(key2);
|
127
|
+
}
|
128
|
+
});
|
129
|
+
}
|
130
|
+
}
|
131
|
+
async function processAndRename(app, oldPath, newPath) {
|
132
|
+
if (app.vault.adapter.insensitive && newPath.toLowerCase() === oldPath.toLowerCase() && (0, import_Path.dirname)(newPath) === (0, import_Path.dirname)(oldPath)) {
|
133
|
+
const tempPath = (0, import_Path.join)((0, import_Path.dirname)(oldPath), "__temp__" + (0, import_Path.basename)(newPath));
|
134
|
+
await processAndRename(app, oldPath, tempPath);
|
135
|
+
await processAndRename(app, tempPath, newPath);
|
136
|
+
return;
|
137
|
+
}
|
138
|
+
const settings = getSettings(app);
|
139
|
+
const renameMap = /* @__PURE__ */ new Map();
|
140
|
+
await fillRenameMap(app, oldPath, newPath, renameMap);
|
141
|
+
const backlinksMap = /* @__PURE__ */ new Map();
|
142
|
+
for (const oldPath2 of renameMap.keys()) {
|
143
|
+
const currentBacklinksMap = await (0, import_MetadataCache.getBacklinksMap)(app, [oldPath2]);
|
144
|
+
for (const [backlinkPath, links] of currentBacklinksMap.entries()) {
|
145
|
+
const newBacklinkPath = renameMap.get(backlinkPath) ?? backlinkPath;
|
146
|
+
const linkJsonToPathMap = backlinksMap.get(newBacklinkPath) ?? /* @__PURE__ */ new Map();
|
147
|
+
backlinksMap.set(newBacklinkPath, linkJsonToPathMap);
|
148
|
+
for (const link of links) {
|
149
|
+
linkJsonToPathMap.set((0, import_Object.toJson)(link), oldPath2);
|
150
|
+
}
|
151
|
+
}
|
152
|
+
}
|
153
|
+
const parentFolders = /* @__PURE__ */ new Set();
|
154
|
+
for (const entry of renameMap.entries()) {
|
155
|
+
const oldRelatedPath = entry[0];
|
156
|
+
let newRelatedPath = entry[1];
|
157
|
+
newRelatedPath = await renameHandled(app, oldRelatedPath, newRelatedPath);
|
158
|
+
renameMap.set(oldRelatedPath, newRelatedPath);
|
159
|
+
parentFolders.add((0, import_Path.dirname)(oldRelatedPath));
|
160
|
+
}
|
161
|
+
if (settings.shouldDeleteEmptyFolders) {
|
162
|
+
for (const parentFolder of parentFolders) {
|
163
|
+
await (0, import_Vault.deleteEmptyFolderHierarchy)(app, parentFolder);
|
139
164
|
}
|
140
165
|
}
|
166
|
+
for (const [newBacklinkPath, linkJsonToPathMap] of backlinksMap.entries()) {
|
167
|
+
await (0, import_Link.editLinks)(app, newBacklinkPath, (link) => {
|
168
|
+
const oldRelatedPath = linkJsonToPathMap.get((0, import_Object.toJson)(link));
|
169
|
+
if (!oldRelatedPath) {
|
170
|
+
return;
|
171
|
+
}
|
172
|
+
const newRelatedPath = renameMap.get(oldRelatedPath);
|
173
|
+
if (!newRelatedPath) {
|
174
|
+
return;
|
175
|
+
}
|
176
|
+
return (0, import_Link.updateLink)({
|
177
|
+
app,
|
178
|
+
link,
|
179
|
+
pathOrFile: newRelatedPath,
|
180
|
+
oldPathOrFile: oldRelatedPath,
|
181
|
+
sourcePathOrFile: newBacklinkPath,
|
182
|
+
renameMap,
|
183
|
+
shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases
|
184
|
+
});
|
185
|
+
});
|
186
|
+
}
|
187
|
+
if ((0, import_FileSystem.isCanvasFile)(newPath)) {
|
188
|
+
await (0, import_Vault.processWithRetry)(app, newPath, (content) => {
|
189
|
+
const canvasData = JSON.parse(content);
|
190
|
+
for (const node of canvasData.nodes) {
|
191
|
+
if (node.type !== "file") {
|
192
|
+
continue;
|
193
|
+
}
|
194
|
+
const newPath2 = renameMap.get(node.file);
|
195
|
+
if (!newPath2) {
|
196
|
+
continue;
|
197
|
+
}
|
198
|
+
node.file = newPath2;
|
199
|
+
}
|
200
|
+
return (0, import_Object.toJson)(canvasData);
|
201
|
+
});
|
202
|
+
} else if ((0, import_FileSystem.isMarkdownFile)(newPath)) {
|
203
|
+
await (0, import_Link.updateLinksInFile)({
|
204
|
+
app,
|
205
|
+
pathOrFile: newPath,
|
206
|
+
oldPathOrFile: oldPath,
|
207
|
+
renameMap,
|
208
|
+
shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases
|
209
|
+
});
|
210
|
+
}
|
141
211
|
}
|
142
|
-
async function handleDelete(app,
|
143
|
-
console.debug(`Handle Delete ${
|
144
|
-
if (!(0, import_FileSystem.isNote)(
|
212
|
+
async function handleDelete(app, path) {
|
213
|
+
console.debug(`Handle Delete ${path}`);
|
214
|
+
if (!(0, import_FileSystem.isNote)(path)) {
|
145
215
|
return;
|
146
216
|
}
|
147
217
|
const settings = getSettings(app);
|
148
218
|
if (!settings.shouldDeleteOrphanAttachments) {
|
149
219
|
return;
|
150
220
|
}
|
151
|
-
const cache = deletedMetadataCacheMap.get(
|
152
|
-
deletedMetadataCacheMap.delete(
|
221
|
+
const cache = deletedMetadataCacheMap.get(path);
|
222
|
+
deletedMetadataCacheMap.delete(path);
|
153
223
|
if (cache) {
|
154
224
|
const links = (0, import_MetadataCache.getAllLinks)(cache);
|
155
225
|
for (const link of links) {
|
156
|
-
const attachmentFile = (0, import_Link.extractLinkFile)(app, link,
|
226
|
+
const attachmentFile = (0, import_Link.extractLinkFile)(app, link, path);
|
157
227
|
if (!attachmentFile) {
|
158
228
|
continue;
|
159
229
|
}
|
160
230
|
if ((0, import_FileSystem.isNote)(attachmentFile)) {
|
161
231
|
continue;
|
162
232
|
}
|
163
|
-
await (0, import_Vault.deleteSafe)(app, attachmentFile,
|
233
|
+
await (0, import_Vault.deleteSafe)(app, attachmentFile, path, settings.shouldDeleteEmptyFolders);
|
164
234
|
}
|
165
235
|
}
|
166
|
-
const attachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(app,
|
236
|
+
const attachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(app, path);
|
167
237
|
const attachmentFolder = (0, import_FileSystem.getFolderOrNull)(app, attachmentFolderPath);
|
168
238
|
if (!attachmentFolder) {
|
169
239
|
return;
|
170
240
|
}
|
171
|
-
await (0, import_Vault.deleteSafe)(app, attachmentFolder,
|
241
|
+
await (0, import_Vault.deleteSafe)(app, attachmentFolder, path, false, settings.shouldDeleteEmptyFolders);
|
172
242
|
}
|
173
|
-
async function fillRenameMap(app,
|
174
|
-
|
243
|
+
async function fillRenameMap(app, oldPath, newPath, renameMap) {
|
244
|
+
renameMap.set(oldPath, newPath);
|
245
|
+
if (!(0, import_FileSystem.isNote)(oldPath)) {
|
175
246
|
return;
|
176
247
|
}
|
177
248
|
const settings = getSettings(app);
|
178
249
|
const oldAttachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(app, oldPath);
|
179
|
-
const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder ? await (0, import_AttachmentPath.getAttachmentFolderPath)(app,
|
250
|
+
const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder ? await (0, import_AttachmentPath.getAttachmentFolderPath)(app, newPath) : oldAttachmentFolderPath;
|
180
251
|
const dummyOldAttachmentFolderPath = await (0, import_AttachmentPath.getAttachmentFolderPath)(app, (0, import_Path.join)((0, import_Path.dirname)(oldPath), "DUMMY_FILE.md"));
|
181
252
|
const oldAttachmentFolder = (0, import_FileSystem.getFolderOrNull)(app, oldAttachmentFolderPath);
|
182
253
|
if (!oldAttachmentFolder) {
|
@@ -185,131 +256,53 @@ async function fillRenameMap(app, file, oldPath, renameMap) {
|
|
185
256
|
if (oldAttachmentFolderPath === newAttachmentFolderPath && !settings.shouldRenameAttachmentFiles) {
|
186
257
|
return;
|
187
258
|
}
|
188
|
-
const
|
259
|
+
const oldAttachmentFiles = [];
|
189
260
|
if (oldAttachmentFolderPath === dummyOldAttachmentFolderPath) {
|
190
|
-
const
|
191
|
-
if (!
|
261
|
+
const oldCache = await (0, import_MetadataCache.getCacheSafe)(app, oldPath);
|
262
|
+
if (!oldCache) {
|
192
263
|
return;
|
193
264
|
}
|
194
|
-
for (const
|
195
|
-
const
|
196
|
-
if (!
|
265
|
+
for (const oldLink of (0, import_MetadataCache.getAllLinks)(oldCache)) {
|
266
|
+
const oldAttachmentFile = (0, import_Link.extractLinkFile)(app, oldLink, oldPath);
|
267
|
+
if (!oldAttachmentFile) {
|
197
268
|
continue;
|
198
269
|
}
|
199
|
-
if (
|
200
|
-
const
|
201
|
-
if (
|
202
|
-
|
270
|
+
if (oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {
|
271
|
+
const oldAttachmentBacklinks = await (0, import_MetadataCache.getBacklinksForFileSafe)(app, oldAttachmentFile);
|
272
|
+
if (oldAttachmentBacklinks.keys().length === 1) {
|
273
|
+
oldAttachmentFiles.push(oldAttachmentFile);
|
203
274
|
}
|
204
275
|
}
|
205
276
|
}
|
206
277
|
} else {
|
207
|
-
import_obsidian.Vault.recurseChildren(oldAttachmentFolder, (
|
208
|
-
if (
|
209
|
-
|
278
|
+
import_obsidian.Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {
|
279
|
+
if (oldAttachmentFile instanceof import_obsidian.TFile) {
|
280
|
+
oldAttachmentFiles.push(oldAttachmentFile);
|
210
281
|
}
|
211
282
|
});
|
212
283
|
}
|
213
|
-
const
|
214
|
-
|
215
|
-
|
284
|
+
const oldBasename = (0, import_Path.basename)(oldPath, (0, import_Path.extname)(oldPath));
|
285
|
+
const newBasename = (0, import_Path.basename)(newPath, (0, import_Path.extname)(newPath));
|
286
|
+
for (const oldAttachmentFile of oldAttachmentFiles) {
|
287
|
+
if ((0, import_FileSystem.isNote)(oldAttachmentFile)) {
|
216
288
|
continue;
|
217
289
|
}
|
218
|
-
const relativePath = (0, import_Path.relative)(oldAttachmentFolderPath,
|
290
|
+
const relativePath = (0, import_Path.relative)(oldAttachmentFolderPath, oldAttachmentFile.path);
|
219
291
|
const newDir = (0, import_Path.join)(newAttachmentFolderPath, (0, import_Path.dirname)(relativePath));
|
220
|
-
const newChildBasename = settings.shouldRenameAttachmentFiles ?
|
221
|
-
let newChildPath = (0, import_Path.join)(newDir, (0, import_Path.makeFileName)(newChildBasename,
|
222
|
-
if (
|
223
|
-
if (settings.shouldDeleteConflictingAttachments) {
|
224
|
-
const newChildFile = (0, import_FileSystem.getFileOrNull)(app, newChildPath);
|
225
|
-
if (newChildFile) {
|
226
|
-
await app.fileManager.trashFile(newChildFile);
|
227
|
-
}
|
228
|
-
} else {
|
229
|
-
newChildPath = app.vault.getAvailablePath((0, import_Path.join)(newDir, newChildBasename), child.extension);
|
230
|
-
}
|
231
|
-
renameMap.set(child.path, newChildPath);
|
232
|
-
}
|
233
|
-
}
|
234
|
-
}
|
235
|
-
async function processRename(app, oldPath, newPath, renameMap) {
|
236
|
-
const settings = getSettings(app);
|
237
|
-
let oldFile = (0, import_FileSystem.getFileOrNull)(app, oldPath);
|
238
|
-
if (oldFile) {
|
239
|
-
const oldFolder = oldFile.parent;
|
240
|
-
newPath = await (0, import_Vault.renameSafe)(app, oldFile, newPath);
|
241
|
-
renameMap.set(oldPath, newPath);
|
242
|
-
if (settings.shouldDeleteEmptyFolders) {
|
243
|
-
await (0, import_Vault.deleteEmptyFolderHierarchy)(app, oldFolder);
|
244
|
-
}
|
245
|
-
}
|
246
|
-
oldFile = (0, import_implementations.createTFileInstance)(app.vault, oldPath);
|
247
|
-
const newFile = (0, import_FileSystem.getFileOrNull)(app, newPath);
|
248
|
-
if (!oldFile.deleted || !newFile) {
|
249
|
-
throw new Error(`Could not rename ${oldPath} to ${newPath}`);
|
250
|
-
}
|
251
|
-
if (!settings.shouldUpdateLinks) {
|
252
|
-
return;
|
253
|
-
}
|
254
|
-
const backlinks = await (0, import_MetadataCache.getBacklinksMap)(app, [oldFile, newFile]);
|
255
|
-
for (const parentNotePath of backlinks.keys()) {
|
256
|
-
let parentNote = (0, import_FileSystem.getFileOrNull)(app, parentNotePath);
|
257
|
-
if (!parentNote) {
|
258
|
-
const newParentNotePath = renameMap.get(parentNotePath);
|
259
|
-
if (newParentNotePath) {
|
260
|
-
parentNote = (0, import_FileSystem.getFileOrNull)(app, newParentNotePath);
|
261
|
-
}
|
262
|
-
}
|
263
|
-
if (!parentNote) {
|
264
|
-
console.warn(`Parent note not found: ${parentNotePath}`);
|
292
|
+
const newChildBasename = settings.shouldRenameAttachmentFiles ? oldAttachmentFile.basename.replaceAll(oldBasename, newBasename) : oldAttachmentFile.basename;
|
293
|
+
let newChildPath = (0, import_Path.join)(newDir, (0, import_Path.makeFileName)(newChildBasename, oldAttachmentFile.extension));
|
294
|
+
if (oldAttachmentFile.path === newChildPath) {
|
265
295
|
continue;
|
266
296
|
}
|
267
|
-
|
268
|
-
const
|
269
|
-
|
270
|
-
|
271
|
-
for (const link of links) {
|
272
|
-
changes.push({
|
273
|
-
startIndex: link.position.start.offset,
|
274
|
-
endIndex: link.position.end.offset,
|
275
|
-
oldContent: link.original,
|
276
|
-
newContent: (0, import_Link.updateLink)({
|
277
|
-
app,
|
278
|
-
link,
|
279
|
-
pathOrFile: newFile,
|
280
|
-
oldPathOrFile: oldPath,
|
281
|
-
sourcePathOrFile: parentNote,
|
282
|
-
renameMap,
|
283
|
-
shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases
|
284
|
-
})
|
285
|
-
});
|
297
|
+
if (settings.shouldDeleteConflictingAttachments) {
|
298
|
+
const newChildFile = (0, import_FileSystem.getFileOrNull)(app, newChildPath);
|
299
|
+
if (newChildFile) {
|
300
|
+
await app.fileManager.trashFile(newChildFile);
|
286
301
|
}
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
|
291
|
-
await (0, import_Vault.processWithRetry)(app, newFile, (content) => {
|
292
|
-
const canvasData = JSON.parse(content);
|
293
|
-
for (const node of canvasData.nodes) {
|
294
|
-
if (node.type !== "file") {
|
295
|
-
continue;
|
296
|
-
}
|
297
|
-
const newPath2 = renameMap.get(node.file);
|
298
|
-
if (!newPath2) {
|
299
|
-
continue;
|
300
|
-
}
|
301
|
-
node.file = newPath2;
|
302
|
-
}
|
303
|
-
return (0, import_Object.toJson)(canvasData);
|
304
|
-
});
|
305
|
-
} else if ((0, import_FileSystem.isMarkdownFile)(newFile)) {
|
306
|
-
await (0, import_Link.updateLinksInFile)({
|
307
|
-
app,
|
308
|
-
pathOrFile: newFile,
|
309
|
-
oldPathOrFile: oldPath,
|
310
|
-
renameMap,
|
311
|
-
shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases
|
312
|
-
});
|
302
|
+
} else {
|
303
|
+
newChildPath = app.vault.getAvailablePath((0, import_Path.join)(newDir, newChildBasename), oldAttachmentFile.extension);
|
304
|
+
}
|
305
|
+
renameMap.set(oldAttachmentFile.path, newChildPath);
|
313
306
|
}
|
314
307
|
}
|
315
308
|
function getSettings(app) {
|
@@ -329,8 +322,17 @@ function handleMetadataDeleted(file, prevCache) {
|
|
329
322
|
deletedMetadataCacheMap.set(file.path, prevCache);
|
330
323
|
}
|
331
324
|
}
|
325
|
+
function makeKey(oldPath, newPath) {
|
326
|
+
return `${oldPath} -> ${newPath}`;
|
327
|
+
}
|
328
|
+
async function renameHandled(app, oldPath, newPath) {
|
329
|
+
newPath = await (0, import_Vault.renameSafe)(app, (0, import_FileSystem.getFile)(app, oldPath), newPath);
|
330
|
+
const key = makeKey(oldPath, newPath);
|
331
|
+
handledRenames.add(key);
|
332
|
+
return newPath;
|
333
|
+
}
|
332
334
|
// Annotate the CommonJS export names for ESM import in node:
|
333
335
|
0 && (module.exports = {
|
334
336
|
registerRenameDeleteHandlers
|
335
337
|
});
|
336
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/RenameDeleteHandler.ts"],
  "sourcesContent": ["var __import_meta_url = globalThis['import.meta.url'] ?? (()=>{if(typeof __filename!==\"string\"){return new URL(window.location.href)}return require(\"node:url\").pathToFileURL(__filename)})();\nvar __process = globalThis['process'] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\nimport type {\n  CachedMetadata,\n  Plugin,\n  TAbstractFile\n} from 'obsidian';\nimport {\n  App,\n  TFile,\n  Vault\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/canvas.js';\nimport { createTFileInstance } from 'obsidian-typings/implementations';\n\nimport { toJson } from '../Object.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  makeFileName,\n  relative\n} from '../Path.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport { getAttachmentFolderPath } from './AttachmentPath.ts';\nimport { chainAsyncFn } from './ChainedPromise.ts';\nimport {\n  getFileOrNull,\n  getFolderOrNull,\n  isCanvasFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport {\n  extractLinkFile,\n  updateLink,\n  updateLinksInFile\n} from './Link.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getBacklinksMap,\n  getCacheSafe\n} from './MetadataCache.ts';\nimport {\n  applyFileChanges,\n  deleteEmptyFolderHierarchy,\n  deleteSafe,\n  processWithRetry,\n  renameSafe\n} from './Vault.ts';\n\nconst specialRenames: SpecialRename[] = [];\nconst deletedMetadataCacheMap = new Map<string, CachedMetadata>();\n\ninterface SpecialRename {\n  oldPath: string;\n  newPath: string;\n  tempPath: string;\n}\n\n/**\n * Settings for the rename/delete handler.\n */\nexport interface RenameDeleteHandlerSettings {\n  /**\n   * Whether to delete conflicting attachments.\n   */\n  shouldDeleteConflictingAttachments: boolean;\n\n  /**\n   * Whether to delete empty folders.\n   */\n  shouldDeleteEmptyFolders: boolean;\n\n  /**\n   * Whether to delete orphan attachments after a delete.\n   */\n  shouldDeleteOrphanAttachments: 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 filename aliases when a note is renamed.\n   */\n  shouldUpdateFilenameAliases: boolean;\n\n  /**\n   * Whether to update links when a note or attachment is renamed.\n   */\n  shouldUpdateLinks: boolean;\n}\n\n/**\n * Registers the rename/delete handlers.\n * @param plugin - The plugin instance.\n * @param settingsBuilder - A function that returns the settings for the rename delete handler.\n * @returns void\n */\nexport function registerRenameDeleteHandlers(plugin: Plugin, settingsBuilder: () => Partial<RenameDeleteHandlerSettings>): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(plugin.app);\n  const pluginId = plugin.manifest.id;\n\n  renameDeleteHandlersMap.set(pluginId, settingsBuilder);\n  logPluginSettingsOrder(plugin.app);\n\n  plugin.register(() => {\n    renameDeleteHandlersMap.delete(pluginId);\n    logPluginSettingsOrder(plugin.app);\n  });\n\n  const app = plugin.app;\n  plugin.registerEvent(\n    app.vault.on('delete', (file) => {\n      if (!shouldInvokeHandler(app, pluginId, 'Delete')) {\n        return;\n      }\n      chainAsyncFn(app, () => handleDelete(app, file));\n    })\n  );\n\n  plugin.registerEvent(\n    app.vault.on('rename', (file, oldPath) => {\n      if (!shouldInvokeHandler(app, pluginId, 'Rename')) {\n        return;\n      }\n      chainAsyncFn(app, () => handleRename(app, file, oldPath));\n    })\n  );\n\n  plugin.registerEvent(\n    app.metadataCache.on('deleted', (file, prevCache) => {\n      handleMetadataDeleted(file, prevCache);\n    })\n  );\n}\n\nfunction shouldInvokeHandler(app: App, pluginId: string, handlerType: string): boolean {\n  const renameDeleteHandlerPluginIds = getRenameDeleteHandlersMap(app);\n  const mainPluginId = Array.from(renameDeleteHandlerPluginIds.keys())[0];\n  if (mainPluginId !== pluginId) {\n    console.debug(`${handlerType} handler for plugin ${pluginId} is skipped, because it is handled by plugin ${mainPluginId ?? '(none)'}`);\n    return false;\n  }\n  return true;\n}\n\nfunction getRenameDeleteHandlersMap(app: App): Map<string, () => Partial<RenameDeleteHandlerSettings>> {\n  return getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n}\n\nfunction logPluginSettingsOrder(app: App): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  console.debug(`Rename/delete handlers will use plugin settings in the following order: ${Array.from(renameDeleteHandlersMap.keys()).join(', ')}`);\n}\n\nasync function handleRename(app: App, file: TAbstractFile, oldPath: string): Promise<void> {\n  console.debug(`Handle Rename ${oldPath} -> ${file.path}`);\n\n  if (!(file instanceof TFile)) {\n    return;\n  }\n\n  const specialRename = specialRenames.find((x) => x.oldPath === file.path);\n  if (specialRename) {\n    const newTempPath = await renameSafe(app, file, specialRename.tempPath);\n    specialRename.tempPath = newTempPath;\n    return;\n  }\n\n  if (app.vault.adapter.insensitive && oldPath.toLowerCase() === file.path.toLowerCase() && dirname(oldPath) === dirname(file.path)) {\n    specialRenames.push({\n      oldPath,\n      newPath: file.path,\n      tempPath: join(file.parent?.path ?? '', '__temp__' + file.name)\n    });\n\n    await renameSafe(app, file, oldPath);\n    return;\n  }\n\n  // eslint-disable-next-line @typescript-eslint/unbound-method\n  const updateAllLinks = app.fileManager.updateAllLinks;\n  try {\n    app.fileManager.updateAllLinks = async (): Promise<void> => {\n      // do nothing\n    };\n\n    const renameMap = new Map<string, string>();\n    await fillRenameMap(app, file, oldPath, renameMap);\n    renameMap.set(oldPath, file.path);\n\n    for (const [oldPath2, newPath2] of renameMap.entries()) {\n      await processRename(app, oldPath2, newPath2, renameMap);\n    }\n  } finally {\n    app.fileManager.updateAllLinks = updateAllLinks;\n\n    const specialRename = specialRenames.find((x) => x.tempPath === file.path);\n    if (specialRename) {\n      await renameSafe(app, file, specialRename.newPath);\n      specialRenames.remove(specialRename);\n    }\n  }\n}\n\nasync function handleDelete(app: App, file: TAbstractFile): Promise<void> {\n  console.debug(`Handle Delete ${file.path}`);\n  if (!isNote(file)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldDeleteOrphanAttachments) {\n    return;\n  }\n\n  const cache = deletedMetadataCacheMap.get(file.path);\n  deletedMetadataCacheMap.delete(file.path);\n  if (cache) {\n    const links = getAllLinks(cache);\n\n    for (const link of links) {\n      const attachmentFile = extractLinkFile(app, link, file.path);\n      if (!attachmentFile) {\n        continue;\n      }\n\n      if (isNote(attachmentFile)) {\n        continue;\n      }\n\n      await deleteSafe(app, attachmentFile, file.path, settings.shouldDeleteEmptyFolders);\n    }\n  }\n\n  const attachmentFolderPath = await getAttachmentFolderPath(app, file.path);\n  const attachmentFolder = getFolderOrNull(app, attachmentFolderPath);\n\n  if (!attachmentFolder) {\n    return;\n  }\n\n  await deleteSafe(app, attachmentFolder, file.path, false, settings.shouldDeleteEmptyFolders);\n}\n\nasync function fillRenameMap(app: App, file: TFile, oldPath: string, renameMap: Map<string, string>): Promise<void> {\n  if (!isNote(file)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n\n  const oldAttachmentFolderPath = await getAttachmentFolderPath(app, oldPath);\n  const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n    ? await getAttachmentFolderPath(app, file.path)\n    : oldAttachmentFolderPath;\n  const dummyOldAttachmentFolderPath = await getAttachmentFolderPath(app, join(dirname(oldPath), 'DUMMY_FILE.md'));\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 children: TFile[] = [];\n\n  if (oldAttachmentFolderPath === dummyOldAttachmentFolderPath) {\n    const cache = await getCacheSafe(app, file);\n    if (!cache) {\n      return;\n    }\n    for (const link of getAllLinks(cache)) {\n      const attachmentFile = extractLinkFile(app, link, oldPath);\n      if (!attachmentFile) {\n        continue;\n      }\n\n      if (attachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n        const backlinks = await getBacklinksForFileSafe(app, attachmentFile);\n        if (backlinks.keys().length === 1) {\n          children.push(attachmentFile);\n        }\n      }\n    }\n  } else {\n    Vault.recurseChildren(oldAttachmentFolder, (child) => {\n      if (child instanceof TFile) {\n        children.push(child);\n      }\n    });\n  }\n\n  const oldNoteBaseName = basename(oldPath, extname(oldPath));\n\n  for (const child of children) {\n    if (isNote(child)) {\n      continue;\n    }\n    const relativePath = relative(oldAttachmentFolderPath, child.path);\n    const newDir = join(newAttachmentFolderPath, dirname(relativePath));\n    const newChildBasename = settings.shouldRenameAttachmentFiles\n      ? child.basename.replaceAll(oldNoteBaseName, file.basename)\n      : child.basename;\n    let newChildPath = join(newDir, makeFileName(newChildBasename, child.extension));\n    if (child.path !== newChildPath) {\n      if (settings.shouldDeleteConflictingAttachments) {\n        const newChildFile = getFileOrNull(app, newChildPath);\n        if (newChildFile) {\n          await app.fileManager.trashFile(newChildFile);\n        }\n      } else {\n        newChildPath = app.vault.getAvailablePath(join(newDir, newChildBasename), child.extension);\n      }\n      renameMap.set(child.path, newChildPath);\n    }\n  }\n}\n\nasync function processRename(app: App, oldPath: string, newPath: string, renameMap: Map<string, string>): Promise<void> {\n  const settings = getSettings(app);\n  let oldFile = getFileOrNull(app, oldPath);\n\n  if (oldFile) {\n    const oldFolder = oldFile.parent;\n    newPath = await renameSafe(app, oldFile, newPath);\n    renameMap.set(oldPath, newPath);\n\n    if (settings.shouldDeleteEmptyFolders) {\n      await deleteEmptyFolderHierarchy(app, oldFolder);\n    }\n  }\n\n  oldFile = createTFileInstance(app.vault, oldPath);\n  const newFile = getFileOrNull(app, newPath);\n\n  if (!oldFile.deleted || !newFile) {\n    throw new Error(`Could not rename ${oldPath} to ${newPath}`);\n  }\n\n  if (!settings.shouldUpdateLinks) {\n    return;\n  }\n\n  const backlinks = await getBacklinksMap(app, [oldFile, newFile]);\n\n  for (const parentNotePath of backlinks.keys()) {\n    let parentNote = getFileOrNull(app, parentNotePath);\n    if (!parentNote) {\n      const newParentNotePath = renameMap.get(parentNotePath);\n      if (newParentNotePath) {\n        parentNote = getFileOrNull(app, newParentNotePath);\n      }\n    }\n\n    if (!parentNote) {\n      console.warn(`Parent note not found: ${parentNotePath}`);\n      continue;\n    }\n\n    await applyFileChanges(app, parentNote, async () => {\n      const backlinks = await getBacklinksMap(app, [oldFile, newFile]);\n      const links = backlinks.get(parentNotePath) ?? [];\n      const changes = [];\n\n      for (const link of links) {\n        changes.push({\n          startIndex: link.position.start.offset,\n          endIndex: link.position.end.offset,\n          oldContent: link.original,\n          newContent: updateLink({\n            app: app,\n            link,\n            pathOrFile: newFile,\n            oldPathOrFile: oldPath,\n            sourcePathOrFile: parentNote,\n            renameMap,\n            shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases\n          })\n        });\n      }\n\n      return changes;\n    });\n  }\n\n  if (isCanvasFile(newFile)) {\n    await processWithRetry(app, newFile, (content) => {\n      const canvasData = JSON.parse(content) as CanvasData;\n      for (const node of canvasData.nodes) {\n        if (node.type !== 'file') {\n          continue;\n        }\n        const newPath = renameMap.get(node.file);\n        if (!newPath) {\n          continue;\n        }\n        node.file = newPath;\n      }\n      return toJson(canvasData);\n    });\n  } else if (isMarkdownFile(newFile)) {\n    await updateLinksInFile({\n      app: app,\n      pathOrFile: newFile,\n      oldPathOrFile: oldPath,\n      renameMap,\n      shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases\n    });\n  }\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  for (const settingsBuilder of settingsBuilders) {\n    const newSettings = settingsBuilder();\n    for (const [key, value] of Object.entries(newSettings) as [keyof RenameDeleteHandlerSettings, boolean][]) {\n      settings[key] ||= value;\n    }\n  }\n\n  return settings;\n}\n\nfunction handleMetadataDeleted(file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  if (isMarkdownFile(file) && prevCache) {\n    deletedMetadataCacheMap.set(file.path, prevCache);\n  }\n}\n"],
  "mappings": "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",
  "names": ["specialRename", "backlinks", "newPath"]
}

|
338
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/RenameDeleteHandler.ts"],
  "sourcesContent": ["var __import_meta_url = globalThis['import.meta.url'] ?? (()=>{if(typeof __filename!==\"string\"){return new URL(window.location.href)}return require(\"node:url\").pathToFileURL(__filename)})();\nvar __process = globalThis['process'] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\nimport type {\n  CachedMetadata,\n  Plugin,\n  TAbstractFile\n} from 'obsidian';\nimport {\n  App,\n  TFile,\n  Vault\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/canvas.js';\n\nimport { toJson } from '../Object.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  makeFileName,\n  relative\n} from '../Path.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport { getAttachmentFolderPath } from './AttachmentPath.ts';\nimport { chain } from './ChainedPromise.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolderOrNull,\n  isCanvasFile,\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  getBacklinksForFileSafe,\n  getBacklinksMap,\n  getCacheSafe\n} from './MetadataCache.ts';\nimport {\n  deleteEmptyFolderHierarchy,\n  deleteSafe,\n  processWithRetry,\n  renameSafe\n} from './Vault.ts';\n\nconst deletedMetadataCacheMap = new Map<string, CachedMetadata>();\nconst handledRenames = new Set<string>();\n\n/**\n * Settings for the rename/delete handler.\n */\nexport interface RenameDeleteHandlerSettings {\n  /**\n   * Whether to delete conflicting attachments.\n   */\n  shouldDeleteConflictingAttachments: boolean;\n\n  /**\n   * Whether to delete empty folders.\n   */\n  shouldDeleteEmptyFolders: boolean;\n\n  /**\n   * Whether to delete orphan attachments after a delete.\n   */\n  shouldDeleteOrphanAttachments: 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 filename aliases when a note is renamed.\n   */\n  shouldUpdateFilenameAliases: boolean;\n\n  /**\n   * Whether to update links when a note or attachment is renamed.\n   */\n  shouldUpdateLinks: boolean;\n}\n\n/**\n * Registers the rename/delete handlers.\n * @param plugin - The plugin instance.\n * @param settingsBuilder - A function that returns the settings for the rename delete handler.\n * @returns void\n */\nexport function registerRenameDeleteHandlers(plugin: Plugin, settingsBuilder: () => Partial<RenameDeleteHandlerSettings>): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(plugin.app);\n  const pluginId = plugin.manifest.id;\n\n  renameDeleteHandlersMap.set(pluginId, settingsBuilder);\n  logPluginSettingsOrder(plugin.app);\n\n  plugin.register(() => {\n    renameDeleteHandlersMap.delete(pluginId);\n    logPluginSettingsOrder(plugin.app);\n  });\n\n  const app = plugin.app;\n  plugin.registerEvent(\n    app.vault.on('delete', (file) => {\n      if (!shouldInvokeHandler(app, pluginId, 'Delete')) {\n        return;\n      }\n      const path = file.path;\n      chain(app, () => handleDelete(app, path));\n    })\n  );\n\n  plugin.registerEvent(\n    app.vault.on('rename', (file, oldPath) => {\n      if (!shouldInvokeHandler(app, pluginId, 'Rename')) {\n        return;\n      }\n      const newPath = file.path;\n      chain(app, () => handleRename(app, oldPath, newPath));\n    })\n  );\n\n  plugin.registerEvent(\n    app.metadataCache.on('deleted', (file, prevCache) => {\n      handleMetadataDeleted(file, prevCache);\n    })\n  );\n}\n\nfunction shouldInvokeHandler(app: App, pluginId: string, handlerType: string): boolean {\n  const renameDeleteHandlerPluginIds = getRenameDeleteHandlersMap(app);\n  const mainPluginId = Array.from(renameDeleteHandlerPluginIds.keys())[0];\n  if (mainPluginId !== pluginId) {\n    console.debug(`${handlerType} handler for plugin ${pluginId} is skipped, because it is handled by plugin ${mainPluginId ?? '(none)'}`);\n    return false;\n  }\n  return true;\n}\n\nfunction getRenameDeleteHandlersMap(app: App): Map<string, () => Partial<RenameDeleteHandlerSettings>> {\n  return getObsidianDevUtilsState(app, 'renameDeleteHandlersMap', new Map<string, () => Partial<RenameDeleteHandlerSettings>>()).value;\n}\n\nfunction logPluginSettingsOrder(app: App): void {\n  const renameDeleteHandlersMap = getRenameDeleteHandlersMap(app);\n  console.debug(`Rename/delete handlers will use plugin settings in the following order: ${Array.from(renameDeleteHandlersMap.keys()).join(', ')}`);\n}\n\nasync function handleRename(app: App, oldPath: string, newPath: string): Promise<void> {\n  console.debug(`Handle Rename ${oldPath} -> ${newPath}`);\n\n  const newFile = getFileOrNull(app, newPath);\n  if (!newFile) {\n    return;\n  }\n\n  const key = makeKey(oldPath, newPath);\n  if (handledRenames.has(key)) {\n    handledRenames.delete(key);\n    return;\n  }\n\n  // eslint-disable-next-line @typescript-eslint/unbound-method\n  const updateAllLinks = app.fileManager.updateAllLinks;\n  app.fileManager.updateAllLinks = async (): Promise<void> => {\n    // do nothing\n  };\n  try {\n    await renameHandled(app, newPath, oldPath);\n    await processAndRename(app, oldPath, newPath);\n  } finally {\n    app.fileManager.updateAllLinks = updateAllLinks;\n    const orphanKeys = Array.from(handledRenames);\n    chain(app, () => {\n      for (const key of orphanKeys) {\n        handledRenames.delete(key);\n      }\n    });\n  }\n}\n\nasync function processAndRename(app: App, oldPath: string, newPath: string): Promise<void> {\n  if (app.vault.adapter.insensitive && newPath.toLowerCase() === oldPath.toLowerCase() && dirname(newPath) === dirname(oldPath)) {\n    const tempPath = join(dirname(oldPath), '__temp__' + basename(newPath));\n    await processAndRename(app, oldPath, tempPath);\n    await processAndRename(app, tempPath, newPath);\n    return;\n  }\n\n  const settings = getSettings(app);\n  const renameMap = new Map<string, string>();\n  await fillRenameMap(app, oldPath, newPath, renameMap);\n\n  const backlinksMap = new Map<string, Map<string, string>>();\n\n  for (const oldPath2 of renameMap.keys()) {\n    const currentBacklinksMap = await getBacklinksMap(app, [oldPath2]);\n    for (const [backlinkPath, links] of currentBacklinksMap.entries()) {\n      const newBacklinkPath = renameMap.get(backlinkPath) ?? backlinkPath;\n      const linkJsonToPathMap = backlinksMap.get(newBacklinkPath) ?? new Map<string, string>();\n      backlinksMap.set(newBacklinkPath, linkJsonToPathMap);\n      for (const link of links) {\n        linkJsonToPathMap.set(toJson(link), oldPath2);\n      }\n    }\n  }\n\n  const parentFolders = new Set<string>();\n\n  for (const entry of renameMap.entries()) {\n    const oldRelatedPath = entry[0];\n    let newRelatedPath = entry[1];\n    newRelatedPath = await renameHandled(app, oldRelatedPath, newRelatedPath);\n    renameMap.set(oldRelatedPath, newRelatedPath);\n    parentFolders.add(dirname(oldRelatedPath));\n  }\n\n  if (settings.shouldDeleteEmptyFolders) {\n    for (const parentFolder of parentFolders) {\n      await deleteEmptyFolderHierarchy(app, parentFolder);\n    }\n  }\n\n  for (const [newBacklinkPath, linkJsonToPathMap] of backlinksMap.entries()) {\n    await editLinks(app, newBacklinkPath, (link) => {\n      const oldRelatedPath = linkJsonToPathMap.get(toJson(link));\n      if (!oldRelatedPath) {\n        return;\n      }\n\n      const newRelatedPath = renameMap.get(oldRelatedPath);\n      if (!newRelatedPath) {\n        return;\n      }\n\n      return updateLink({\n        app: app,\n        link,\n        pathOrFile: newRelatedPath,\n        oldPathOrFile: oldRelatedPath,\n        sourcePathOrFile: newBacklinkPath,\n        renameMap,\n        shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases\n      });\n    });\n  }\n\n  if (isCanvasFile(newPath)) {\n    await processWithRetry(app, newPath, (content) => {\n      const canvasData = JSON.parse(content) as CanvasData;\n      for (const node of canvasData.nodes) {\n        if (node.type !== 'file') {\n          continue;\n        }\n        const newPath = renameMap.get(node.file);\n        if (!newPath) {\n          continue;\n        }\n        node.file = newPath;\n      }\n      return toJson(canvasData);\n    });\n  } else if (isMarkdownFile(newPath)) {\n    await updateLinksInFile({\n      app: app,\n      pathOrFile: newPath,\n      oldPathOrFile: oldPath,\n      renameMap,\n      shouldUpdateFilenameAlias: settings.shouldUpdateFilenameAliases\n    });\n  }\n}\n\nasync function handleDelete(app: App, path: string): Promise<void> {\n  console.debug(`Handle Delete ${path}`);\n  if (!isNote(path)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n  if (!settings.shouldDeleteOrphanAttachments) {\n    return;\n  }\n\n  const cache = deletedMetadataCacheMap.get(path);\n  deletedMetadataCacheMap.delete(path);\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 (isNote(attachmentFile)) {\n        continue;\n      }\n\n      await deleteSafe(app, attachmentFile, path, settings.shouldDeleteEmptyFolders);\n    }\n  }\n\n  const attachmentFolderPath = await getAttachmentFolderPath(app, path);\n  const attachmentFolder = getFolderOrNull(app, attachmentFolderPath);\n\n  if (!attachmentFolder) {\n    return;\n  }\n\n  await deleteSafe(app, attachmentFolder, path, false, settings.shouldDeleteEmptyFolders);\n}\n\nasync function fillRenameMap(app: App, oldPath: string, newPath: string, renameMap: Map<string, string>): Promise<void> {\n  renameMap.set(oldPath, newPath);\n\n  if (!isNote(oldPath)) {\n    return;\n  }\n\n  const settings = getSettings(app);\n\n  const oldAttachmentFolderPath = await getAttachmentFolderPath(app, oldPath);\n  const newAttachmentFolderPath = settings.shouldRenameAttachmentFolder\n    ? await getAttachmentFolderPath(app, newPath)\n    : oldAttachmentFolderPath;\n  const dummyOldAttachmentFolderPath = await getAttachmentFolderPath(app, join(dirname(oldPath), 'DUMMY_FILE.md'));\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 (oldAttachmentFolderPath === dummyOldAttachmentFolderPath) {\n    const oldCache = await getCacheSafe(app, oldPath);\n    if (!oldCache) {\n      return;\n    }\n    for (const oldLink of getAllLinks(oldCache)) {\n      const oldAttachmentFile = extractLinkFile(app, oldLink, oldPath);\n      if (!oldAttachmentFile) {\n        continue;\n      }\n\n      if (oldAttachmentFile.path.startsWith(oldAttachmentFolderPath)) {\n        const oldAttachmentBacklinks = await getBacklinksForFileSafe(app, oldAttachmentFile);\n        if (oldAttachmentBacklinks.keys().length === 1) {\n          oldAttachmentFiles.push(oldAttachmentFile);\n        }\n      }\n    }\n  } else {\n    Vault.recurseChildren(oldAttachmentFolder, (oldAttachmentFile) => {\n      if (oldAttachmentFile instanceof TFile) {\n        oldAttachmentFiles.push(oldAttachmentFile);\n      }\n    });\n  }\n\n  const oldBasename = basename(oldPath, extname(oldPath));\n  const newBasename = basename(newPath, extname(newPath));\n\n  for (const oldAttachmentFile of oldAttachmentFiles) {\n    if (isNote(oldAttachmentFile)) {\n      continue;\n    }\n    const relativePath = relative(oldAttachmentFolderPath, oldAttachmentFile.path);\n    const newDir = join(newAttachmentFolderPath, dirname(relativePath));\n    const newChildBasename = settings.shouldRenameAttachmentFiles\n      ? oldAttachmentFile.basename.replaceAll(oldBasename, newBasename)\n      : oldAttachmentFile.basename;\n    let newChildPath = join(newDir, makeFileName(newChildBasename, oldAttachmentFile.extension));\n\n    if (oldAttachmentFile.path === newChildPath) {\n      continue;\n    }\n\n    if (settings.shouldDeleteConflictingAttachments) {\n      const newChildFile = getFileOrNull(app, newChildPath);\n      if (newChildFile) {\n        await app.fileManager.trashFile(newChildFile);\n      }\n    } else {\n      newChildPath = app.vault.getAvailablePath(join(newDir, newChildBasename), oldAttachmentFile.extension);\n    }\n    renameMap.set(oldAttachmentFile.path, newChildPath);\n  }\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  for (const settingsBuilder of settingsBuilders) {\n    const newSettings = settingsBuilder();\n    for (const [key, value] of Object.entries(newSettings) as [keyof RenameDeleteHandlerSettings, boolean][]) {\n      settings[key] ||= value;\n    }\n  }\n\n  return settings;\n}\n\nfunction handleMetadataDeleted(file: TAbstractFile, prevCache: CachedMetadata | null): void {\n  if (isMarkdownFile(file) && prevCache) {\n    deletedMetadataCacheMap.set(file.path, prevCache);\n  }\n}\n\nfunction makeKey(oldPath: string, newPath: string): string {\n  return `${oldPath} -> ${newPath}`;\n}\n\nasync function renameHandled(app: App, oldPath: string, newPath: string): Promise<string> {\n  newPath = await renameSafe(app, getFile(app, oldPath), newPath);\n  const key = makeKey(oldPath, newPath);\n  handledRenames.add(key);\n  return newPath;\n}\n"],
  "mappings": "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",
  "names": ["key", "newPath"]
}

|