refactor plugin, split into modules

This commit is contained in:
Nystik
2026-03-27 19:53:19 +01:00
parent 59a1c293bf
commit 47cf768e26
7 changed files with 448 additions and 216 deletions

View File

@@ -0,0 +1,95 @@
const { Notice, TFile, TFolder } = require("obsidian");
function getVaultId() {
return window.__currentVaultId || "";
}
function triggerDownload(endpoint, filePath, downloadName) {
const vaultId = getVaultId();
const url =
`/api/fs/${endpoint}` +
`?vault=${encodeURIComponent(vaultId)}` +
`&path=${encodeURIComponent(filePath)}`;
const a = document.createElement("a");
a.href = url;
a.download = downloadName;
a.click();
}
function showFilePicker(app, targetFolder = null) {
const input = document.createElement("input");
input.type = "file";
input.multiple = true;
input.style.display = "none";
input.addEventListener("change", async () => {
const files = Array.from(input.files || []);
if (files.length === 0) return;
const folder = targetFolder || app.vault.getRoot();
const folderPath = folder.path;
new Notice(`Uploading ${files.length} file(s)...`);
let successCount = 0;
let errorCount = 0;
for (const file of files) {
try {
const arrayBuffer = await file.arrayBuffer();
const targetPath = folderPath
? `${folderPath}/${file.name}`
: file.name;
await app.vault.createBinary(targetPath, arrayBuffer);
successCount++;
} catch (e) {
console.error("[ignis-bridge] Upload failed:", file.name, e);
errorCount++;
}
}
if (successCount > 0) {
new Notice(`Uploaded ${successCount} file(s) successfully`);
}
if (errorCount > 0) {
new Notice(`Failed to upload ${errorCount} file(s)`, 5000);
}
input.remove();
});
document.body.appendChild(input);
input.click();
}
function addFileMenuItems(menu, file) {
menu.addItem((item) => {
item
.setTitle("Download")
.setIcon("download")
.onClick(() => triggerDownload("download", file.path, file.name));
});
}
function addFolderMenuItems(menu, folder, app) {
menu.addItem((item) => {
item
.setTitle("Download as ZIP")
.setIcon("download")
.onClick(() =>
triggerDownload("download-zip", folder.path, `${folder.name}.zip`),
);
});
menu.addItem((item) => {
item
.setTitle("Upload file")
.setIcon("upload")
.onClick(() => showFilePicker(app, folder));
});
}
module.exports = { showFilePicker, addFileMenuItems, addFolderMenuItems };

34
plugin/src/main.js Normal file
View File

@@ -0,0 +1,34 @@
const { Plugin, TFile, TFolder } = require("obsidian");
const { showFilePicker, addFileMenuItems, addFolderMenuItems } = require("./file-actions");
const { patchSettingsModal, unpatchSettingsModal } = require("./settings/inject");
window.__obsidianAPI = require("obsidian");
class IgnisBridgePlugin extends Plugin {
async onload() {
console.log("[ignis-bridge] Plugin loaded");
patchSettingsModal(this);
this.addRibbonIcon("upload", "Upload file", () => {
showFilePicker(this.app);
});
this.registerEvent(
this.app.workspace.on("file-menu", (menu, file) => {
if (file instanceof TFile) {
addFileMenuItems(menu, file);
} else if (file instanceof TFolder) {
addFolderMenuItems(menu, file, this.app);
}
}),
);
}
onunload() {
unpatchSettingsModal(this);
console.log("[ignis-bridge] Plugin unloaded");
}
}
module.exports = IgnisBridgePlugin;

View File

@@ -0,0 +1,17 @@
const { Setting } = require("obsidian");
function display(containerEl) {
containerEl.createEl("h2", { text: "Ignis General Settings" });
new Setting(containerEl)
.setName("Example toggle")
.setDesc("This is a test toggle to prove the Setting API works.")
.addToggle((toggle) => {
toggle.setValue(false);
toggle.onChange((value) => {
console.log("[ignis] Toggle:", value);
});
});
}
module.exports = { display };

View File

@@ -0,0 +1,86 @@
const generalTab = require("./general-tab");
const serverPluginsTab = require("./server-plugins-tab");
function createNavEl(tab, setting) {
const nav = document.createElement("div");
nav.className = "vertical-tab-nav-item tappable";
const title = document.createElement("div");
title.className = "vertical-tab-nav-item-title";
title.textContent = tab.name;
nav.appendChild(title);
const chevron = document.createElement("div");
chevron.className = "vertical-tab-nav-item-chevron";
nav.appendChild(chevron);
nav.addEventListener("click", () => {
setting.openTab(tab);
});
return nav;
}
function createTab(id, name, displayFn) {
const tab = {
id,
name,
containerEl: createDiv("vertical-tab-content"),
navEl: null,
display() {
this.containerEl.empty();
displayFn(this.containerEl);
},
hide() {
this.containerEl.empty();
},
};
return tab;
}
function injectIgnisSettings(setting) {
const group = document.createElement("div");
group.className = "vertical-tab-header-group";
const title = document.createElement("div");
title.className = "vertical-tab-header-group-title";
title.textContent = "Ignis";
group.appendChild(title);
const items = document.createElement("div");
items.className = "vertical-tab-header-group-items";
group.appendChild(items);
const tabs = [
createTab("ignis-general", "General", generalTab.display),
createTab("ignis-server-plugins", "Server Plugins", serverPluginsTab.display),
];
for (const tab of tabs) {
tab.navEl = createNavEl(tab, setting);
items.appendChild(tab.navEl);
}
setting.tabHeadersEl.appendChild(group);
}
function patchSettingsModal(plugin) {
const original = plugin.app.setting.onOpen;
plugin._originalOnOpen = original;
plugin.app.setting.onOpen = function () {
original.call(this);
injectIgnisSettings(this);
};
}
function unpatchSettingsModal(plugin) {
if (plugin._originalOnOpen) {
plugin.app.setting.onOpen = plugin._originalOnOpen;
}
}
module.exports = { patchSettingsModal, unpatchSettingsModal };

View File

@@ -0,0 +1,14 @@
const { Setting } = require("obsidian");
function display(containerEl) {
containerEl.createEl("h2", { text: "Server Plugins" });
new Setting(containerEl)
.setName("Example text input")
.setDesc("This is a test input to prove a second tab works.")
.addText((text) => {
text.setPlaceholder("Type something...");
});
}
module.exports = { display };