obsidian-dev-utils 28.4.0 → 28.6.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 +9 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/FileChange.cjs +133 -45
- package/dist/lib/cjs/obsidian/FileChange.d.cts +31 -60
- package/dist/lib/cjs/obsidian/FrontmatterLinkCacheWithOffsets.cjs +43 -0
- package/dist/lib/cjs/obsidian/FrontmatterLinkCacheWithOffsets.d.cts +30 -0
- package/dist/lib/cjs/obsidian/Link.cjs +92 -66
- package/dist/lib/cjs/obsidian/Link.d.cts +19 -0
- package/dist/lib/cjs/obsidian/MetadataCache.cjs +8 -1
- package/dist/lib/cjs/obsidian/Reference.cjs +6 -38
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/FileChange.d.mts +31 -60
- package/dist/lib/esm/obsidian/FileChange.mjs +138 -45
- package/dist/lib/esm/obsidian/FrontmatterLinkCacheWithOffsets.d.mts +30 -0
- package/dist/lib/esm/obsidian/FrontmatterLinkCacheWithOffsets.mjs +19 -0
- package/dist/lib/esm/obsidian/Link.d.mts +19 -0
- package/dist/lib/esm/obsidian/Link.mjs +92 -68
- package/dist/lib/esm/obsidian/MetadataCache.mjs +8 -1
- package/dist/lib/esm/obsidian/Reference.mjs +6 -38
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/obsidian/FrontmatterLinkCacheWithOffsets/package.json +6 -0
- package/package.json +7 -6
|
@@ -5,6 +5,10 @@ if you want to view the source, please visit the github repository of this plugi
|
|
|
5
5
|
|
|
6
6
|
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
|
|
7
7
|
|
|
8
|
+
import {
|
|
9
|
+
isFrontmatterLinkCache,
|
|
10
|
+
isReferenceCache
|
|
11
|
+
} from "obsidian-typings/implementations";
|
|
8
12
|
import {
|
|
9
13
|
deepEqual,
|
|
10
14
|
getNestedPropertyValue,
|
|
@@ -19,7 +23,11 @@ import {
|
|
|
19
23
|
parseFrontmatter,
|
|
20
24
|
setFrontmatter
|
|
21
25
|
} from "./Frontmatter.mjs";
|
|
22
|
-
import {
|
|
26
|
+
import { isFrontmatterLinkCacheWithOffsets } from "./FrontmatterLinkCacheWithOffsets.mjs";
|
|
27
|
+
import {
|
|
28
|
+
isCanvasReference,
|
|
29
|
+
referenceToFileChange
|
|
30
|
+
} from "./Reference.mjs";
|
|
23
31
|
import { process } from "./Vault.mjs";
|
|
24
32
|
async function applyContentChanges(content, path, changesProvider) {
|
|
25
33
|
let changes = await resolveValue(changesProvider);
|
|
@@ -31,38 +39,18 @@ async function applyContentChanges(content, path, changesProvider) {
|
|
|
31
39
|
console.error(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));
|
|
32
40
|
hasFrontmatterError = true;
|
|
33
41
|
}
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
const actualContent = content.slice(change.startIndex, change.endIndex);
|
|
37
|
-
if (actualContent !== change.oldContent) {
|
|
38
|
-
console.warn("Content mismatch", {
|
|
39
|
-
actualContent,
|
|
40
|
-
endIndex: change.endIndex,
|
|
41
|
-
expectedContent: change.oldContent,
|
|
42
|
-
path,
|
|
43
|
-
startIndex: change.startIndex
|
|
44
|
-
});
|
|
45
|
-
return null;
|
|
46
|
-
}
|
|
47
|
-
} else if (isFrontmatterChange(change)) {
|
|
48
|
-
const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);
|
|
49
|
-
if (actualContent !== change.oldContent) {
|
|
50
|
-
console.warn("Content mismatch", {
|
|
51
|
-
actualContent,
|
|
52
|
-
expectedContent: change.oldContent,
|
|
53
|
-
frontmatterKey: change.frontmatterKey,
|
|
54
|
-
path
|
|
55
|
-
});
|
|
56
|
-
return null;
|
|
57
|
-
}
|
|
58
|
-
}
|
|
42
|
+
if (!validateChanges(changes, content, frontmatter, path)) {
|
|
43
|
+
return null;
|
|
59
44
|
}
|
|
60
45
|
changes.sort((a, b) => {
|
|
61
46
|
if (isContentChange(a) && isContentChange(b)) {
|
|
62
|
-
return a.
|
|
47
|
+
return a.reference.position.start.offset - b.reference.position.start.offset;
|
|
48
|
+
}
|
|
49
|
+
if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) {
|
|
50
|
+
return a.reference.cleanKey.localeCompare(b.reference.cleanKey) || a.reference.startOffset - b.reference.startOffset;
|
|
63
51
|
}
|
|
64
52
|
if (isFrontmatterChange(a) && isFrontmatterChange(b)) {
|
|
65
|
-
return a.
|
|
53
|
+
return a.reference.key.localeCompare(b.reference.key);
|
|
66
54
|
}
|
|
67
55
|
return isContentChange(a) ? -1 : 1;
|
|
68
56
|
});
|
|
@@ -84,7 +72,7 @@ async function applyContentChanges(content, path, changesProvider) {
|
|
|
84
72
|
if (!previousChange) {
|
|
85
73
|
continue;
|
|
86
74
|
}
|
|
87
|
-
if (isContentChange(previousChange) && isContentChange(change) && previousChange.
|
|
75
|
+
if (isContentChange(previousChange) && isContentChange(change) && previousChange.reference.position.end.offset && previousChange.reference.position.end.offset > change.reference.position.start.offset) {
|
|
88
76
|
console.warn("Overlapping changes", {
|
|
89
77
|
change,
|
|
90
78
|
previousChange
|
|
@@ -95,22 +83,38 @@ async function applyContentChanges(content, path, changesProvider) {
|
|
|
95
83
|
let newContent = "";
|
|
96
84
|
let lastIndex = 0;
|
|
97
85
|
let frontmatterChanged = false;
|
|
86
|
+
const frontmatterChangesWithOffsetMap = /* @__PURE__ */ new Map();
|
|
98
87
|
for (const change of changes) {
|
|
99
88
|
if (isContentChange(change)) {
|
|
100
|
-
newContent += content.slice(lastIndex, change.
|
|
89
|
+
newContent += content.slice(lastIndex, change.reference.position.start.offset);
|
|
101
90
|
newContent += change.newContent;
|
|
102
|
-
lastIndex = change.
|
|
91
|
+
lastIndex = change.reference.position.end.offset;
|
|
92
|
+
} else if (isFrontmatterChangeWithOffsets(change)) {
|
|
93
|
+
if (hasFrontmatterError) {
|
|
94
|
+
console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {
|
|
95
|
+
change
|
|
96
|
+
});
|
|
97
|
+
} else {
|
|
98
|
+
let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.cleanKey);
|
|
99
|
+
if (!frontmatterChangesWithOffsets) {
|
|
100
|
+
frontmatterChangesWithOffsets = [];
|
|
101
|
+
frontmatterChangesWithOffsetMap.set(change.reference.cleanKey, frontmatterChangesWithOffsets);
|
|
102
|
+
}
|
|
103
|
+
frontmatterChangesWithOffsets.push(change);
|
|
104
|
+
frontmatterChanged = true;
|
|
105
|
+
}
|
|
103
106
|
} else if (isFrontmatterChange(change)) {
|
|
104
107
|
if (hasFrontmatterError) {
|
|
105
108
|
console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {
|
|
106
109
|
change
|
|
107
110
|
});
|
|
108
111
|
} else {
|
|
109
|
-
setNestedPropertyValue(frontmatter, change.
|
|
112
|
+
setNestedPropertyValue(frontmatter, change.reference.key, change.newContent);
|
|
110
113
|
frontmatterChanged = true;
|
|
111
114
|
}
|
|
112
115
|
}
|
|
113
116
|
}
|
|
117
|
+
await applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path);
|
|
114
118
|
newContent += content.slice(lastIndex);
|
|
115
119
|
if (frontmatterChanged) {
|
|
116
120
|
newContent = setFrontmatter(newContent, frontmatter);
|
|
@@ -126,19 +130,22 @@ async function applyFileChanges(app, pathOrFile, changesProvider, processOptions
|
|
|
126
130
|
}, processOptions);
|
|
127
131
|
}
|
|
128
132
|
function isCanvasChange(change) {
|
|
129
|
-
return
|
|
133
|
+
return isCanvasReference(change.reference);
|
|
130
134
|
}
|
|
131
135
|
function isCanvasFileNodeChange(change) {
|
|
132
|
-
return isCanvasChange(change) && change.type === "file";
|
|
136
|
+
return isCanvasChange(change) && change.reference.type === "file";
|
|
133
137
|
}
|
|
134
138
|
function isCanvasTextNodeChange(change) {
|
|
135
|
-
return isCanvasChange(change) && change.type === "text";
|
|
139
|
+
return isCanvasChange(change) && change.reference.type === "text";
|
|
136
140
|
}
|
|
137
141
|
function isContentChange(fileChange) {
|
|
138
|
-
return fileChange.
|
|
142
|
+
return isReferenceCache(fileChange.reference);
|
|
139
143
|
}
|
|
140
144
|
function isFrontmatterChange(fileChange) {
|
|
141
|
-
return fileChange.
|
|
145
|
+
return isFrontmatterLinkCache(fileChange.reference);
|
|
146
|
+
}
|
|
147
|
+
function isFrontmatterChangeWithOffsets(fileChange) {
|
|
148
|
+
return isFrontmatterLinkCacheWithOffsets(fileChange.reference);
|
|
142
149
|
}
|
|
143
150
|
async function applyCanvasChanges(content, path, changesProvider) {
|
|
144
151
|
const changes = await resolveValue(changesProvider);
|
|
@@ -152,10 +159,10 @@ async function applyCanvasChanges(content, path, changesProvider) {
|
|
|
152
159
|
});
|
|
153
160
|
return null;
|
|
154
161
|
}
|
|
155
|
-
const node = canvasData.nodes[change.nodeIndex];
|
|
162
|
+
const node = canvasData.nodes[change.reference.nodeIndex];
|
|
156
163
|
if (!node) {
|
|
157
164
|
console.warn("Node not found", {
|
|
158
|
-
nodeIndex: change.nodeIndex,
|
|
165
|
+
nodeIndex: change.reference.nodeIndex,
|
|
159
166
|
path
|
|
160
167
|
});
|
|
161
168
|
return null;
|
|
@@ -165,7 +172,7 @@ async function applyCanvasChanges(content, path, changesProvider) {
|
|
|
165
172
|
console.warn("Content mismatch", {
|
|
166
173
|
actualContent: node.file,
|
|
167
174
|
expectedContent: change.oldContent,
|
|
168
|
-
nodeIndex: change.nodeIndex,
|
|
175
|
+
nodeIndex: change.reference.nodeIndex,
|
|
169
176
|
path,
|
|
170
177
|
type: "file"
|
|
171
178
|
});
|
|
@@ -173,10 +180,10 @@ async function applyCanvasChanges(content, path, changesProvider) {
|
|
|
173
180
|
}
|
|
174
181
|
node.file = change.newContent;
|
|
175
182
|
} else if (isCanvasTextNodeChange(change)) {
|
|
176
|
-
let canvasTextChangesForNode = canvasTextChanges.get(change.nodeIndex);
|
|
183
|
+
let canvasTextChangesForNode = canvasTextChanges.get(change.reference.nodeIndex);
|
|
177
184
|
if (!canvasTextChangesForNode) {
|
|
178
185
|
canvasTextChangesForNode = [];
|
|
179
|
-
canvasTextChanges.set(change.nodeIndex, canvasTextChangesForNode);
|
|
186
|
+
canvasTextChanges.set(change.reference.nodeIndex, canvasTextChangesForNode);
|
|
180
187
|
}
|
|
181
188
|
canvasTextChangesForNode.push(change);
|
|
182
189
|
}
|
|
@@ -197,11 +204,44 @@ async function applyCanvasChanges(content, path, changesProvider) {
|
|
|
197
204
|
});
|
|
198
205
|
return null;
|
|
199
206
|
}
|
|
200
|
-
const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.originalReference, change.newContent));
|
|
207
|
+
const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent));
|
|
201
208
|
node.text = await applyContentChanges(node.text, `${path}.node${nodeIndex.toString()}.VIRTUAL_FILE.md`, contentChanges);
|
|
202
209
|
}
|
|
203
210
|
return JSON.stringify(canvasData, null, " ");
|
|
204
211
|
}
|
|
212
|
+
async function applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path) {
|
|
213
|
+
for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {
|
|
214
|
+
const propertyValue = getNestedPropertyValue(frontmatter, key);
|
|
215
|
+
if (typeof propertyValue !== "string") {
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
const contentChanges = frontmatterChangesWithOffsets.map((change) => ({
|
|
219
|
+
newContent: change.newContent,
|
|
220
|
+
oldContent: change.oldContent,
|
|
221
|
+
reference: {
|
|
222
|
+
link: "",
|
|
223
|
+
original: "",
|
|
224
|
+
position: {
|
|
225
|
+
end: {
|
|
226
|
+
col: change.reference.endOffset,
|
|
227
|
+
line: 0,
|
|
228
|
+
offset: change.reference.endOffset
|
|
229
|
+
},
|
|
230
|
+
start: {
|
|
231
|
+
col: change.reference.startOffset,
|
|
232
|
+
line: 0,
|
|
233
|
+
offset: change.reference.startOffset
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
}));
|
|
238
|
+
const newPropertyValue = await applyContentChanges(propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);
|
|
239
|
+
if (newPropertyValue === null) {
|
|
240
|
+
return;
|
|
241
|
+
}
|
|
242
|
+
setNestedPropertyValue(frontmatter, key, newPropertyValue);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
205
245
|
function parseJsonSafe(content) {
|
|
206
246
|
let parsed;
|
|
207
247
|
try {
|
|
@@ -214,6 +254,58 @@ function parseJsonSafe(content) {
|
|
|
214
254
|
}
|
|
215
255
|
return parsed;
|
|
216
256
|
}
|
|
257
|
+
function validateChanges(changes, content, frontmatter, path) {
|
|
258
|
+
for (const change of changes) {
|
|
259
|
+
if (isContentChange(change)) {
|
|
260
|
+
const startOffset = change.reference.position.start.offset;
|
|
261
|
+
const endOffset = change.reference.position.end.offset;
|
|
262
|
+
const actualContent = content.slice(startOffset, endOffset);
|
|
263
|
+
if (actualContent !== change.oldContent) {
|
|
264
|
+
console.warn("Content mismatch", {
|
|
265
|
+
actualContent,
|
|
266
|
+
endOffset,
|
|
267
|
+
expectedContent: change.oldContent,
|
|
268
|
+
path,
|
|
269
|
+
startOffset
|
|
270
|
+
});
|
|
271
|
+
return false;
|
|
272
|
+
}
|
|
273
|
+
} else if (isFrontmatterChangeWithOffsets(change)) {
|
|
274
|
+
const propertyValue = getNestedPropertyValue(frontmatter, change.reference.cleanKey);
|
|
275
|
+
if (typeof propertyValue !== "string") {
|
|
276
|
+
console.warn("Property value is not a string", {
|
|
277
|
+
frontmatterKey: change.reference.cleanKey,
|
|
278
|
+
path,
|
|
279
|
+
propertyValue
|
|
280
|
+
});
|
|
281
|
+
return false;
|
|
282
|
+
}
|
|
283
|
+
const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset);
|
|
284
|
+
if (actualContent !== change.oldContent) {
|
|
285
|
+
console.warn("Content mismatch", {
|
|
286
|
+
actualContent,
|
|
287
|
+
expectedContent: change.oldContent,
|
|
288
|
+
frontmatterKey: change.reference.cleanKey,
|
|
289
|
+
path,
|
|
290
|
+
startOffset: change.reference.startOffset
|
|
291
|
+
});
|
|
292
|
+
return false;
|
|
293
|
+
}
|
|
294
|
+
} else if (isFrontmatterChange(change)) {
|
|
295
|
+
const actualContent = getNestedPropertyValue(frontmatter, change.reference.key);
|
|
296
|
+
if (actualContent !== change.oldContent) {
|
|
297
|
+
console.warn("Content mismatch", {
|
|
298
|
+
actualContent,
|
|
299
|
+
expectedContent: change.oldContent,
|
|
300
|
+
frontmatterKey: change.reference.key,
|
|
301
|
+
path
|
|
302
|
+
});
|
|
303
|
+
return false;
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
return true;
|
|
308
|
+
}
|
|
217
309
|
export {
|
|
218
310
|
applyContentChanges,
|
|
219
311
|
applyFileChanges,
|
|
@@ -221,6 +313,7 @@ export {
|
|
|
221
313
|
isCanvasFileNodeChange,
|
|
222
314
|
isCanvasTextNodeChange,
|
|
223
315
|
isContentChange,
|
|
224
|
-
isFrontmatterChange
|
|
316
|
+
isFrontmatterChange,
|
|
317
|
+
isFrontmatterChangeWithOffsets
|
|
225
318
|
};
|
|
226
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  Reference\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport type { GenericObject } from '../Object.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../Object.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getPath,\n  isCanvasFile\n} from './FileSystem.ts';\nimport {\n  parseFrontmatter,\n  setFrontmatter\n} from './Frontmatter.ts';\nimport { referenceToFileChange } from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * Represents a canvas change in the Vault.\n */\nexport interface CanvasChange extends FileChange {\n  /**\n   * Whether the change is a canvas change.\n   */\n  isCanvas: true;\n\n  /**\n   * The index of the node in the canvas.\n   */\n  nodeIndex: number;\n\n  /**\n   * The type of link.\n   */\n  type: 'file' | 'text';\n}\n\n/**\n * Represents a change in a file node in a canvas.\n */\nexport interface CanvasFileNodeChange extends CanvasChange {\n  /**\n   * The type of link.\n   */\n  type: 'file';\n}\n\n/**\n * Represents a change in a text node in a canvas.\n */\nexport interface CanvasTextNodeChange extends CanvasChange {\n  /**\n   * The original reference.\n   */\n  originalReference: Reference;\n\n  /**\n   * The type of link.\n   */\n  type: 'text';\n}\n\n/**\n * Represents a content body change in the Vault.\n */\nexport interface ContentChange extends FileChange {\n  /**\n   * The end index of the change in the file content.\n   */\n  endIndex: number;\n\n  /**\n   * The start index of the change in the file content.\n   */\n  startIndex: number;\n}\n\n/**\n * Represents a file change in the Vault.\n */\nexport interface FileChange {\n  /**\n   * The new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * The old content that will be replaced.\n   */\n  oldContent: string;\n}\n\n/**\n * Represents a frontmatter change in the Vault.\n */\nexport interface FrontmatterChange extends FileChange {\n  /**\n   * The key in the frontmatter to use for the link.\n   */\n  frontmatterKey: string;\n}\n\n/**\n * Applies a series of content changes to the specified content.\n *\n * @param content - The content to which the changes should be applied.\n * @param path - The path to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of content changes to apply.\n * @returns A {@link Promise} that resolves to the updated content or to `null` if update didn't succeed.\n */\nexport async function applyContentChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  let changes = await resolveValue(changesProvider);\n  let frontmatter: CombinedFrontmatter<unknown> = {};\n  let hasFrontmatterError = false;\n  try {\n    frontmatter = parseFrontmatter(content);\n  } catch (error) {\n    console.error(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));\n    hasFrontmatterError = true;\n  }\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const actualContent = content.slice(change.startIndex, change.endIndex);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          endIndex: change.endIndex,\n          expectedContent: change.oldContent,\n          path,\n          startIndex: change.startIndex\n        });\n\n        return null;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.frontmatterKey,\n          path\n        });\n\n        return null;\n      }\n    }\n  }\n\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.startIndex - b.startIndex;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.frontmatterKey.localeCompare(b.frontmatterKey);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  changes = changes.filter((change, index) => {\n    if (change.oldContent === change.newContent) {\n      return false;\n    }\n    if (index === 0) {\n      return true;\n    }\n    return !deepEqual(change, changes[index - 1]);\n  });\n\n  for (let i = 1; i < changes.length; i++) {\n    const change = changes[i];\n    if (!change) {\n      continue;\n    }\n    const previousChange = changes[i - 1];\n    if (!previousChange) {\n      continue;\n    }\n\n    if (\n      isContentChange(previousChange) && isContentChange(change) && previousChange.endIndex && change.startIndex\n      && previousChange.endIndex > change.startIndex\n    ) {\n      console.warn('Overlapping changes', {\n        change,\n        previousChange\n      });\n      return null;\n    }\n  }\n\n  let newContent = '';\n  let lastIndex = 0;\n  let frontmatterChanged = false;\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.startIndex);\n      newContent += change.newContent;\n      lastIndex = change.endIndex;\n    } else if (isFrontmatterChange(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {\n          change\n        });\n      } else {\n        setNestedPropertyValue(frontmatter, change.frontmatterKey, change.newContent);\n        frontmatterChanged = true;\n      }\n    }\n  }\n\n  newContent += content.slice(lastIndex);\n  if (frontmatterChanged) {\n    newContent = setFrontmatter(newContent, frontmatter);\n  }\n\n  return newContent;\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param processOptions - Optional options for processing/retrying the operation.\n *\n * @returns A {@link Promise} that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(\n  app: App,\n  pathOrFile: PathOrFile,\n  changesProvider: ValueProvider<FileChange[]>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await process(app, pathOrFile, async (content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return applyCanvasChanges(content, getPath(app, pathOrFile), changesProvider);\n    }\n\n    return await applyContentChanges(content, getPath(app, pathOrFile), changesProvider);\n  }, processOptions);\n}\n\n/**\n * Checks if a file change is a canvas change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas change.\n */\nexport function isCanvasChange(change: FileChange): change is CanvasChange {\n  return !!(change as Partial<CanvasChange>).isCanvas;\n}\n\n/**\n * Checks if a file change is a canvas file node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas file node change.\n */\nexport function isCanvasFileNodeChange(change: FileChange): change is CanvasFileNodeChange {\n  return isCanvasChange(change) && change.type === 'file';\n}\n\n/**\n * Checks if a file change is a canvas text node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas text node change.\n */\nexport function isCanvasTextNodeChange(change: FileChange): change is CanvasTextNodeChange {\n  return isCanvasChange(change) && change.type === 'text';\n}\n\n/**\n * Checks if a file change is a content change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a content change.\n */\nexport function isContentChange(fileChange: FileChange): fileChange is ContentChange {\n  return (fileChange as Partial<ContentChange>).startIndex !== undefined;\n}\n\n/**\n * Checks if a file change is a frontmatter change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change.\n */\nexport function isFrontmatterChange(fileChange: FileChange): fileChange is FrontmatterChange {\n  return (fileChange as Partial<FrontmatterChange>).frontmatterKey !== undefined;\n}\n\nasync function applyCanvasChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  const changes = await resolveValue(changesProvider);\n  const canvasData = parseJsonSafe(content) as CanvasData;\n\n  const canvasTextChanges = new Map<number, CanvasTextNodeChange[]>();\n\n  for (const change of changes) {\n    if (!isCanvasChange(change)) {\n      console.warn('Only canvas changes are supported for canvas files', {\n        change,\n        path\n      });\n      return null;\n    }\n\n    const node = canvasData.nodes[change.nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex: change.nodeIndex,\n        path\n      });\n      return null;\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.nodeIndex,\n          path,\n          type: 'file'\n        });\n\n        return null;\n      }\n      node.file = change.newContent;\n    } else if (isCanvasTextNodeChange(change)) {\n      let canvasTextChangesForNode = canvasTextChanges.get(change.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.nodeIndex, canvasTextChangesForNode);\n      }\n\n      canvasTextChangesForNode.push(change);\n    }\n  }\n\n  for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {\n    const node = canvasData.nodes[nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    if (typeof node.text !== 'string') {\n      console.warn('Node text is not a string', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.originalReference, change.newContent));\n    node.text = await applyContentChanges(node.text, `${path}.node${nodeIndex.toString()}.VIRTUAL_FILE.md`, contentChanges);\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nfunction parseJsonSafe(content: string): GenericObject {\n  let parsed: unknown;\n  try {\n    parsed = JSON.parse(content);\n  } catch {\n    parsed = null;\n  }\n\n  if (parsed === null || typeof parsed !== 'object') {\n    parsed = {};\n  }\n\n  return parsed as GenericObject;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
319
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  FrontmatterLinkCache,\n  Reference,\n  ReferenceCache\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport {\n  isFrontmatterLinkCache,\n  isReferenceCache\n} from 'obsidian-typings/implementations';\n\nimport type { GenericObject } from '../Object.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\nimport type { FrontmatterLinkCacheWithOffsets } from './FrontmatterLinkCacheWithOffsets.ts';\nimport type {\n  CanvasFileNodeReference,\n  CanvasReference,\n  CanvasTextNodeReference\n} from './Reference.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../Object.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getPath,\n  isCanvasFile\n} from './FileSystem.ts';\nimport {\n  parseFrontmatter,\n  setFrontmatter\n} from './Frontmatter.ts';\nimport { isFrontmatterLinkCacheWithOffsets } from './FrontmatterLinkCacheWithOffsets.ts';\nimport {\n  isCanvasReference,\n  referenceToFileChange\n} from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * Represents a file change in the Vault.\n */\nexport interface FileChange {\n  /**\n   * The new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * The old content that will be replaced.\n   */\n  oldContent: string;\n\n  /**\n   * The reference that caused the change.\n   */\n  reference: Reference;\n}\ntype CanvasChange = { reference: CanvasReference } & FileChange;\ntype CanvasFileNodeChange = { reference: CanvasFileNodeReference } & FileChange;\ntype CanvasTextNodeChange = { reference: CanvasTextNodeReference } & FileChange;\ntype ContentChange = { reference: ReferenceCache } & FileChange;\ntype FrontmatterChange = { reference: FrontmatterLinkCache } & FileChange;\ntype FrontmatterChangeWithOffsets = { reference: FrontmatterLinkCacheWithOffsets } & FileChange;\n\n/**\n * Applies a series of content changes to the specified content.\n *\n * @param content - The content to which the changes should be applied.\n * @param path - The path to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of content changes to apply.\n * @returns A {@link Promise} that resolves to the updated content or to `null` if update didn't succeed.\n */\nexport async function applyContentChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  let changes = await resolveValue(changesProvider);\n  let frontmatter: CombinedFrontmatter<unknown> = {};\n  let hasFrontmatterError = false;\n  try {\n    frontmatter = parseFrontmatter(content);\n  } catch (error) {\n    console.error(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));\n    hasFrontmatterError = true;\n  }\n\n  if (!validateChanges(changes, content, frontmatter, path)) {\n    return null;\n  }\n\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.reference.position.start.offset - b.reference.position.start.offset;\n    }\n\n    if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) {\n      return a.reference.cleanKey.localeCompare(b.reference.cleanKey) || a.reference.startOffset - b.reference.startOffset;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.reference.key.localeCompare(b.reference.key);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  changes = changes.filter((change, index) => {\n    if (change.oldContent === change.newContent) {\n      return false;\n    }\n    if (index === 0) {\n      return true;\n    }\n    return !deepEqual(change, changes[index - 1]);\n  });\n\n  for (let i = 1; i < changes.length; i++) {\n    const change = changes[i];\n    if (!change) {\n      continue;\n    }\n    const previousChange = changes[i - 1];\n    if (!previousChange) {\n      continue;\n    }\n\n    if (\n      isContentChange(previousChange) && isContentChange(change) && previousChange.reference.position.end.offset\n      && previousChange.reference.position.end.offset > change.reference.position.start.offset\n    ) {\n      console.warn('Overlapping changes', {\n        change,\n        previousChange\n      });\n      return null;\n    }\n  }\n\n  let newContent = '';\n  let lastIndex = 0;\n  let frontmatterChanged = false;\n\n  const frontmatterChangesWithOffsetMap = new Map<string, FrontmatterChangeWithOffsets[]>();\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.reference.position.start.offset);\n      newContent += change.newContent;\n      lastIndex = change.reference.position.end.offset;\n    } else if (isFrontmatterChangeWithOffsets(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {\n          change\n        });\n      } else {\n        let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.cleanKey);\n        if (!frontmatterChangesWithOffsets) {\n          frontmatterChangesWithOffsets = [];\n          frontmatterChangesWithOffsetMap.set(change.reference.cleanKey, frontmatterChangesWithOffsets);\n        }\n        frontmatterChangesWithOffsets.push(change);\n        frontmatterChanged = true;\n      }\n    } else if (isFrontmatterChange(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {\n          change\n        });\n      } else {\n        setNestedPropertyValue(frontmatter, change.reference.key, change.newContent);\n        frontmatterChanged = true;\n      }\n    }\n  }\n\n  await applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path);\n\n  newContent += content.slice(lastIndex);\n  if (frontmatterChanged) {\n    newContent = setFrontmatter(newContent, frontmatter);\n  }\n\n  return newContent;\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param processOptions - Optional options for processing/retrying the operation.\n *\n * @returns A {@link Promise} that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(\n  app: App,\n  pathOrFile: PathOrFile,\n  changesProvider: ValueProvider<FileChange[]>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await process(app, pathOrFile, async (content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return applyCanvasChanges(content, getPath(app, pathOrFile), changesProvider);\n    }\n\n    return await applyContentChanges(content, getPath(app, pathOrFile), changesProvider);\n  }, processOptions);\n}\n\n/**\n * Checks if a file change is a canvas change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas change.\n */\nexport function isCanvasChange(change: FileChange): change is CanvasChange {\n  return isCanvasReference(change.reference);\n}\n\n/**\n * Checks if a file change is a canvas file node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas file node change.\n */\nexport function isCanvasFileNodeChange(change: FileChange): change is CanvasFileNodeChange {\n  return isCanvasChange(change) && change.reference.type === 'file';\n}\n\n/**\n * Checks if a file change is a canvas text node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas text node change.\n */\nexport function isCanvasTextNodeChange(change: FileChange): change is CanvasTextNodeChange {\n  return isCanvasChange(change) && change.reference.type === 'text';\n}\n\n/**\n * Checks if a file change is a content change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a content change.\n */\nexport function isContentChange(fileChange: FileChange): fileChange is ContentChange {\n  return isReferenceCache(fileChange.reference);\n}\n\n/**\n * Checks if a file change is a frontmatter change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change.\n */\nexport function isFrontmatterChange(fileChange: FileChange): fileChange is FrontmatterChange {\n  return isFrontmatterLinkCache(fileChange.reference);\n}\n\n/**\n * Checks if a file change is a frontmatter change with offsets.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change with offsets.\n */\nexport function isFrontmatterChangeWithOffsets(fileChange: FileChange): fileChange is FrontmatterChangeWithOffsets {\n  return isFrontmatterLinkCacheWithOffsets(fileChange.reference);\n}\n\nasync function applyCanvasChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  const changes = await resolveValue(changesProvider);\n  const canvasData = parseJsonSafe(content) as CanvasData;\n\n  const canvasTextChanges = new Map<number, CanvasTextNodeChange[]>();\n\n  for (const change of changes) {\n    if (!isCanvasChange(change)) {\n      console.warn('Only canvas changes are supported for canvas files', {\n        change,\n        path\n      });\n      return null;\n    }\n\n    const node = canvasData.nodes[change.reference.nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex: change.reference.nodeIndex,\n        path\n      });\n      return null;\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.reference.nodeIndex,\n          path,\n          type: 'file'\n        });\n\n        return null;\n      }\n      node.file = change.newContent;\n    } else if (isCanvasTextNodeChange(change)) {\n      let canvasTextChangesForNode = canvasTextChanges.get(change.reference.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.reference.nodeIndex, canvasTextChangesForNode);\n      }\n\n      canvasTextChangesForNode.push(change);\n    }\n  }\n\n  for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {\n    const node = canvasData.nodes[nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    if (typeof node.text !== 'string') {\n      console.warn('Node text is not a string', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent));\n    node.text = await applyContentChanges(node.text, `${path}.node${nodeIndex.toString()}.VIRTUAL_FILE.md`, contentChanges);\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nasync function applyFrontmatterChangesWithOffsets(\n  frontmatter: CombinedFrontmatter<unknown>,\n  frontmatterChangesWithOffsetMap: Map<string, FrontmatterChangeWithOffsets[]>,\n  path: string\n): Promise<void> {\n  for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {\n    const propertyValue = getNestedPropertyValue(frontmatter, key);\n    if (typeof propertyValue !== 'string') {\n      return;\n    }\n\n    const contentChanges: ContentChange[] = frontmatterChangesWithOffsets.map((change) => ({\n      newContent: change.newContent,\n      oldContent: change.oldContent,\n      reference: {\n        link: '',\n        original: '',\n        position: {\n          end: {\n            col: change.reference.endOffset,\n            line: 0,\n            offset: change.reference.endOffset\n          },\n          start: {\n            col: change.reference.startOffset,\n            line: 0,\n            offset: change.reference.startOffset\n          }\n        }\n      }\n    } as ContentChange));\n\n    const newPropertyValue = await applyContentChanges(propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);\n    if (newPropertyValue === null) {\n      return;\n    }\n\n    setNestedPropertyValue(frontmatter, key, newPropertyValue);\n  }\n}\n\nfunction parseJsonSafe(content: string): GenericObject {\n  let parsed: unknown;\n  try {\n    parsed = JSON.parse(content);\n  } catch {\n    parsed = null;\n  }\n\n  if (parsed === null || typeof parsed !== 'object') {\n    parsed = {};\n  }\n\n  return parsed as GenericObject;\n}\n\nfunction validateChanges(changes: FileChange[], content: string, frontmatter: CombinedFrontmatter<unknown>, path: string): boolean {\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const startOffset = change.reference.position.start.offset;\n      const endOffset = change.reference.position.end.offset;\n      const actualContent = content.slice(startOffset, endOffset);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          endOffset,\n          expectedContent: change.oldContent,\n          path,\n          startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChangeWithOffsets(change)) {\n      const propertyValue = getNestedPropertyValue(frontmatter, change.reference.cleanKey);\n      if (typeof propertyValue !== 'string') {\n        console.warn('Property value is not a string', {\n          frontmatterKey: change.reference.cleanKey,\n          path,\n          propertyValue\n        });\n        return false;\n      }\n\n      const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.cleanKey,\n          path,\n          startOffset: change.reference.startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.reference.key);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.key,\n          path\n        });\n\n        return false;\n      }\n    }\n  }\n\n  return true;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @packageDocumentation
|
|
3
|
+
*
|
|
4
|
+
* This module provides a type for frontmatter link cache with offsets.
|
|
5
|
+
*/
|
|
6
|
+
import type { FrontmatterLinkCache, Reference } from 'obsidian';
|
|
7
|
+
/**
|
|
8
|
+
* Type for frontmatter link cache with offsets.
|
|
9
|
+
*/
|
|
10
|
+
export interface FrontmatterLinkCacheWithOffsets extends FrontmatterLinkCache {
|
|
11
|
+
/**
|
|
12
|
+
* The frontmatter key without offsets.
|
|
13
|
+
*/
|
|
14
|
+
cleanKey: string;
|
|
15
|
+
/**
|
|
16
|
+
* The end offset of the link in the property value.
|
|
17
|
+
*/
|
|
18
|
+
endOffset: number;
|
|
19
|
+
/**
|
|
20
|
+
* The start offset of the link in the property value.
|
|
21
|
+
*/
|
|
22
|
+
startOffset: number;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Checks if the reference is a frontmatter link cache with offsets.
|
|
26
|
+
*
|
|
27
|
+
* @param reference - The reference to check.
|
|
28
|
+
* @returns Whether the reference is a frontmatter link cache with offsets.
|
|
29
|
+
*/
|
|
30
|
+
export declare function isFrontmatterLinkCacheWithOffsets(reference: Reference): reference is FrontmatterLinkCacheWithOffsets;
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/*
|
|
2
|
+
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
|
3
|
+
if you want to view the source, please visit the github repository of this plugin
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
|
|
7
|
+
|
|
8
|
+
import { isFrontmatterLinkCache } from "obsidian-typings/implementations";
|
|
9
|
+
function isFrontmatterLinkCacheWithOffsets(reference) {
|
|
10
|
+
if (!isFrontmatterLinkCache(reference)) {
|
|
11
|
+
return false;
|
|
12
|
+
}
|
|
13
|
+
const frontmatterLinkCacheEx = reference;
|
|
14
|
+
return frontmatterLinkCacheEx.cleanKey !== void 0 && frontmatterLinkCacheEx.startOffset !== void 0 && frontmatterLinkCacheEx.endOffset !== void 0;
|
|
15
|
+
}
|
|
16
|
+
export {
|
|
17
|
+
isFrontmatterLinkCacheWithOffsets
|
|
18
|
+
};
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL0Zyb250bWF0dGVyTGlua0NhY2hlV2l0aE9mZnNldHMudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgYSB0eXBlIGZvciBmcm9udG1hdHRlciBsaW5rIGNhY2hlIHdpdGggb2Zmc2V0cy5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7XG4gIEZyb250bWF0dGVyTGlua0NhY2hlLFxuICBSZWZlcmVuY2Vcbn0gZnJvbSAnb2JzaWRpYW4nO1xuXG5pbXBvcnQgeyBpc0Zyb250bWF0dGVyTGlua0NhY2hlIH0gZnJvbSAnb2JzaWRpYW4tdHlwaW5ncy9pbXBsZW1lbnRhdGlvbnMnO1xuXG4vKipcbiAqIFR5cGUgZm9yIGZyb250bWF0dGVyIGxpbmsgY2FjaGUgd2l0aCBvZmZzZXRzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEZyb250bWF0dGVyTGlua0NhY2hlV2l0aE9mZnNldHMgZXh0ZW5kcyBGcm9udG1hdHRlckxpbmtDYWNoZSB7XG4gIC8qKlxuICAgKiBUaGUgZnJvbnRtYXR0ZXIga2V5IHdpdGhvdXQgb2Zmc2V0cy5cbiAgICovXG4gIGNsZWFuS2V5OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFRoZSBlbmQgb2Zmc2V0IG9mIHRoZSBsaW5rIGluIHRoZSBwcm9wZXJ0eSB2YWx1ZS5cbiAgICovXG4gIGVuZE9mZnNldDogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBUaGUgc3RhcnQgb2Zmc2V0IG9mIHRoZSBsaW5rIGluIHRoZSBwcm9wZXJ0eSB2YWx1ZS5cbiAgICovXG4gIHN0YXJ0T2Zmc2V0OiBudW1iZXI7XG59XG5cbi8qKlxuICogQ2hlY2tzIGlmIHRoZSByZWZlcmVuY2UgaXMgYSBmcm9udG1hdHRlciBsaW5rIGNhY2hlIHdpdGggb2Zmc2V0cy5cbiAqXG4gKiBAcGFyYW0gcmVmZXJlbmNlIC0gVGhlIHJlZmVyZW5jZSB0byBjaGVjay5cbiAqIEByZXR1cm5zIFdoZXRoZXIgdGhlIHJlZmVyZW5jZSBpcyBhIGZyb250bWF0dGVyIGxpbmsgY2FjaGUgd2l0aCBvZmZzZXRzLlxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNGcm9udG1hdHRlckxpbmtDYWNoZVdpdGhPZmZzZXRzKHJlZmVyZW5jZTogUmVmZXJlbmNlKTogcmVmZXJlbmNlIGlzIEZyb250bWF0dGVyTGlua0NhY2hlV2l0aE9mZnNldHMge1xuICBpZiAoIWlzRnJvbnRtYXR0ZXJMaW5rQ2FjaGUocmVmZXJlbmNlKSkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIGNvbnN0IGZyb250bWF0dGVyTGlua0NhY2hlRXggPSByZWZlcmVuY2UgYXMgUGFydGlhbDxGcm9udG1hdHRlckxpbmtDYWNoZVdpdGhPZmZzZXRzPjtcbiAgcmV0dXJuIGZyb250bWF0dGVyTGlua0NhY2hlRXguY2xlYW5LZXkgIT09IHVuZGVmaW5lZCAmJiBmcm9udG1hdHRlckxpbmtDYWNoZUV4LnN0YXJ0T2Zmc2V0ICE9PSB1bmRlZmluZWQgJiYgZnJvbnRtYXR0ZXJMaW5rQ2FjaGVFeC5lbmRPZmZzZXQgIT09IHVuZGVmaW5lZDtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFXQSxTQUFTLDhCQUE4QjtBQTRCaEMsU0FBUyxrQ0FBa0MsV0FBb0U7QUFDcEgsTUFBSSxDQUFDLHVCQUF1QixTQUFTLEdBQUc7QUFDdEMsV0FBTztBQUFBLEVBQ1Q7QUFFQSxRQUFNLHlCQUF5QjtBQUMvQixTQUFPLHVCQUF1QixhQUFhLFVBQWEsdUJBQXVCLGdCQUFnQixVQUFhLHVCQUF1QixjQUFjO0FBQ25KOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
@@ -122,6 +122,10 @@ export interface ParseLinkResult {
|
|
|
122
122
|
* The encoded URL of the link.
|
|
123
123
|
*/
|
|
124
124
|
encodedUrl?: string;
|
|
125
|
+
/**
|
|
126
|
+
* The end offset of the link in the original text.
|
|
127
|
+
*/
|
|
128
|
+
endOffset: number;
|
|
125
129
|
/**
|
|
126
130
|
* Indicates if the link has angle brackets.
|
|
127
131
|
*/
|
|
@@ -138,6 +142,14 @@ export interface ParseLinkResult {
|
|
|
138
142
|
* Indicates if the link is a wikilink.
|
|
139
143
|
*/
|
|
140
144
|
isWikilink: boolean;
|
|
145
|
+
/**
|
|
146
|
+
* The raw link text.
|
|
147
|
+
*/
|
|
148
|
+
raw: string;
|
|
149
|
+
/**
|
|
150
|
+
* The start offset of the link in the original text.
|
|
151
|
+
*/
|
|
152
|
+
startOffset: number;
|
|
141
153
|
/**
|
|
142
154
|
* The title of the link.
|
|
143
155
|
*/
|
|
@@ -366,6 +378,13 @@ export declare function generateMarkdownLink(options: GenerateMarkdownLinkOption
|
|
|
366
378
|
* @returns The parsed link.
|
|
367
379
|
*/
|
|
368
380
|
export declare function parseLink(str: string): null | ParseLinkResult;
|
|
381
|
+
/**
|
|
382
|
+
* Parses all links in a string.
|
|
383
|
+
*
|
|
384
|
+
* @param str - The string to parse the links in.
|
|
385
|
+
* @returns The parsed links.
|
|
386
|
+
*/
|
|
387
|
+
export declare function parseLinks(str: string): ParseLinkResult[];
|
|
369
388
|
/**
|
|
370
389
|
* Determines if the alias of a link should be reset.
|
|
371
390
|
*
|