发布文章时内容中的本地图片没有被成功上传
只要你文章中引用的本地图片没有放在Obsidian仓库的根目录下,即使在插件设置中打开了 替换媒体链接,图片也无法成功上传至博客中。
其原因是插件在取文件名称时,没有对路径进行剥离(也许插件作者所有图片都在根目录?又或者是作者自己压根不用Obsidian和自己的插件)
修复位置在 updatePostImages 方法中的上半部分。
以下为原始代码:
async updatePostImages(params) {
var _a;
const { postParams, auth } = params;
const activeFile = this.plugin.app.workspace.getActiveFile();
if (activeFile === null) {
throw new Error(this.plugin.i18n.t("error_noActiveFile"));
}
const { activeEditor } = this.plugin.app.workspace;
if (activeEditor && activeEditor.editor) {
const images = getImages(postParams.content);
for (const img of images) {
if (!img.srcIsUrl) {
const splitFile = img.src.split(".");
const ext = splitFile.pop();
const fileName = splitFile.join(".");
let filePath = await this.plugin.app.vault.getAvailablePathForAttachments(
fileName,
ext,
activeFile
);
const pathRegex = /(.*) \d+\.(.*)/;
filePath = filePath.replace(pathRegex, "$1.$2");
const imgFile = this.plugin.app.vault.getAbstractFileByPath(filePath);
以下为修复后的代码:
async updatePostImages(params) {
var _a;
const { postParams, auth } = params;
const activeFile = this.plugin.app.workspace.getActiveFile();
if (activeFile === null) {
throw new Error(this.plugin.i18n.t("error_noActiveFile"));
}
const { activeEditor } = this.plugin.app.workspace;
if (activeEditor && activeEditor.editor) {
const images = getImages(postParams.content);
for (const img of images) {
if (!img.srcIsUrl) {
const splitFile = img.src.split(".");
const ext = splitFile.pop();
// const fileName = splitFile.join(".");
// 这里如果不是在根路径,原本的fileName取出来还带着整个相对路径,因此需要取最后的真正filename
const fileName = splitFile.pop().split("/").pop();
let filePath = await this.plugin.app.vault.getAvailablePathForAttachments(
fileName,
ext,
activeFile
);
const pathRegex = /(.*) \d+\.(.*)/;
filePath = filePath.replace(pathRegex, "$1.$2");
const imgFile = this.plugin.app.vault.getAbstractFileByPath(filePath);
发布过的文章,修改后再次发布时又发布为新文章
曾发布过的文章,修改后再次发布时,文章头部的 文档属性 会被清除,导致缺失了 postId 后,被当做新文章直接发布到博客中,而不是更新原有的文章。
导致BUG的原因是插件在上传本地图片后,修改文章中的本地图片路径为网络链接后,使用 activeEditor.editor.setValue方法修改了编辑器中的内容,而这部分内容是不包含原始文章中头部的 文档属性 的,自此文章丢失了文档属性。而插件后续虽然会尝试再次写入文档属性数据,但由于Obsidian自身机制的原因,编辑器中的内容会后续被保存到本地文件中,致使插件后续写入的文档属性失效。
修复位置依然在 updatePostImages 方法中,而这次是下半部分。
以下为原始代码:
if (result.code === 0 /* OK */) {
postParams.content = postParams.content.replace(img.original, ![${imgFile.name}](${result.data.url})
);
} else {
if (result.error.code === 2 /* ServerInternalError */) {
new import_obsidian8.Notice(result.error.message, ERROR_NOTICE_TIMEOUT);
} else {
new import_obsidian8.Notice(this.plugin.i18n.t("error_mediaUploadFailed", {
name: imgFile.name
}), ERROR_NOTICE_TIMEOUT);
}
}
}
} else {
}
}
if (this.plugin.settings.replaceMediaLinks) {
activeEditor.editor.setValue(postParams.content);
}
}
}
以下为修复后的代码:
if (result.code === 0 /* OK */) {
postParams.content = postParams.content.replace(img.original, ![${imgFile.name}](${result.data.url})
);
} else {
if (result.error.code === 2 /* ServerInternalError */) {
new import_obsidian8.Notice(result.error.message, ERROR_NOTICE_TIMEOUT);
} else {
new import_obsidian8.Notice(this.plugin.i18n.t("error_mediaUploadFailed", {
name: imgFile.name
}), ERROR_NOTICE_TIMEOUT);
}
}
}
} else {
}
}
if (this.plugin.settings.replaceMediaLinks) {
// activeEditor.editor.setValue(postParams.content);
// 这里取原始的MatterData格式化为字符串加到文件内容前面,以保证MatterData不会被清除
let matterDataString = null;
const activeFile = this.plugin.app.workspace.getActiveFile();
if (activeFile) {
let thiz = this
await this.plugin.app.fileManager.processFrontMatter(activeFile, (fm) => {
if (fm) {
matterDataString = thiz.matterDataFormatString(fm);
}
});
}
let finalContent = postParams.content;
if (matterDataString) {
finalContent = matterDataString + '\n' + finalContent;
}
activeEditor.editor.setValue(finalContent);
// 保存到文件,否则后续processFrontMatter拿不到准确数据
const file = this.plugin.app.workspace.getActiveFile();
if (file) {
await app.vault.modify(file, finalContent);
}
}
}
}
其中,matterDataFormatString 方法的实现如下,放在与 updatePostImages 方法同级即可:
matterDataFormatString(md) {
let result = '---\n';
if (md.profileName) {
result += 'profileName: ' + md.profileName + '\n';
}
if (md.postId) {
result += 'postId: \"' + md.postId + '\"\n';
}
if (md.postType) {
result += 'postType: ' + md.postType + '\n';
}
if (md.categories) {
result += 'categories:\n'
for (let index = 0; index < md.categories.length; index++) {
result += ' - ' + md.categories[index] + '\n';
}
}
for (let key in md) {
if (md.hasOwnProperty(key)
&& key != 'profileName'
&& key != 'postId'
&& key != 'postType'
&& key != 'categories') {
result += key + ': ' + md[key] + '\n';
}
}
result += '---';
return result;
}
Obsidian 其他插件:[[Obsidian 插件及使用小技巧|Obsidian 插件及使用小技巧]]