From 3ec5caf330b7cdd3460c4c1a92d92b344112f90d Mon Sep 17 00:00:00 2001 From: Ellpeck Date: Mon, 30 Jan 2023 12:11:19 +0100 Subject: [PATCH] fixed custom frames in new windows closes #43 closes #46 --- src/frame.ts | 18 ++++++++++----- src/main.ts | 2 +- src/view.ts | 2 +- .../plugins/obsidian-custom-frames/data.json | 11 ++++++++++ .../plugins/obsidian-custom-frames/main | 22 +++++++++++++------ .../plugins/obsidian-custom-frames/main.js | 22 +++++++++++++------ 6 files changed, 56 insertions(+), 21 deletions(-) diff --git a/src/frame.ts b/src/frame.ts index c7c99a1..1167e86 100644 --- a/src/frame.ts +++ b/src/frame.ts @@ -12,19 +12,29 @@ export class CustomFrame { this.data = data; } - create(additionalStyle: string = undefined, urlSuffix: string = undefined): any { + create(parent: HTMLElement, additionalStyle: string = undefined, urlSuffix: string = undefined): void { let style = `padding: ${this.settings.padding}px;`; if (additionalStyle) style += additionalStyle; if (Platform.isDesktopApp && !this.data.forceIframe) { - this.frame = document.createElement("webview"); + let frameDoc = parent.doc; + this.frame = frameDoc.createElement("webview"); + parent.appendChild(this.frame); this.frame.setAttribute("allowpopups", ""); this.frame.addEventListener("dom-ready", () => { this.frame.setZoomFactor(this.data.zoomLevel); this.frame.insertCSS(this.data.customCss); }); + this.frame.addEventListener("destroyed", () => { + // recreate the webview if it was moved to a new window + if (frameDoc != parent.doc) { + this.frame.detach(); + this.create(parent, additionalStyle, urlSuffix); + } + }); } else { - this.frame = document.createElement("iframe"); + this.frame = parent.doc.createElement("iframe"); + parent.appendChild(this.frame); this.frame.setAttribute("sandbox", "allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation"); this.frame.setAttribute("allow", "encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;"); style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`; @@ -40,8 +50,6 @@ export class CustomFrame { src += urlSuffix; } this.frame.setAttribute("src", src); - - return this.frame; } refresh(): void { diff --git a/src/main.ts b/src/main.ts index f021850..49656b8 100644 --- a/src/main.ts +++ b/src/main.ts @@ -68,7 +68,7 @@ export default class CustomFramesPlugin extends Plugin { urlSuffix ||= ""; let frame = new CustomFrame(this.settings, data); - e.appendChild(frame.create(style, urlSuffix)); + frame.create(e, style, urlSuffix); }); } diff --git a/src/view.ts b/src/view.ts index f1aba41..5b111eb 100644 --- a/src/view.ts +++ b/src/view.ts @@ -53,7 +53,7 @@ export class CustomFrameView extends ItemView { onload(): void { this.contentEl.empty(); this.contentEl.addClass("custom-frames-view"); - this.contentEl.appendChild(this.frame.create()); + this.frame.create(this.contentEl); } onPaneMenu(menu: Menu, source: string): void { diff --git a/test-vault/.obsidian/plugins/obsidian-custom-frames/data.json b/test-vault/.obsidian/plugins/obsidian-custom-frames/data.json index 0143e9a..6151f36 100644 --- a/test-vault/.obsidian/plugins/obsidian-custom-frames/data.json +++ b/test-vault/.obsidian/plugins/obsidian-custom-frames/data.json @@ -32,6 +32,17 @@ "zoomLevel": 1, "forceIframe": false, "customCss": "/* hide the menu bar, \"Keep\" text, and logo */\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child[class*=\" \"],\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child,\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child > div:first-child {\ndisplay: none !important;\n}" + }, + { + "url": "https://scholar.google.com", + "displayName": "Scholar", + "icon": "book", + "hideOnMobile": false, + "addRibbonIcon": true, + "openInCenter": false, + "zoomLevel": 1, + "forceIframe": false, + "customCss": "" } ], "padding": 5 diff --git a/test-vault/.obsidian/plugins/obsidian-custom-frames/main b/test-vault/.obsidian/plugins/obsidian-custom-frames/main index f5f8b48..cf86647 100644 --- a/test-vault/.obsidian/plugins/obsidian-custom-frames/main +++ b/test-vault/.obsidian/plugins/obsidian-custom-frames/main @@ -181,19 +181,28 @@ var CustomFrame = class { this.settings = settings; this.data = data; } - create(additionalStyle = void 0, urlSuffix = void 0) { + create(parent, additionalStyle = void 0, urlSuffix = void 0) { let style = `padding: ${this.settings.padding}px;`; if (additionalStyle) style += additionalStyle; if (import_obsidian.Platform.isDesktopApp && !this.data.forceIframe) { - this.frame = document.createElement("webview"); + let frameDoc = parent.doc; + this.frame = frameDoc.createElement("webview"); + parent.appendChild(this.frame); this.frame.setAttribute("allowpopups", ""); this.frame.addEventListener("dom-ready", () => { this.frame.setZoomFactor(this.data.zoomLevel); this.frame.insertCSS(this.data.customCss); }); + this.frame.addEventListener("destroyed", () => { + if (frameDoc != parent.doc) { + this.frame.detach(); + this.create(parent, additionalStyle, urlSuffix); + } + }); } else { - this.frame = document.createElement("iframe"); + this.frame = parent.doc.createElement("iframe"); + parent.appendChild(this.frame); this.frame.setAttribute("sandbox", "allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation"); this.frame.setAttribute("allow", "encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;"); style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`; @@ -208,7 +217,6 @@ var CustomFrame = class { src += urlSuffix; } this.frame.setAttribute("src", src); - return this.frame; } refresh() { if (this.frame instanceof HTMLIFrameElement) { @@ -422,7 +430,7 @@ var _CustomFrameView = class extends import_obsidian3.ItemView { onload() { this.contentEl.empty(); this.contentEl.addClass("custom-frames-view"); - this.contentEl.appendChild(this.frame.create()); + this.frame.create(this.contentEl); } onPaneMenu(menu, source) { super.onPaneMenu(menu, source); @@ -539,7 +547,7 @@ var CustomFramesPlugin = class extends import_obsidian4.Plugin { let urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim(); urlSuffix || (urlSuffix = ""); let frame = new CustomFrame(this.settings, data); - e.appendChild(frame.create(style, urlSuffix)); + frame.create(e, style, urlSuffix); }); }); } @@ -570,4 +578,4 @@ var CustomFramesPlugin = class extends import_obsidian4.Plugin { }); } }; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/frame.ts", "src/settings.ts", "src/settings-tab.ts", "src/view.ts"],
  "sourcesContent": ["import { Plugin, Platform, WorkspaceLeaf } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFramesSettings, defaultSettings, getIcon, getId } from \"./settings\";\r\nimport { CustomFramesSettingTab } from \"./settings-tab\";\r\nimport { CustomFrameView } from \"./view\";\r\n\r\nexport default class CustomFramesPlugin extends Plugin {\r\n\r\n\tsettings: CustomFramesSettings;\r\n\r\n\tasync onload(): Promise<void> {\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tfor (let frame of this.settings.frames) {\r\n\t\t\tif (!frame.url || !frame.displayName)\r\n\t\t\t\tcontinue;\r\n\t\t\tlet name = `custom-frames-${getId(frame)}`;\r\n\t\t\tif (Platform.isMobileApp && frame.hideOnMobile) {\r\n\t\t\t\tconsole.log(`Skipping frame ${name} which is hidden on mobile`);\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\t\t\ttry {\r\n\t\t\t\tconsole.log(`Registering frame ${name} for URL ${frame.url}`);\r\n\r\n\t\t\t\tthis.registerView(name, l => new CustomFrameView(l, this.settings, frame, name));\r\n\t\t\t\tthis.addCommand({\r\n\t\t\t\t\tid: `open-${name}`,\r\n\t\t\t\t\tname: `Open ${frame.displayName}`,\r\n\t\t\t\t\tcallback: () => this.openLeaf(name, frame.openInCenter, false),\r\n\t\t\t\t});\r\n\r\n\t\t\t\tif (frame.addRibbonIcon)\r\n\t\t\t\t\tthis.addRibbonIcon(getIcon(frame), `Open ${frame.displayName}`,\r\n\t\t\t\t\t\te => this.openLeaf(name, frame.openInCenter, Platform.isMacOS ? e.metaKey : e.ctrlKey));\r\n\t\t\t} catch {\r\n\t\t\t\tconsole.error(`Couldn't register frame ${name}, is there already one with the same name?`);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tthis.addSettingTab(new CustomFramesSettingTab(this.app, this));\r\n\r\n\t\tthis.registerMarkdownCodeBlockProcessor(\"custom-frames\", (s, e) => {\r\n\t\t\te.empty();\r\n\t\t\te.addClass(\"custom-frames-view-file\");\r\n\r\n\t\t\tlet frameMatch = /frame:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet frameName = frameMatch && frameMatch[1].trim();\r\n\t\t\tif (!frameName) {\r\n\t\t\t\te.createSpan({ text: \"Couldn't parse frame name\" });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tlet data = this.settings.frames.find(f => f.displayName == frameName);\r\n\t\t\tif (!data) {\r\n\t\t\t\te.createSpan({ text: `Couldn't find a frame with name ${frameName}` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (Platform.isMobileApp && data.hideOnMobile) {\r\n\t\t\t\te.createSpan({ text: `${frameName} is hidden on mobile` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tlet styleMatch = /style:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet style = styleMatch && styleMatch[1].trim();\r\n\t\t\tstyle ||= \"height: 600px;\";\r\n\r\n\t\t\tlet urlSuffixMatch = /urlsuffix:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim();\r\n\t\t\turlSuffix ||= \"\";\r\n\r\n\t\t\tlet frame = new CustomFrame(this.settings, data);\r\n\t\t\te.appendChild(frame.create(style, urlSuffix));\r\n\t\t});\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tthis.settings = Object.assign({}, defaultSettings, await this.loadData());\r\n\t}\r\n\r\n\tasync saveSettings() {\r\n\t\tawait this.saveData(this.settings);\r\n\t}\r\n\r\n\tprivate async openLeaf(name: string, center: boolean, split: boolean): Promise<void> {\r\n\t\tlet leaf: WorkspaceLeaf;\r\n\t\tif (center) {\r\n\t\t\tleaf = this.app.workspace.getLeaf(split);\r\n\t\t\tawait leaf.setViewState({ type: name, active: true });\r\n\t\t} else {\r\n\t\t\tif (!this.app.workspace.getLeavesOfType(name).length)\r\n\t\t\t\tawait this.app.workspace.getRightLeaf(false).setViewState({ type: name, active: true });\r\n\t\t\tleaf = this.app.workspace.getLeavesOfType(name)[0];\r\n\t\t\tthis.app.workspace.revealLeaf(leaf);\r\n\t\t}\r\n\t\tif (leaf.view instanceof CustomFrameView)\r\n\t\t\tleaf.view.focus();\r\n\t}\r\n}\r\n", "import { Platform } from \"obsidian\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getId } from \"./settings\";\r\n\r\nexport class CustomFrame {\r\n\r\n    private readonly settings: CustomFramesSettings;\r\n    private readonly data: CustomFrameSettings;\r\n    private frame: HTMLIFrameElement | any;\r\n\r\n    constructor(settings: CustomFramesSettings, data: CustomFrameSettings) {\r\n        this.settings = settings;\r\n        this.data = data;\r\n    }\r\n\r\n    create(additionalStyle: string = undefined, urlSuffix: string = undefined): any {\r\n        let style = `padding: ${this.settings.padding}px;`;\r\n        if (additionalStyle)\r\n            style += additionalStyle;\r\n        if (Platform.isDesktopApp && !this.data.forceIframe) {\r\n            this.frame = document.createElement(\"webview\");\r\n            this.frame.setAttribute(\"allowpopups\", \"\");\r\n            this.frame.addEventListener(\"dom-ready\", () => {\r\n                this.frame.setZoomFactor(this.data.zoomLevel);\r\n                this.frame.insertCSS(this.data.customCss);\r\n            });\r\n        } else {\r\n            this.frame = document.createElement(\"iframe\");\r\n            this.frame.setAttribute(\"sandbox\", \"allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation\");\r\n            this.frame.setAttribute(\"allow\", \"encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;\");\r\n            style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`;\r\n        }\r\n        this.frame.addClass(\"custom-frames-frame\");\r\n        this.frame.addClass(`custom-frames-${getId(this.data)}`);\r\n        this.frame.setAttribute(\"style\", style);\r\n\r\n        let src = this.data.url;\r\n        if (urlSuffix) {\r\n            if (!urlSuffix.startsWith(\"/\"))\r\n                src += \"/\";\r\n            src += urlSuffix;\r\n        }\r\n        this.frame.setAttribute(\"src\", src);\r\n\r\n        return this.frame;\r\n    }\r\n\r\n    refresh(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.location.reload();\r\n        } else {\r\n            this.frame.reload();\r\n        }\r\n    }\r\n\r\n    return(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.open(this.data.url);\r\n        } else {\r\n            this.frame.loadURL(this.data.url);\r\n        }\r\n    }\r\n\r\n    goBack(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.back();\r\n        } else {\r\n            this.frame.goBack();\r\n        }\r\n    }\r\n\r\n    goForward(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.forward();\r\n        } else {\r\n            this.frame.goForward();\r\n        }\r\n    }\r\n\r\n    toggleDevTools(): void {\r\n        if (!(this.frame instanceof HTMLIFrameElement)) {\r\n            if (!this.frame.isDevToolsOpened()) {\r\n                this.frame.openDevTools();\r\n            } else {\r\n                this.frame.closeDevTools();\r\n            }\r\n        }\r\n    }\r\n\r\n    getCurrentUrl(): string {\r\n        return this.frame instanceof HTMLIFrameElement ? this.frame.contentWindow.location.href : this.frame.getURL();\r\n    }\r\n\r\n    focus(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.focus();\r\n        } else {\r\n            this.frame.focus();\r\n        }\r\n    }\r\n}\r\n", "export const defaultSettings: CustomFramesSettings = {\r\n    frames: [],\r\n    padding: 5\r\n};\r\nexport const presets: Record<string, CustomFrameSettings> = {\r\n    \"obsidian\": {\r\n        url: \"https://forum.obsidian.md/\",\r\n        displayName: \"Obsidian Forum\",\r\n        icon: \"edit\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"detexify\": {\r\n        url: \"https://detexify.kirelabs.org/classify.html\",\r\n        displayName: \"Detexify\",\r\n        icon: \"type\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: false,\r\n        zoomLevel: .95,\r\n        forceIframe: false,\r\n        customCss: `/* hide info clutter and ad banner */\r\n#classify--info-area,\r\n.adsbygoogle {\r\n\tdisplay: none !important\r\n}`\r\n    },\r\n    \"calendar\": {\r\n        url: \"https://calendar.google.com/calendar\",\r\n        displayName: \"Google Calendar\",\r\n        icon: \"calendar\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar, \"Keep\" text, and logo */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child[class*=\" \"],\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child,\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child > div:first-child {\r\ndisplay: none !important;\r\n}`\r\n    },\r\n    \"keep\": {\r\n        url: \"https://keep.google.com\",\r\n        displayName: \"Google Keep\",\r\n        icon: \"files\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar and the \"Keep\" text */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child,\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child > a:first-child > span {\r\n\tdisplay: none !important;\r\n}`\r\n    },\r\n    \"todoist\": {\r\n        url: \"https://todoist.com\",\r\n        displayName: \"Todoist\",\r\n        icon: \"list-checks\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the help, home, search, and productivity overview buttons, create extra space, and prevent toast pop-up from acting weird */\r\n[aria-label=\"Go to Home view\"], #quick_find, [aria-label=\"Productivity\"], [aria-label=\"Help & Feedback\"] {\r\n\tdisplay: none !important;\r\n}\r\n\r\n.view_content {\r\n\tpadding-left: 15px;\r\n}\r\n\r\n.view_header {\r\n\tpadding-left: 15px;\r\n\tpadding-top: 10px;\r\n}\r\n\r\n.undo_toast {\r\n\twidth: 95%;\r\n}`\r\n    },\r\n    \"notion\": {\r\n        url: \"https://www.notion.so/\",\r\n        displayName: \"Notion\",\r\n        icon: \"box\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"twitter\": {\r\n        url: \"https://twitter.com\",\r\n        displayName: \"Twitter\",\r\n        icon: \"twitter\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    }\r\n};\r\n\r\nexport interface CustomFramesSettings {\r\n    frames: CustomFrameSettings[];\r\n    padding: number;\r\n}\r\n\r\nexport interface CustomFrameSettings {\r\n    url: string;\r\n    displayName: string;\r\n    icon: string;\r\n    hideOnMobile: boolean;\r\n    addRibbonIcon: boolean;\r\n    openInCenter: boolean;\r\n    zoomLevel: number;\r\n    forceIframe: boolean;\r\n    customCss: string;\r\n}\r\n\r\nexport function getIcon(settings: CustomFrameSettings) {\r\n    return settings.icon ? `lucide-${settings.icon}` : \"documents\";\r\n}\r\n\r\nexport function getId(settings: CustomFrameSettings) {\r\n    return settings.displayName.toLowerCase().replace(/\\s/g, \"-\");\r\n}\r\n", "import { App, ButtonComponent, DropdownComponent, PluginSettingTab, Setting } from \"obsidian\";\r\nimport { defaultSettings, presets } from \"./settings\";\r\nimport CustomFramesPlugin from \"./main\";\r\n\r\nexport class CustomFramesSettingTab extends PluginSettingTab {\r\n\r\n    plugin: CustomFramesPlugin;\r\n\r\n    constructor(app: App, plugin: CustomFramesPlugin) {\r\n        super(app, plugin);\r\n        this.plugin = plugin;\r\n    }\r\n\r\n    display(): void {\r\n        this.containerEl.empty();\r\n        this.containerEl.createEl(\"h2\", { text: \"Custom Frames Settings\" });\r\n        this.containerEl.createEl(\"p\", { text: \"Please note that Obsidian has to be restarted or reloaded for most of these settings to take effect.\", cls: \"mod-warning\" });\r\n\r\n        new Setting(this.containerEl)\r\n            .setName(\"Frame Padding\")\r\n            .setDesc(\"The padding that should be left around the inside of custom frame panes, in pixels.\")\r\n            .addText(t => {\r\n                t.inputEl.type = \"number\";\r\n                t.setValue(String(this.plugin.settings.padding));\r\n                t.onChange(async v => {\r\n                    this.plugin.settings.padding = v.length ? Number(v) : defaultSettings.padding;\r\n                    await this.plugin.saveSettings();\r\n                });\r\n            });\r\n\r\n        for (let frame of this.plugin.settings.frames) {\r\n            let heading = this.containerEl.createEl(\"h3\", { text: frame.displayName || \"Unnamed Frame\" });\r\n            let toggle = new ButtonComponent(this.containerEl)\r\n                .setButtonText(\"Show Settings\")\r\n                .setClass(\"custom-frames-show\")\r\n                .onClick(async () => {\r\n                    content.hidden = !content.hidden;\r\n                    toggle.setButtonText(content.hidden ? \"Show Settings\" : \"Hide Settings\");\r\n                });\r\n            let content = this.containerEl.createDiv();\r\n            content.hidden = true;\r\n\r\n            new Setting(content)\r\n                .setName(\"Display Name\")\r\n                .setDesc(\"The display name that this frame should have.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.displayName);\r\n                    t.onChange(async v => {\r\n                        frame.displayName = v;\r\n                        heading.setText(frame.displayName || \"Unnamed Frame\");\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Icon\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"The icon that this frame's pane should have. The names of any \" });\r\n                    f.createEl(\"a\", { text: \"Lucide icons\", href: \"https://lucide.dev/\" });\r\n                    f.createSpan({ text: \" can be used.\" });\r\n                }))\r\n                .addText(t => {\r\n                    t.setValue(frame.icon);\r\n                    t.onChange(async v => {\r\n                        frame.icon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"URL\")\r\n                .setDesc(\"The URL that should be opened in this frame.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.url);\r\n                    t.onChange(async v => {\r\n                        frame.url = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Disable on Mobile\")\r\n                .setDesc(\"Custom Frames is a lot more restricted on mobile devices and doesn't allow for the same types of content to be displayed. If a frame doesn't work as expected on mobile, it can be disabled.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.hideOnMobile);\r\n                    t.onChange(async v => {\r\n                        frame.hideOnMobile = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Add Ribbon Icon\")\r\n                .setDesc(\"Whether a button to open this frame should be added to the ribbon.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.addRibbonIcon);\r\n                    t.onChange(async v => {\r\n                        frame.addRibbonIcon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Open in Center\")\r\n                .setDesc(\"Whether this frame should be opened in the unpinned center editor rather than one of the panes on the side. This is useful for sites that don't work well in a narrow view, or sites that don't require a note to be open when viewed.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.openInCenter);\r\n                    t.onChange(async v => {\r\n                        frame.openInCenter = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Force iframe\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"Whether this frame should use iframes on desktop as opposed to Electron webviews.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Only enable this setting if the frame is causing issues or frequent crashes. This setting causes all Desktop-only settings to be ignored.\" });\r\n                }))\r\n                .addToggle(t => {\r\n                    t.setValue(frame.forceIframe);\r\n                    t.onChange(async v => {\r\n                        frame.forceIframe = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Page Zoom\")\r\n                .setDesc(\"The zoom that this frame's page should be displayed with, as a percentage.\")\r\n                .addText(t => {\r\n                    t.inputEl.type = \"number\";\r\n                    t.setValue(String(frame.zoomLevel * 100));\r\n                    t.onChange(async v => {\r\n                        frame.zoomLevel = v.length ? Number(v) / 100 : 1;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Additional CSS\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"A snippet of additional CSS that should be applied to this frame.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Note that this is only applied on Desktop.\" });\r\n                }))\r\n                .addTextArea(t => {\r\n                    t.inputEl.rows = 5;\r\n                    t.inputEl.cols = 50;\r\n                    t.setValue(frame.customCss);\r\n                    t.onChange(async v => {\r\n                        frame.customCss = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new ButtonComponent(content)\r\n                .setButtonText(\"Remove Frame\")\r\n                .onClick(async () => {\r\n                    this.plugin.settings.frames.remove(frame);\r\n                    await this.plugin.saveSettings();\r\n                    this.display();\r\n                });\r\n        }\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"Create a new frame, either from a preset shipped with the plugin, or a custom one that you can edit yourself. Each frame's pane can be opened using the \\\"Custom Frames: Open\\\" command.\" });\r\n\r\n        let addDiv = this.containerEl.createDiv();\r\n        let dropdown = new DropdownComponent(addDiv);\r\n        dropdown.addOption(\"new\", \"Custom\");\r\n        for (let key of Object.keys(presets))\r\n            dropdown.addOption(key, presets[key].displayName);\r\n        new ButtonComponent(addDiv)\r\n            .setButtonText(\"Add Frame\")\r\n            .setClass(\"custom-frames-add\")\r\n            .onClick(async () => {\r\n                let option = dropdown.getValue();\r\n                if (option == \"new\") {\r\n                    this.plugin.settings.frames.push({\r\n                        url: \"\",\r\n                        displayName: \"New Frame\",\r\n                        icon: \"\",\r\n                        hideOnMobile: true,\r\n                        addRibbonIcon: false,\r\n                        openInCenter: false,\r\n                        zoomLevel: 1,\r\n                        forceIframe: false,\r\n                        customCss: \"\"\r\n                    });\r\n                } else {\r\n                    this.plugin.settings.frames.push(presets[option]);\r\n                }\r\n                await this.plugin.saveSettings();\r\n                this.display();\r\n            });\r\n\r\n        var disclaimer = this.containerEl.createEl(\"p\", { cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \"Please be advised that, when adding a site as a custom frame, you potentially expose personal information you enter to other plugins you have installed. For more information, see \" });\r\n        disclaimer.createEl(\"a\", { text: \"this discussion\", href: \"https://github.com/Ellpeck/ObsidianCustomFrames/issues/54#issuecomment-1210879685\", cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \".\" });\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"If you like this plugin and want to support its development, you can do so through my website by clicking this fancy image!\" });\r\n        this.containerEl.createEl(\"a\", { href: \"https://ellpeck.de/support\" })\r\n            .createEl(\"img\", { attr: { src: \"https://ellpeck.de/res/generalsupport.png\" }, cls: \"custom-frames-support\" });\r\n    }\r\n}\r\n", "import { ItemView, WorkspaceLeaf, Menu } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getIcon } from \"./settings\";\r\n\r\nexport class CustomFrameView extends ItemView {\r\n\r\n    private static readonly actions: Action[] = [\r\n        {\r\n            name: \"Return to original page\",\r\n            icon: \"home\",\r\n            action: v => v.frame.return()\r\n        }, {\r\n            name: \"Open dev tools\",\r\n            icon: \"binary\",\r\n            action: v => v.frame.toggleDevTools()\r\n        }, {\r\n            name: \"Copy link\",\r\n            icon: \"link\",\r\n            action: v => navigator.clipboard.writeText(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Open in browser\",\r\n            icon: \"globe\",\r\n            action: v => open(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Refresh\",\r\n            icon: \"refresh-cw\",\r\n            action: v => v.frame.refresh()\r\n        }, {\r\n            name: \"Go back\",\r\n            icon: \"arrow-left\",\r\n            action: v => v.frame.goBack()\r\n        }, {\r\n            name: \"Go forward\",\r\n            icon: \"arrow-right\",\r\n            action: v => v.frame.goForward()\r\n        }\r\n    ];\r\n\r\n    private readonly data: CustomFrameSettings;\r\n    private readonly name: string;\r\n    private frame: CustomFrame;\r\n\r\n    constructor(leaf: WorkspaceLeaf, settings: CustomFramesSettings, data: CustomFrameSettings, name: string) {\r\n        super(leaf);\r\n        this.data = data;\r\n        this.name = name;\r\n        this.frame = new CustomFrame(settings, data);\r\n\r\n        for (let action of CustomFrameView.actions)\r\n            this.addAction(action.icon, action.name, () => action.action(this));\r\n    }\r\n\r\n    onload(): void {\r\n        this.contentEl.empty();\r\n        this.contentEl.addClass(\"custom-frames-view\");\r\n        this.contentEl.appendChild(this.frame.create());\r\n    }\r\n\r\n    onPaneMenu(menu: Menu, source: string): void {\r\n        super.onPaneMenu(menu, source);\r\n        for (let action of CustomFrameView.actions) {\r\n            menu.addItem(i => {\r\n                i.setTitle(action.name);\r\n                i.setIcon(action.icon);\r\n                i.onClick(() => action.action(this));\r\n            });\r\n        }\r\n    }\r\n\r\n    getViewType(): string {\r\n        return this.name;\r\n    }\r\n\r\n    getDisplayText(): string {\r\n        return this.data.displayName;\r\n    }\r\n\r\n    getIcon(): string {\r\n        return getIcon(this.data);\r\n    }\r\n\r\n    focus(): void {\r\n        this.frame.focus();\r\n    }\r\n}\r\n\r\ninterface Action {\r\n    name: string;\r\n    icon: string;\r\n    action: (view: CustomFrameView) => any;\r\n}\r\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA,uBAAgD;;;ACAhD,sBAAyB;;;ACAlB,IAAM,kBAAwC;AAAA,EACjD,QAAQ;AAAA,EACR,SAAS;AAAA;AAEN,IAAM,UAA+C;AAAA,EACxD,YAAY;AAAA,IACR,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA,EAEf,YAAY;AAAA,IACR,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMf,YAAY;AAAA,IACR,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOf,QAAQ;AAAA,IACJ,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMf,WAAW;AAAA,IACP,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkBf,UAAU;AAAA,IACN,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA,EAEf,WAAW;AAAA,IACP,KAAK;AAAA,IACL,aAAa;AAAA,IACb,MAAM;AAAA,IACN,cAAc;AAAA,IACd,eAAe;AAAA,IACf,cAAc;AAAA,IACd,WAAW;AAAA,IACX,aAAa;AAAA,IACb,WAAW;AAAA;AAAA;AAqBZ,iBAAiB,UAA+B;AACnD,SAAO,SAAS,OAAO,UAAU,SAAS,SAAS;AAAA;AAGhD,eAAe,UAA+B;AACjD,SAAO,SAAS,YAAY,cAAc,QAAQ,OAAO;AAAA;;;ADpItD,wBAAkB;AAAA,EAMrB,YAAY,UAAgC,MAA2B;AACnE,SAAK,WAAW;AAChB,SAAK,OAAO;AAAA;AAAA,EAGhB,OAAO,kBAA0B,QAAW,YAAoB,QAAgB;AAC5E,QAAI,QAAQ,YAAY,KAAK,SAAS;AACtC,QAAI;AACA,eAAS;AACb,QAAI,yBAAS,gBAAgB,CAAC,KAAK,KAAK,aAAa;AACjD,WAAK,QAAQ,SAAS,cAAc;AACpC,WAAK,MAAM,aAAa,eAAe;AACvC,WAAK,MAAM,iBAAiB,aAAa,MAAM;AAC3C,aAAK,MAAM,cAAc,KAAK,KAAK;AACnC,aAAK,MAAM,UAAU,KAAK,KAAK;AAAA;AAAA,WAEhC;AACH,WAAK,QAAQ,SAAS,cAAc;AACpC,WAAK,MAAM,aAAa,WAAW;AACnC,WAAK,MAAM,aAAa,SAAS;AACjC,eAAS,oBAAoB,KAAK,KAAK;AAAA;AAE3C,SAAK,MAAM,SAAS;AACpB,SAAK,MAAM,SAAS,iBAAiB,MAAM,KAAK;AAChD,SAAK,MAAM,aAAa,SAAS;AAEjC,QAAI,MAAM,KAAK,KAAK;AACpB,QAAI,WAAW;AACX,UAAI,CAAC,UAAU,WAAW;AACtB,eAAO;AACX,aAAO;AAAA;AAEX,SAAK,MAAM,aAAa,OAAO;AAE/B,WAAO,KAAK;AAAA;AAAA,EAGhB,UAAgB;AACZ,QAAI,KAAK,iBAAiB,mBAAmB;AACzC,WAAK,MAAM,cAAc,SAAS;AAAA,WAC/B;AACH,WAAK,MAAM;AAAA;AAAA;AAAA,EAInB,SAAe;AACX,QAAI,KAAK,iBAAiB,mBAAmB;AACzC,WAAK,MAAM,cAAc,KAAK,KAAK,KAAK;AAAA,WACrC;AACH,WAAK,MAAM,QAAQ,KAAK,KAAK;AAAA;AAAA;AAAA,EAIrC,SAAe;AACX,QAAI,KAAK,iBAAiB,mBAAmB;AACzC,WAAK,MAAM,cAAc,QAAQ;AAAA,WAC9B;AACH,WAAK,MAAM;AAAA;AAAA;AAAA,EAInB,YAAkB;AACd,QAAI,KAAK,iBAAiB,mBAAmB;AACzC,WAAK,MAAM,cAAc,QAAQ;AAAA,WAC9B;AACH,WAAK,MAAM;AAAA;AAAA;AAAA,EAInB,iBAAuB;AACnB,QAAI,CAAE,MAAK,iBAAiB,oBAAoB;AAC5C,UAAI,CAAC,KAAK,MAAM,oBAAoB;AAChC,aAAK,MAAM;AAAA,aACR;AACH,aAAK,MAAM;AAAA;AAAA;AAAA;AAAA,EAKvB,gBAAwB;AACpB,WAAO,KAAK,iBAAiB,oBAAoB,KAAK,MAAM,cAAc,SAAS,OAAO,KAAK,MAAM;AAAA;AAAA,EAGzG,QAAc;AACV,QAAI,KAAK,iBAAiB,mBAAmB;AACzC,WAAK,MAAM,cAAc;AAAA,WACtB;AACH,WAAK,MAAM;AAAA;AAAA;AAAA;;;AEhGvB,uBAAmF;AAI5E,2CAAqC,kCAAiB;AAAA,EAIzD,YAAY,KAAU,QAA4B;AAC9C,UAAM,KAAK;AACX,SAAK,SAAS;AAAA;AAAA,EAGlB,UAAgB;AACZ,SAAK,YAAY;AACjB,SAAK,YAAY,SAAS,MAAM,EAAE,MAAM;AACxC,SAAK,YAAY,SAAS,KAAK,EAAE,MAAM,wGAAwG,KAAK;AAEpJ,QAAI,yBAAQ,KAAK,aACZ,QAAQ,iBACR,QAAQ,uFACR,QAAQ,OAAK;AACV,QAAE,QAAQ,OAAO;AACjB,QAAE,SAAS,OAAO,KAAK,OAAO,SAAS;AACvC,QAAE,SAAS,CAAM,MAAK;AAClB,aAAK,OAAO,SAAS,UAAU,EAAE,SAAS,OAAO,KAAK,gBAAgB;AACtE,cAAM,KAAK,OAAO;AAAA;AAAA;AAI9B,aAAS,SAAS,KAAK,OAAO,SAAS,QAAQ;AAC3C,UAAI,UAAU,KAAK,YAAY,SAAS,MAAM,EAAE,MAAM,MAAM,eAAe;AAC3E,UAAI,SAAS,IAAI,iCAAgB,KAAK,aACjC,cAAc,iBACd,SAAS,sBACT,QAAQ,MAAY;AACjB,gBAAQ,SAAS,CAAC,QAAQ;AAC1B,eAAO,cAAc,QAAQ,SAAS,kBAAkB;AAAA;AAEhE,UAAI,UAAU,KAAK,YAAY;AAC/B,cAAQ,SAAS;AAEjB,UAAI,yBAAQ,SACP,QAAQ,gBACR,QAAQ,iDACR,QAAQ,OAAK;AACV,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,cAAc;AACpB,kBAAQ,QAAQ,MAAM,eAAe;AACrC,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,QACR,QAAQ,eAAe,OAAK;AACzB,UAAE,WAAW,EAAE,MAAM;AACrB,UAAE,SAAS,KAAK,EAAE,MAAM,gBAAgB,MAAM;AAC9C,UAAE,WAAW,EAAE,MAAM;AAAA,UAExB,QAAQ,OAAK;AACV,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,OAAO;AACb,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,OACR,QAAQ,gDACR,QAAQ,OAAK;AACV,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,MAAM;AACZ,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,qBACR,QAAQ,gMACR,UAAU,OAAK;AACZ,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,eAAe;AACrB,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,mBACR,QAAQ,sEACR,UAAU,OAAK;AACZ,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,gBAAgB;AACtB,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,kBACR,QAAQ,0OACR,UAAU,OAAK;AACZ,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,eAAe;AACrB,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,gBACR,QAAQ,eAAe,OAAK;AACzB,UAAE,WAAW,EAAE,MAAM;AACrB,UAAE,SAAS;AACX,UAAE,SAAS,MAAM,EAAE,MAAM;AAAA,UAE5B,UAAU,OAAK;AACZ,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,cAAc;AACpB,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,aACR,QAAQ,8EACR,QAAQ,OAAK;AACV,UAAE,QAAQ,OAAO;AACjB,UAAE,SAAS,OAAO,MAAM,YAAY;AACpC,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,YAAY,EAAE,SAAS,OAAO,KAAK,MAAM;AAC/C,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,yBAAQ,SACP,QAAQ,kBACR,QAAQ,eAAe,OAAK;AACzB,UAAE,WAAW,EAAE,MAAM;AACrB,UAAE,SAAS;AACX,UAAE,SAAS,MAAM,EAAE,MAAM;AAAA,UAE5B,YAAY,OAAK;AACd,UAAE,QAAQ,OAAO;AACjB,UAAE,QAAQ,OAAO;AACjB,UAAE,SAAS,MAAM;AACjB,UAAE,SAAS,CAAM,MAAK;AAClB,gBAAM,YAAY;AAClB,gBAAM,KAAK,OAAO;AAAA;AAAA;AAG9B,UAAI,iCAAgB,SACf,cAAc,gBACd,QAAQ,MAAY;AACjB,aAAK,OAAO,SAAS,OAAO,OAAO;AACnC,cAAM,KAAK,OAAO;AAClB,aAAK;AAAA;AAAA;AAIjB,SAAK,YAAY,SAAS;AAC1B,SAAK,YAAY,SAAS,KAAK,EAAE,MAAM;AAEvC,QAAI,SAAS,KAAK,YAAY;AAC9B,QAAI,WAAW,IAAI,mCAAkB;AACrC,aAAS,UAAU,OAAO;AAC1B,aAAS,OAAO,OAAO,KAAK;AACxB,eAAS,UAAU,KAAK,QAAQ,KAAK;AACzC,QAAI,iCAAgB,QACf,cAAc,aACd,SAAS,qBACT,QAAQ,MAAY;AACjB,UAAI,SAAS,SAAS;AACtB,UAAI,UAAU,OAAO;AACjB,aAAK,OAAO,SAAS,OAAO,KAAK;AAAA,UAC7B,KAAK;AAAA,UACL,aAAa;AAAA,UACb,MAAM;AAAA,UACN,cAAc;AAAA,UACd,eAAe;AAAA,UACf,cAAc;AAAA,UACd,WAAW;AAAA,UACX,aAAa;AAAA,UACb,WAAW;AAAA;AAAA,aAEZ;AACH,aAAK,OAAO,SAAS,OAAO,KAAK,QAAQ;AAAA;AAE7C,YAAM,KAAK,OAAO;AAClB,WAAK;AAAA;AAGb,QAAI,aAAa,KAAK,YAAY,SAAS,KAAK,EAAE,KAAK;AACvD,eAAW,WAAW,EAAE,MAAM;AAC9B,eAAW,SAAS,KAAK,EAAE,MAAM,mBAAmB,MAAM,qFAAqF,KAAK;AACpJ,eAAW,WAAW,EAAE,MAAM;AAE9B,SAAK,YAAY,SAAS;AAC1B,SAAK,YAAY,SAAS,KAAK,EAAE,MAAM;AACvC,SAAK,YAAY,SAAS,KAAK,EAAE,MAAM,gCAClC,SAAS,OAAO,EAAE,MAAM,EAAE,KAAK,+CAA+C,KAAK;AAAA;AAAA;;;ACrMhG,uBAA8C;AAIvC,qCAA8B,0BAAS;AAAA,EAsC1C,YAAY,MAAqB,UAAgC,MAA2B,MAAc;AACtG,UAAM;AACN,SAAK,OAAO;AACZ,SAAK,OAAO;AACZ,SAAK,QAAQ,IAAI,YAAY,UAAU;AAEvC,aAAS,UAAU,iBAAgB;AAC/B,WAAK,UAAU,OAAO,MAAM,OAAO,MAAM,MAAM,OAAO,OAAO;AAAA;AAAA,EAGrE,SAAe;AACX,SAAK,UAAU;AACf,SAAK,UAAU,SAAS;AACxB,SAAK,UAAU,YAAY,KAAK,MAAM;AAAA;AAAA,EAG1C,WAAW,MAAY,QAAsB;AACzC,UAAM,WAAW,MAAM;AACvB,aAAS,UAAU,iBAAgB,SAAS;AACxC,WAAK,QAAQ,OAAK;AACd,UAAE,SAAS,OAAO;AAClB,UAAE,QAAQ,OAAO;AACjB,UAAE,QAAQ,MAAM,OAAO,OAAO;AAAA;AAAA;AAAA;AAAA,EAK1C,cAAsB;AAClB,WAAO,KAAK;AAAA;AAAA,EAGhB,iBAAyB;AACrB,WAAO,KAAK,KAAK;AAAA;AAAA,EAGrB,UAAkB;AACd,WAAO,QAAQ,KAAK;AAAA;AAAA,EAGxB,QAAc;AACV,SAAK,MAAM;AAAA;AAAA;AA9EZ;AAEqB,AAFrB,gBAEqB,UAAoB;AAAA,EACxC;AAAA,IACI,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,EAAE,MAAM;AAAA;AAAA,EACtB;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,EAAE,MAAM;AAAA;AAAA,EACtB;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,UAAU,UAAU,UAAU,EAAE,MAAM;AAAA;AAAA,EACpD;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,KAAK,EAAE,MAAM;AAAA;AAAA,EAC3B;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,EAAE,MAAM;AAAA;AAAA,EACtB;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,EAAE,MAAM;AAAA;AAAA,EACtB;AAAA,IACC,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,OAAK,EAAE,MAAM;AAAA;AAAA;;;AJ5BjC,uCAAgD,wBAAO;AAAA,EAIhD,SAAwB;AAAA;AAC7B,YAAM,KAAK;AAEX,eAAS,SAAS,KAAK,SAAS,QAAQ;AACvC,YAAI,CAAC,MAAM,OAAO,CAAC,MAAM;AACxB;AACD,YAAI,OAAO,iBAAiB,MAAM;AAClC,YAAI,0BAAS,eAAe,MAAM,cAAc;AAC/C,kBAAQ,IAAI,kBAAkB;AAC9B;AAAA;AAED,YAAI;AACH,kBAAQ,IAAI,qBAAqB,gBAAgB,MAAM;AAEvD,eAAK,aAAa,MAAM,OAAK,IAAI,gBAAgB,GAAG,KAAK,UAAU,OAAO;AAC1E,eAAK,WAAW;AAAA,YACf,IAAI,QAAQ;AAAA,YACZ,MAAM,QAAQ,MAAM;AAAA,YACpB,UAAU,MAAM,KAAK,SAAS,MAAM,MAAM,cAAc;AAAA;AAGzD,cAAI,MAAM;AACT,iBAAK,cAAc,QAAQ,QAAQ,QAAQ,MAAM,eAChD,OAAK,KAAK,SAAS,MAAM,MAAM,cAAc,0BAAS,UAAU,EAAE,UAAU,EAAE;AAAA,iBACzE,GAAN;AACD,kBAAQ,MAAM,2BAA2B;AAAA;AAAA;AAI3C,WAAK,cAAc,IAAI,uBAAuB,KAAK,KAAK;AAExD,WAAK,mCAAmC,iBAAiB,CAAC,GAAG,MAAM;AAClE,UAAE;AACF,UAAE,SAAS;AAEX,YAAI,aAAa,mBAAmB,KAAK;AACzC,YAAI,YAAY,cAAc,WAAW,GAAG;AAC5C,YAAI,CAAC,WAAW;AACf,YAAE,WAAW,EAAE,MAAM;AACrB;AAAA;AAED,YAAI,OAAO,KAAK,SAAS,OAAO,KAAK,OAAK,EAAE,eAAe;AAC3D,YAAI,CAAC,MAAM;AACV,YAAE,WAAW,EAAE,MAAM,mCAAmC;AACxD;AAAA;AAED,YAAI,0BAAS,eAAe,KAAK,cAAc;AAC9C,YAAE,WAAW,EAAE,MAAM,GAAG;AACxB;AAAA;AAGD,YAAI,aAAa,mBAAmB,KAAK;AACzC,YAAI,QAAQ,cAAc,WAAW,GAAG;AACxC,0BAAU;AAEV,YAAI,iBAAiB,uBAAuB,KAAK;AACjD,YAAI,YAAY,kBAAkB,eAAe,GAAG;AACpD,kCAAc;AAEd,YAAI,QAAQ,IAAI,YAAY,KAAK,UAAU;AAC3C,UAAE,YAAY,MAAM,OAAO,OAAO;AAAA;AAAA;AAAA;AAAA,EAI9B,eAAe;AAAA;AACpB,WAAK,WAAW,OAAO,OAAO,IAAI,iBAAiB,MAAM,KAAK;AAAA;AAAA;AAAA,EAGzD,eAAe;AAAA;AACpB,YAAM,KAAK,SAAS,KAAK;AAAA;AAAA;AAAA,EAGZ,SAAS,MAAc,QAAiB,OAA+B;AAAA;AACpF,UAAI;AACJ,UAAI,QAAQ;AACX,eAAO,KAAK,IAAI,UAAU,QAAQ;AAClC,cAAM,KAAK,aAAa,EAAE,MAAM,MAAM,QAAQ;AAAA,aACxC;AACN,YAAI,CAAC,KAAK,IAAI,UAAU,gBAAgB,MAAM;AAC7C,gBAAM,KAAK,IAAI,UAAU,aAAa,OAAO,aAAa,EAAE,MAAM,MAAM,QAAQ;AACjF,eAAO,KAAK,IAAI,UAAU,gBAAgB,MAAM;AAChD,aAAK,IAAI,UAAU,WAAW;AAAA;AAE/B,UAAI,KAAK,gBAAgB;AACxB,aAAK,KAAK;AAAA;AAAA;AAAA;",
  "names": []
}
 +//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/frame.ts", "src/settings.ts", "src/settings-tab.ts", "src/view.ts"],
  "sourcesContent": ["import { Plugin, Platform, WorkspaceLeaf } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFramesSettings, defaultSettings, getIcon, getId } from \"./settings\";\r\nimport { CustomFramesSettingTab } from \"./settings-tab\";\r\nimport { CustomFrameView } from \"./view\";\r\n\r\nexport default class CustomFramesPlugin extends Plugin {\r\n\r\n\tsettings: CustomFramesSettings;\r\n\r\n\tasync onload(): Promise<void> {\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tfor (let frame of this.settings.frames) {\r\n\t\t\tif (!frame.url || !frame.displayName)\r\n\t\t\t\tcontinue;\r\n\t\t\tlet name = `custom-frames-${getId(frame)}`;\r\n\t\t\tif (Platform.isMobileApp && frame.hideOnMobile) {\r\n\t\t\t\tconsole.log(`Skipping frame ${name} which is hidden on mobile`);\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\t\t\ttry {\r\n\t\t\t\tconsole.log(`Registering frame ${name} for URL ${frame.url}`);\r\n\r\n\t\t\t\tthis.registerView(name, l => new CustomFrameView(l, this.settings, frame, name));\r\n\t\t\t\tthis.addCommand({\r\n\t\t\t\t\tid: `open-${name}`,\r\n\t\t\t\t\tname: `Open ${frame.displayName}`,\r\n\t\t\t\t\tcallback: () => this.openLeaf(name, frame.openInCenter, false),\r\n\t\t\t\t});\r\n\r\n\t\t\t\tif (frame.addRibbonIcon)\r\n\t\t\t\t\tthis.addRibbonIcon(getIcon(frame), `Open ${frame.displayName}`,\r\n\t\t\t\t\t\te => this.openLeaf(name, frame.openInCenter, Platform.isMacOS ? e.metaKey : e.ctrlKey));\r\n\t\t\t} catch {\r\n\t\t\t\tconsole.error(`Couldn't register frame ${name}, is there already one with the same name?`);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tthis.addSettingTab(new CustomFramesSettingTab(this.app, this));\r\n\r\n\t\tthis.registerMarkdownCodeBlockProcessor(\"custom-frames\", (s, e) => {\r\n\t\t\te.empty();\r\n\t\t\te.addClass(\"custom-frames-view-file\");\r\n\r\n\t\t\tlet frameMatch = /frame:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet frameName = frameMatch && frameMatch[1].trim();\r\n\t\t\tif (!frameName) {\r\n\t\t\t\te.createSpan({ text: \"Couldn't parse frame name\" });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tlet data = this.settings.frames.find(f => f.displayName == frameName);\r\n\t\t\tif (!data) {\r\n\t\t\t\te.createSpan({ text: `Couldn't find a frame with name ${frameName}` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (Platform.isMobileApp && data.hideOnMobile) {\r\n\t\t\t\te.createSpan({ text: `${frameName} is hidden on mobile` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tlet styleMatch = /style:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet style = styleMatch && styleMatch[1].trim();\r\n\t\t\tstyle ||= \"height: 600px;\";\r\n\r\n\t\t\tlet urlSuffixMatch = /urlsuffix:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim();\r\n\t\t\turlSuffix ||= \"\";\r\n\r\n\t\t\tlet frame = new CustomFrame(this.settings, data);\r\n\t\t\tframe.create(e, style, urlSuffix);\r\n\t\t});\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tthis.settings = Object.assign({}, defaultSettings, await this.loadData());\r\n\t}\r\n\r\n\tasync saveSettings() {\r\n\t\tawait this.saveData(this.settings);\r\n\t}\r\n\r\n\tprivate async openLeaf(name: string, center: boolean, split: boolean): Promise<void> {\r\n\t\tlet leaf: WorkspaceLeaf;\r\n\t\tif (center) {\r\n\t\t\tleaf = this.app.workspace.getLeaf(split);\r\n\t\t\tawait leaf.setViewState({ type: name, active: true });\r\n\t\t} else {\r\n\t\t\tif (!this.app.workspace.getLeavesOfType(name).length)\r\n\t\t\t\tawait this.app.workspace.getRightLeaf(false).setViewState({ type: name, active: true });\r\n\t\t\tleaf = this.app.workspace.getLeavesOfType(name)[0];\r\n\t\t\tthis.app.workspace.revealLeaf(leaf);\r\n\t\t}\r\n\t\tif (leaf.view instanceof CustomFrameView)\r\n\t\t\tleaf.view.focus();\r\n\t}\r\n}\r\n", "import { Platform } from \"obsidian\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getId } from \"./settings\";\r\n\r\nexport class CustomFrame {\r\n\r\n    private readonly settings: CustomFramesSettings;\r\n    private readonly data: CustomFrameSettings;\r\n    private frame: HTMLIFrameElement | any;\r\n\r\n    constructor(settings: CustomFramesSettings, data: CustomFrameSettings) {\r\n        this.settings = settings;\r\n        this.data = data;\r\n    }\r\n\r\n    create(parent: HTMLElement, additionalStyle: string = undefined, urlSuffix: string = undefined): void {\r\n        let style = `padding: ${this.settings.padding}px;`;\r\n        if (additionalStyle)\r\n            style += additionalStyle;\r\n        if (Platform.isDesktopApp && !this.data.forceIframe) {\r\n            let frameDoc = parent.doc;\r\n            this.frame = frameDoc.createElement(\"webview\");\r\n            parent.appendChild(this.frame);\r\n            this.frame.setAttribute(\"allowpopups\", \"\");\r\n            this.frame.addEventListener(\"dom-ready\", () => {\r\n                this.frame.setZoomFactor(this.data.zoomLevel);\r\n                this.frame.insertCSS(this.data.customCss);\r\n            });\r\n            this.frame.addEventListener(\"destroyed\", () => {\r\n                // recreate the webview if it was moved to a new window\r\n                if (frameDoc != parent.doc) {\r\n                    this.frame.detach();\r\n                    this.create(parent, additionalStyle, urlSuffix);\r\n                }\r\n            });\r\n        } else {\r\n            this.frame = parent.doc.createElement(\"iframe\");\r\n            parent.appendChild(this.frame);\r\n            this.frame.setAttribute(\"sandbox\", \"allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation\");\r\n            this.frame.setAttribute(\"allow\", \"encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;\");\r\n            style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`;\r\n        }\r\n        this.frame.addClass(\"custom-frames-frame\");\r\n        this.frame.addClass(`custom-frames-${getId(this.data)}`);\r\n        this.frame.setAttribute(\"style\", style);\r\n\r\n        let src = this.data.url;\r\n        if (urlSuffix) {\r\n            if (!urlSuffix.startsWith(\"/\"))\r\n                src += \"/\";\r\n            src += urlSuffix;\r\n        }\r\n        this.frame.setAttribute(\"src\", src);\r\n    }\r\n\r\n    refresh(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.location.reload();\r\n        } else {\r\n            this.frame.reload();\r\n        }\r\n    }\r\n\r\n    return(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.open(this.data.url);\r\n        } else {\r\n            this.frame.loadURL(this.data.url);\r\n        }\r\n    }\r\n\r\n    goBack(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.back();\r\n        } else {\r\n            this.frame.goBack();\r\n        }\r\n    }\r\n\r\n    goForward(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.forward();\r\n        } else {\r\n            this.frame.goForward();\r\n        }\r\n    }\r\n\r\n    toggleDevTools(): void {\r\n        if (!(this.frame instanceof HTMLIFrameElement)) {\r\n            if (!this.frame.isDevToolsOpened()) {\r\n                this.frame.openDevTools();\r\n            } else {\r\n                this.frame.closeDevTools();\r\n            }\r\n        }\r\n    }\r\n\r\n    getCurrentUrl(): string {\r\n        return this.frame instanceof HTMLIFrameElement ? this.frame.contentWindow.location.href : this.frame.getURL();\r\n    }\r\n\r\n    focus(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.focus();\r\n        } else {\r\n            this.frame.focus();\r\n        }\r\n    }\r\n}\r\n", "export const defaultSettings: CustomFramesSettings = {\r\n    frames: [],\r\n    padding: 5\r\n};\r\nexport const presets: Record<string, CustomFrameSettings> = {\r\n    \"obsidian\": {\r\n        url: \"https://forum.obsidian.md/\",\r\n        displayName: \"Obsidian Forum\",\r\n        icon: \"edit\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"detexify\": {\r\n        url: \"https://detexify.kirelabs.org/classify.html\",\r\n        displayName: \"Detexify\",\r\n        icon: \"type\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: false,\r\n        zoomLevel: .95,\r\n        forceIframe: false,\r\n        customCss: `/* hide info clutter and ad banner */\r\n#classify--info-area,\r\n.adsbygoogle {\r\n\tdisplay: none !important\r\n}`\r\n    },\r\n    \"calendar\": {\r\n        url: \"https://calendar.google.com/calendar\",\r\n        displayName: \"Google Calendar\",\r\n        icon: \"calendar\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar, \"Keep\" text, and logo */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child[class*=\" \"],\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child,\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child > div:first-child {\r\ndisplay: none !important;\r\n}`\r\n    },\r\n    \"keep\": {\r\n        url: \"https://keep.google.com\",\r\n        displayName: \"Google Keep\",\r\n        icon: \"files\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar and the \"Keep\" text */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child,\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child > a:first-child > span {\r\n\tdisplay: none !important;\r\n}`\r\n    },\r\n    \"todoist\": {\r\n        url: \"https://todoist.com\",\r\n        displayName: \"Todoist\",\r\n        icon: \"list-checks\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the help, home, search, and productivity overview buttons, create extra space, and prevent toast pop-up from acting weird */\r\n[aria-label=\"Go to Home view\"], #quick_find, [aria-label=\"Productivity\"], [aria-label=\"Help & Feedback\"] {\r\n\tdisplay: none !important;\r\n}\r\n\r\n.view_content {\r\n\tpadding-left: 15px;\r\n}\r\n\r\n.view_header {\r\n\tpadding-left: 15px;\r\n\tpadding-top: 10px;\r\n}\r\n\r\n.undo_toast {\r\n\twidth: 95%;\r\n}`\r\n    },\r\n    \"notion\": {\r\n        url: \"https://www.notion.so/\",\r\n        displayName: \"Notion\",\r\n        icon: \"box\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"twitter\": {\r\n        url: \"https://twitter.com\",\r\n        displayName: \"Twitter\",\r\n        icon: \"twitter\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    }\r\n};\r\n\r\nexport interface CustomFramesSettings {\r\n    frames: CustomFrameSettings[];\r\n    padding: number;\r\n}\r\n\r\nexport interface CustomFrameSettings {\r\n    url: string;\r\n    displayName: string;\r\n    icon: string;\r\n    hideOnMobile: boolean;\r\n    addRibbonIcon: boolean;\r\n    openInCenter: boolean;\r\n    zoomLevel: number;\r\n    forceIframe: boolean;\r\n    customCss: string;\r\n}\r\n\r\nexport function getIcon(settings: CustomFrameSettings) {\r\n    return settings.icon ? `lucide-${settings.icon}` : \"documents\";\r\n}\r\n\r\nexport function getId(settings: CustomFrameSettings) {\r\n    return settings.displayName.toLowerCase().replace(/\\s/g, \"-\");\r\n}\r\n", "import { App, ButtonComponent, DropdownComponent, PluginSettingTab, Setting } from \"obsidian\";\r\nimport { defaultSettings, presets } from \"./settings\";\r\nimport CustomFramesPlugin from \"./main\";\r\n\r\nexport class CustomFramesSettingTab extends PluginSettingTab {\r\n\r\n    plugin: CustomFramesPlugin;\r\n\r\n    constructor(app: App, plugin: CustomFramesPlugin) {\r\n        super(app, plugin);\r\n        this.plugin = plugin;\r\n    }\r\n\r\n    display(): void {\r\n        this.containerEl.empty();\r\n        this.containerEl.createEl(\"h2\", { text: \"Custom Frames Settings\" });\r\n        this.containerEl.createEl(\"p\", { text: \"Please note that Obsidian has to be restarted or reloaded for most of these settings to take effect.\", cls: \"mod-warning\" });\r\n\r\n        new Setting(this.containerEl)\r\n            .setName(\"Frame Padding\")\r\n            .setDesc(\"The padding that should be left around the inside of custom frame panes, in pixels.\")\r\n            .addText(t => {\r\n                t.inputEl.type = \"number\";\r\n                t.setValue(String(this.plugin.settings.padding));\r\n                t.onChange(async v => {\r\n                    this.plugin.settings.padding = v.length ? Number(v) : defaultSettings.padding;\r\n                    await this.plugin.saveSettings();\r\n                });\r\n            });\r\n\r\n        for (let frame of this.plugin.settings.frames) {\r\n            let heading = this.containerEl.createEl(\"h3\", { text: frame.displayName || \"Unnamed Frame\" });\r\n            let toggle = new ButtonComponent(this.containerEl)\r\n                .setButtonText(\"Show Settings\")\r\n                .setClass(\"custom-frames-show\")\r\n                .onClick(async () => {\r\n                    content.hidden = !content.hidden;\r\n                    toggle.setButtonText(content.hidden ? \"Show Settings\" : \"Hide Settings\");\r\n                });\r\n            let content = this.containerEl.createDiv();\r\n            content.hidden = true;\r\n\r\n            new Setting(content)\r\n                .setName(\"Display Name\")\r\n                .setDesc(\"The display name that this frame should have.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.displayName);\r\n                    t.onChange(async v => {\r\n                        frame.displayName = v;\r\n                        heading.setText(frame.displayName || \"Unnamed Frame\");\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Icon\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"The icon that this frame's pane should have. The names of any \" });\r\n                    f.createEl(\"a\", { text: \"Lucide icons\", href: \"https://lucide.dev/\" });\r\n                    f.createSpan({ text: \" can be used.\" });\r\n                }))\r\n                .addText(t => {\r\n                    t.setValue(frame.icon);\r\n                    t.onChange(async v => {\r\n                        frame.icon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"URL\")\r\n                .setDesc(\"The URL that should be opened in this frame.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.url);\r\n                    t.onChange(async v => {\r\n                        frame.url = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Disable on Mobile\")\r\n                .setDesc(\"Custom Frames is a lot more restricted on mobile devices and doesn't allow for the same types of content to be displayed. If a frame doesn't work as expected on mobile, it can be disabled.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.hideOnMobile);\r\n                    t.onChange(async v => {\r\n                        frame.hideOnMobile = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Add Ribbon Icon\")\r\n                .setDesc(\"Whether a button to open this frame should be added to the ribbon.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.addRibbonIcon);\r\n                    t.onChange(async v => {\r\n                        frame.addRibbonIcon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Open in Center\")\r\n                .setDesc(\"Whether this frame should be opened in the unpinned center editor rather than one of the panes on the side. This is useful for sites that don't work well in a narrow view, or sites that don't require a note to be open when viewed.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.openInCenter);\r\n                    t.onChange(async v => {\r\n                        frame.openInCenter = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Force iframe\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"Whether this frame should use iframes on desktop as opposed to Electron webviews.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Only enable this setting if the frame is causing issues or frequent crashes. This setting causes all Desktop-only settings to be ignored.\" });\r\n                }))\r\n                .addToggle(t => {\r\n                    t.setValue(frame.forceIframe);\r\n                    t.onChange(async v => {\r\n                        frame.forceIframe = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Page Zoom\")\r\n                .setDesc(\"The zoom that this frame's page should be displayed with, as a percentage.\")\r\n                .addText(t => {\r\n                    t.inputEl.type = \"number\";\r\n                    t.setValue(String(frame.zoomLevel * 100));\r\n                    t.onChange(async v => {\r\n                        frame.zoomLevel = v.length ? Number(v) / 100 : 1;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Additional CSS\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"A snippet of additional CSS that should be applied to this frame.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Note that this is only applied on Desktop.\" });\r\n                }))\r\n                .addTextArea(t => {\r\n                    t.inputEl.rows = 5;\r\n                    t.inputEl.cols = 50;\r\n                    t.setValue(frame.customCss);\r\n                    t.onChange(async v => {\r\n                        frame.customCss = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new ButtonComponent(content)\r\n                .setButtonText(\"Remove Frame\")\r\n                .onClick(async () => {\r\n                    this.plugin.settings.frames.remove(frame);\r\n                    await this.plugin.saveSettings();\r\n                    this.display();\r\n                });\r\n        }\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"Create a new frame, either from a preset shipped with the plugin, or a custom one that you can edit yourself. Each frame's pane can be opened using the \\\"Custom Frames: Open\\\" command.\" });\r\n\r\n        let addDiv = this.containerEl.createDiv();\r\n        let dropdown = new DropdownComponent(addDiv);\r\n        dropdown.addOption(\"new\", \"Custom\");\r\n        for (let key of Object.keys(presets))\r\n            dropdown.addOption(key, presets[key].displayName);\r\n        new ButtonComponent(addDiv)\r\n            .setButtonText(\"Add Frame\")\r\n            .setClass(\"custom-frames-add\")\r\n            .onClick(async () => {\r\n                let option = dropdown.getValue();\r\n                if (option == \"new\") {\r\n                    this.plugin.settings.frames.push({\r\n                        url: \"\",\r\n                        displayName: \"New Frame\",\r\n                        icon: \"\",\r\n                        hideOnMobile: true,\r\n                        addRibbonIcon: false,\r\n                        openInCenter: false,\r\n                        zoomLevel: 1,\r\n                        forceIframe: false,\r\n                        customCss: \"\"\r\n                    });\r\n                } else {\r\n                    this.plugin.settings.frames.push(presets[option]);\r\n                }\r\n                await this.plugin.saveSettings();\r\n                this.display();\r\n            });\r\n\r\n        var disclaimer = this.containerEl.createEl(\"p\", { cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \"Please be advised that, when adding a site as a custom frame, you potentially expose personal information you enter to other plugins you have installed. For more information, see \" });\r\n        disclaimer.createEl(\"a\", { text: \"this discussion\", href: \"https://github.com/Ellpeck/ObsidianCustomFrames/issues/54#issuecomment-1210879685\", cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \".\" });\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"If you like this plugin and want to support its development, you can do so through my website by clicking this fancy image!\" });\r\n        this.containerEl.createEl(\"a\", { href: \"https://ellpeck.de/support\" })\r\n            .createEl(\"img\", { attr: { src: \"https://ellpeck.de/res/generalsupport.png\" }, cls: \"custom-frames-support\" });\r\n    }\r\n}\r\n", "import { ItemView, WorkspaceLeaf, Menu } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getIcon } from \"./settings\";\r\n\r\nexport class CustomFrameView extends ItemView {\r\n\r\n    private static readonly actions: Action[] = [\r\n        {\r\n            name: \"Return to original page\",\r\n            icon: \"home\",\r\n            action: v => v.frame.return()\r\n        }, {\r\n            name: \"Open dev tools\",\r\n            icon: \"binary\",\r\n            action: v => v.frame.toggleDevTools()\r\n        }, {\r\n            name: \"Copy link\",\r\n            icon: \"link\",\r\n            action: v => navigator.clipboard.writeText(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Open in browser\",\r\n            icon: \"globe\",\r\n            action: v => open(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Refresh\",\r\n            icon: \"refresh-cw\",\r\n            action: v => v.frame.refresh()\r\n        }, {\r\n            name: \"Go back\",\r\n            icon: \"arrow-left\",\r\n            action: v => v.frame.goBack()\r\n        }, {\r\n            name: \"Go forward\",\r\n            icon: \"arrow-right\",\r\n            action: v => v.frame.goForward()\r\n        }\r\n    ];\r\n\r\n    private readonly data: CustomFrameSettings;\r\n    private readonly name: string;\r\n    private frame: CustomFrame;\r\n\r\n    constructor(leaf: WorkspaceLeaf, settings: CustomFramesSettings, data: CustomFrameSettings, name: string) {\r\n        super(leaf);\r\n        this.data = data;\r\n        this.name = name;\r\n        this.frame = new CustomFrame(settings, data);\r\n\r\n        for (let action of CustomFrameView.actions)\r\n            this.addAction(action.icon, action.name, () => action.action(this));\r\n    }\r\n\r\n    onload(): void {\r\n        this.contentEl.empty();\r\n        this.contentEl.addClass(\"custom-frames-view\");\r\n        this.frame.create(this.contentEl);\r\n    }\r\n\r\n    onPaneMenu(menu: Menu, source: string): void {\r\n        super.onPaneMenu(menu, source);\r\n        for (let action of CustomFrameView.actions) {\r\n            menu.addItem(i => {\r\n                i.setTitle(action.name);\r\n                i.setIcon(action.icon);\r\n                i.onClick(() => action.action(this));\r\n            });\r\n        }\r\n    }\r\n\r\n    getViewType(): string {\r\n        return this.name;\r\n    }\r\n\r\n    getDisplayText(): string {\r\n        return this.data.displayName;\r\n    }\r\n\r\n    getIcon(): string {\r\n        return getIcon(this.data);\r\n    }\r\n\r\n    focus(): void {\r\n        this.frame.focus();\r\n    }\r\n}\r\n\r\ninterface Action {\r\n    name: string;\r\n    icon: string;\r\n    action: (view: CustomFrameView) => any;\r\n}\r\n"],
  "mappings": "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",
  "names": []
}
 diff --git a/test-vault/.obsidian/plugins/obsidian-custom-frames/main.js b/test-vault/.obsidian/plugins/obsidian-custom-frames/main.js index f5f8b48..cf86647 100644 --- a/test-vault/.obsidian/plugins/obsidian-custom-frames/main.js +++ b/test-vault/.obsidian/plugins/obsidian-custom-frames/main.js @@ -181,19 +181,28 @@ var CustomFrame = class { this.settings = settings; this.data = data; } - create(additionalStyle = void 0, urlSuffix = void 0) { + create(parent, additionalStyle = void 0, urlSuffix = void 0) { let style = `padding: ${this.settings.padding}px;`; if (additionalStyle) style += additionalStyle; if (import_obsidian.Platform.isDesktopApp && !this.data.forceIframe) { - this.frame = document.createElement("webview"); + let frameDoc = parent.doc; + this.frame = frameDoc.createElement("webview"); + parent.appendChild(this.frame); this.frame.setAttribute("allowpopups", ""); this.frame.addEventListener("dom-ready", () => { this.frame.setZoomFactor(this.data.zoomLevel); this.frame.insertCSS(this.data.customCss); }); + this.frame.addEventListener("destroyed", () => { + if (frameDoc != parent.doc) { + this.frame.detach(); + this.create(parent, additionalStyle, urlSuffix); + } + }); } else { - this.frame = document.createElement("iframe"); + this.frame = parent.doc.createElement("iframe"); + parent.appendChild(this.frame); this.frame.setAttribute("sandbox", "allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation"); this.frame.setAttribute("allow", "encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;"); style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`; @@ -208,7 +217,6 @@ var CustomFrame = class { src += urlSuffix; } this.frame.setAttribute("src", src); - return this.frame; } refresh() { if (this.frame instanceof HTMLIFrameElement) { @@ -422,7 +430,7 @@ var _CustomFrameView = class extends import_obsidian3.ItemView { onload() { this.contentEl.empty(); this.contentEl.addClass("custom-frames-view"); - this.contentEl.appendChild(this.frame.create()); + this.frame.create(this.contentEl); } onPaneMenu(menu, source) { super.onPaneMenu(menu, source); @@ -539,7 +547,7 @@ var CustomFramesPlugin = class extends import_obsidian4.Plugin { let urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim(); urlSuffix || (urlSuffix = ""); let frame = new CustomFrame(this.settings, data); - e.appendChild(frame.create(style, urlSuffix)); + frame.create(e, style, urlSuffix); }); }); } @@ -570,4 +578,4 @@ var CustomFramesPlugin = class extends import_obsidian4.Plugin { }); } }; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/frame.ts", "src/settings.ts", "src/settings-tab.ts", "src/view.ts"],
  "sourcesContent": ["import { Plugin, Platform, WorkspaceLeaf } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFramesSettings, defaultSettings, getIcon, getId } from \"./settings\";\r\nimport { CustomFramesSettingTab } from \"./settings-tab\";\r\nimport { CustomFrameView } from \"./view\";\r\n\r\nexport default class CustomFramesPlugin extends Plugin {\r\n\r\n\tsettings: CustomFramesSettings;\r\n\r\n\tasync onload(): Promise<void> {\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tfor (let frame of this.settings.frames) {\r\n\t\t\tif (!frame.url || !frame.displayName)\r\n\t\t\t\tcontinue;\r\n\t\t\tlet name = `custom-frames-${getId(frame)}`;\r\n\t\t\tif (Platform.isMobileApp && frame.hideOnMobile) {\r\n\t\t\t\tconsole.log(`Skipping frame ${name} which is hidden on mobile`);\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\t\t\ttry {\r\n\t\t\t\tconsole.log(`Registering frame ${name} for URL ${frame.url}`);\r\n\r\n\t\t\t\tthis.registerView(name, l => new CustomFrameView(l, this.settings, frame, name));\r\n\t\t\t\tthis.addCommand({\r\n\t\t\t\t\tid: `open-${name}`,\r\n\t\t\t\t\tname: `Open ${frame.displayName}`,\r\n\t\t\t\t\tcallback: () => this.openLeaf(name, frame.openInCenter, false),\r\n\t\t\t\t});\r\n\r\n\t\t\t\tif (frame.addRibbonIcon)\r\n\t\t\t\t\tthis.addRibbonIcon(getIcon(frame), `Open ${frame.displayName}`,\r\n\t\t\t\t\t\te => this.openLeaf(name, frame.openInCenter, Platform.isMacOS ? e.metaKey : e.ctrlKey));\r\n\t\t\t} catch {\r\n\t\t\t\tconsole.error(`Couldn't register frame ${name}, is there already one with the same name?`);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tthis.addSettingTab(new CustomFramesSettingTab(this.app, this));\r\n\r\n\t\tthis.registerMarkdownCodeBlockProcessor(\"custom-frames\", (s, e) => {\r\n\t\t\te.empty();\r\n\t\t\te.addClass(\"custom-frames-view-file\");\r\n\r\n\t\t\tlet frameMatch = /frame:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet frameName = frameMatch && frameMatch[1].trim();\r\n\t\t\tif (!frameName) {\r\n\t\t\t\te.createSpan({ text: \"Couldn't parse frame name\" });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tlet data = this.settings.frames.find(f => f.displayName == frameName);\r\n\t\t\tif (!data) {\r\n\t\t\t\te.createSpan({ text: `Couldn't find a frame with name ${frameName}` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (Platform.isMobileApp && data.hideOnMobile) {\r\n\t\t\t\te.createSpan({ text: `${frameName} is hidden on mobile` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tlet styleMatch = /style:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet style = styleMatch && styleMatch[1].trim();\r\n\t\t\tstyle ||= \"height: 600px;\";\r\n\r\n\t\t\tlet urlSuffixMatch = /urlsuffix:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim();\r\n\t\t\turlSuffix ||= \"\";\r\n\r\n\t\t\tlet frame = new CustomFrame(this.settings, data);\r\n\t\t\te.appendChild(frame.create(style, urlSuffix));\r\n\t\t});\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tthis.settings = Object.assign({}, defaultSettings, await this.loadData());\r\n\t}\r\n\r\n\tasync saveSettings() {\r\n\t\tawait this.saveData(this.settings);\r\n\t}\r\n\r\n\tprivate async openLeaf(name: string, center: boolean, split: boolean): Promise<void> {\r\n\t\tlet leaf: WorkspaceLeaf;\r\n\t\tif (center) {\r\n\t\t\tleaf = this.app.workspace.getLeaf(split);\r\n\t\t\tawait leaf.setViewState({ type: name, active: true });\r\n\t\t} else {\r\n\t\t\tif (!this.app.workspace.getLeavesOfType(name).length)\r\n\t\t\t\tawait this.app.workspace.getRightLeaf(false).setViewState({ type: name, active: true });\r\n\t\t\tleaf = this.app.workspace.getLeavesOfType(name)[0];\r\n\t\t\tthis.app.workspace.revealLeaf(leaf);\r\n\t\t}\r\n\t\tif (leaf.view instanceof CustomFrameView)\r\n\t\t\tleaf.view.focus();\r\n\t}\r\n}\r\n", "import { Platform } from \"obsidian\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getId } from \"./settings\";\r\n\r\nexport class CustomFrame {\r\n\r\n    private readonly settings: CustomFramesSettings;\r\n    private readonly data: CustomFrameSettings;\r\n    private frame: HTMLIFrameElement | any;\r\n\r\n    constructor(settings: CustomFramesSettings, data: CustomFrameSettings) {\r\n        this.settings = settings;\r\n        this.data = data;\r\n    }\r\n\r\n    create(additionalStyle: string = undefined, urlSuffix: string = undefined): any {\r\n        let style = `padding: ${this.settings.padding}px;`;\r\n        if (additionalStyle)\r\n            style += additionalStyle;\r\n        if (Platform.isDesktopApp && !this.data.forceIframe) {\r\n            this.frame = document.createElement(\"webview\");\r\n            this.frame.setAttribute(\"allowpopups\", \"\");\r\n            this.frame.addEventListener(\"dom-ready\", () => {\r\n                this.frame.setZoomFactor(this.data.zoomLevel);\r\n                this.frame.insertCSS(this.data.customCss);\r\n            });\r\n        } else {\r\n            this.frame = document.createElement(\"iframe\");\r\n            this.frame.setAttribute(\"sandbox\", \"allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation\");\r\n            this.frame.setAttribute(\"allow\", \"encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;\");\r\n            style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`;\r\n        }\r\n        this.frame.addClass(\"custom-frames-frame\");\r\n        this.frame.addClass(`custom-frames-${getId(this.data)}`);\r\n        this.frame.setAttribute(\"style\", style);\r\n\r\n        let src = this.data.url;\r\n        if (urlSuffix) {\r\n            if (!urlSuffix.startsWith(\"/\"))\r\n                src += \"/\";\r\n            src += urlSuffix;\r\n        }\r\n        this.frame.setAttribute(\"src\", src);\r\n\r\n        return this.frame;\r\n    }\r\n\r\n    refresh(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.location.reload();\r\n        } else {\r\n            this.frame.reload();\r\n        }\r\n    }\r\n\r\n    return(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.open(this.data.url);\r\n        } else {\r\n            this.frame.loadURL(this.data.url);\r\n        }\r\n    }\r\n\r\n    goBack(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.back();\r\n        } else {\r\n            this.frame.goBack();\r\n        }\r\n    }\r\n\r\n    goForward(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.forward();\r\n        } else {\r\n            this.frame.goForward();\r\n        }\r\n    }\r\n\r\n    toggleDevTools(): void {\r\n        if (!(this.frame instanceof HTMLIFrameElement)) {\r\n            if (!this.frame.isDevToolsOpened()) {\r\n                this.frame.openDevTools();\r\n            } else {\r\n                this.frame.closeDevTools();\r\n            }\r\n        }\r\n    }\r\n\r\n    getCurrentUrl(): string {\r\n        return this.frame instanceof HTMLIFrameElement ? this.frame.contentWindow.location.href : this.frame.getURL();\r\n    }\r\n\r\n    focus(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.focus();\r\n        } else {\r\n            this.frame.focus();\r\n        }\r\n    }\r\n}\r\n", "export const defaultSettings: CustomFramesSettings = {\r\n    frames: [],\r\n    padding: 5\r\n};\r\nexport const presets: Record<string, CustomFrameSettings> = {\r\n    \"obsidian\": {\r\n        url: \"https://forum.obsidian.md/\",\r\n        displayName: \"Obsidian Forum\",\r\n        icon: \"edit\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"detexify\": {\r\n        url: \"https://detexify.kirelabs.org/classify.html\",\r\n        displayName: \"Detexify\",\r\n        icon: \"type\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: false,\r\n        zoomLevel: .95,\r\n        forceIframe: false,\r\n        customCss: `/* hide info clutter and ad banner */\r\n#classify--info-area,\r\n.adsbygoogle {\r\n\tdisplay: none !important\r\n}`\r\n    },\r\n    \"calendar\": {\r\n        url: \"https://calendar.google.com/calendar\",\r\n        displayName: \"Google Calendar\",\r\n        icon: \"calendar\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar, \"Keep\" text, and logo */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child[class*=\" \"],\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child,\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child > div:first-child {\r\ndisplay: none !important;\r\n}`\r\n    },\r\n    \"keep\": {\r\n        url: \"https://keep.google.com\",\r\n        displayName: \"Google Keep\",\r\n        icon: \"files\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar and the \"Keep\" text */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child,\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child > a:first-child > span {\r\n\tdisplay: none !important;\r\n}`\r\n    },\r\n    \"todoist\": {\r\n        url: \"https://todoist.com\",\r\n        displayName: \"Todoist\",\r\n        icon: \"list-checks\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the help, home, search, and productivity overview buttons, create extra space, and prevent toast pop-up from acting weird */\r\n[aria-label=\"Go to Home view\"], #quick_find, [aria-label=\"Productivity\"], [aria-label=\"Help & Feedback\"] {\r\n\tdisplay: none !important;\r\n}\r\n\r\n.view_content {\r\n\tpadding-left: 15px;\r\n}\r\n\r\n.view_header {\r\n\tpadding-left: 15px;\r\n\tpadding-top: 10px;\r\n}\r\n\r\n.undo_toast {\r\n\twidth: 95%;\r\n}`\r\n    },\r\n    \"notion\": {\r\n        url: \"https://www.notion.so/\",\r\n        displayName: \"Notion\",\r\n        icon: \"box\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"twitter\": {\r\n        url: \"https://twitter.com\",\r\n        displayName: \"Twitter\",\r\n        icon: \"twitter\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    }\r\n};\r\n\r\nexport interface CustomFramesSettings {\r\n    frames: CustomFrameSettings[];\r\n    padding: number;\r\n}\r\n\r\nexport interface CustomFrameSettings {\r\n    url: string;\r\n    displayName: string;\r\n    icon: string;\r\n    hideOnMobile: boolean;\r\n    addRibbonIcon: boolean;\r\n    openInCenter: boolean;\r\n    zoomLevel: number;\r\n    forceIframe: boolean;\r\n    customCss: string;\r\n}\r\n\r\nexport function getIcon(settings: CustomFrameSettings) {\r\n    return settings.icon ? `lucide-${settings.icon}` : \"documents\";\r\n}\r\n\r\nexport function getId(settings: CustomFrameSettings) {\r\n    return settings.displayName.toLowerCase().replace(/\\s/g, \"-\");\r\n}\r\n", "import { App, ButtonComponent, DropdownComponent, PluginSettingTab, Setting } from \"obsidian\";\r\nimport { defaultSettings, presets } from \"./settings\";\r\nimport CustomFramesPlugin from \"./main\";\r\n\r\nexport class CustomFramesSettingTab extends PluginSettingTab {\r\n\r\n    plugin: CustomFramesPlugin;\r\n\r\n    constructor(app: App, plugin: CustomFramesPlugin) {\r\n        super(app, plugin);\r\n        this.plugin = plugin;\r\n    }\r\n\r\n    display(): void {\r\n        this.containerEl.empty();\r\n        this.containerEl.createEl(\"h2\", { text: \"Custom Frames Settings\" });\r\n        this.containerEl.createEl(\"p\", { text: \"Please note that Obsidian has to be restarted or reloaded for most of these settings to take effect.\", cls: \"mod-warning\" });\r\n\r\n        new Setting(this.containerEl)\r\n            .setName(\"Frame Padding\")\r\n            .setDesc(\"The padding that should be left around the inside of custom frame panes, in pixels.\")\r\n            .addText(t => {\r\n                t.inputEl.type = \"number\";\r\n                t.setValue(String(this.plugin.settings.padding));\r\n                t.onChange(async v => {\r\n                    this.plugin.settings.padding = v.length ? Number(v) : defaultSettings.padding;\r\n                    await this.plugin.saveSettings();\r\n                });\r\n            });\r\n\r\n        for (let frame of this.plugin.settings.frames) {\r\n            let heading = this.containerEl.createEl(\"h3\", { text: frame.displayName || \"Unnamed Frame\" });\r\n            let toggle = new ButtonComponent(this.containerEl)\r\n                .setButtonText(\"Show Settings\")\r\n                .setClass(\"custom-frames-show\")\r\n                .onClick(async () => {\r\n                    content.hidden = !content.hidden;\r\n                    toggle.setButtonText(content.hidden ? \"Show Settings\" : \"Hide Settings\");\r\n                });\r\n            let content = this.containerEl.createDiv();\r\n            content.hidden = true;\r\n\r\n            new Setting(content)\r\n                .setName(\"Display Name\")\r\n                .setDesc(\"The display name that this frame should have.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.displayName);\r\n                    t.onChange(async v => {\r\n                        frame.displayName = v;\r\n                        heading.setText(frame.displayName || \"Unnamed Frame\");\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Icon\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"The icon that this frame's pane should have. The names of any \" });\r\n                    f.createEl(\"a\", { text: \"Lucide icons\", href: \"https://lucide.dev/\" });\r\n                    f.createSpan({ text: \" can be used.\" });\r\n                }))\r\n                .addText(t => {\r\n                    t.setValue(frame.icon);\r\n                    t.onChange(async v => {\r\n                        frame.icon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"URL\")\r\n                .setDesc(\"The URL that should be opened in this frame.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.url);\r\n                    t.onChange(async v => {\r\n                        frame.url = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Disable on Mobile\")\r\n                .setDesc(\"Custom Frames is a lot more restricted on mobile devices and doesn't allow for the same types of content to be displayed. If a frame doesn't work as expected on mobile, it can be disabled.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.hideOnMobile);\r\n                    t.onChange(async v => {\r\n                        frame.hideOnMobile = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Add Ribbon Icon\")\r\n                .setDesc(\"Whether a button to open this frame should be added to the ribbon.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.addRibbonIcon);\r\n                    t.onChange(async v => {\r\n                        frame.addRibbonIcon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Open in Center\")\r\n                .setDesc(\"Whether this frame should be opened in the unpinned center editor rather than one of the panes on the side. This is useful for sites that don't work well in a narrow view, or sites that don't require a note to be open when viewed.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.openInCenter);\r\n                    t.onChange(async v => {\r\n                        frame.openInCenter = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Force iframe\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"Whether this frame should use iframes on desktop as opposed to Electron webviews.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Only enable this setting if the frame is causing issues or frequent crashes. This setting causes all Desktop-only settings to be ignored.\" });\r\n                }))\r\n                .addToggle(t => {\r\n                    t.setValue(frame.forceIframe);\r\n                    t.onChange(async v => {\r\n                        frame.forceIframe = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Page Zoom\")\r\n                .setDesc(\"The zoom that this frame's page should be displayed with, as a percentage.\")\r\n                .addText(t => {\r\n                    t.inputEl.type = \"number\";\r\n                    t.setValue(String(frame.zoomLevel * 100));\r\n                    t.onChange(async v => {\r\n                        frame.zoomLevel = v.length ? Number(v) / 100 : 1;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Additional CSS\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"A snippet of additional CSS that should be applied to this frame.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Note that this is only applied on Desktop.\" });\r\n                }))\r\n                .addTextArea(t => {\r\n                    t.inputEl.rows = 5;\r\n                    t.inputEl.cols = 50;\r\n                    t.setValue(frame.customCss);\r\n                    t.onChange(async v => {\r\n                        frame.customCss = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new ButtonComponent(content)\r\n                .setButtonText(\"Remove Frame\")\r\n                .onClick(async () => {\r\n                    this.plugin.settings.frames.remove(frame);\r\n                    await this.plugin.saveSettings();\r\n                    this.display();\r\n                });\r\n        }\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"Create a new frame, either from a preset shipped with the plugin, or a custom one that you can edit yourself. Each frame's pane can be opened using the \\\"Custom Frames: Open\\\" command.\" });\r\n\r\n        let addDiv = this.containerEl.createDiv();\r\n        let dropdown = new DropdownComponent(addDiv);\r\n        dropdown.addOption(\"new\", \"Custom\");\r\n        for (let key of Object.keys(presets))\r\n            dropdown.addOption(key, presets[key].displayName);\r\n        new ButtonComponent(addDiv)\r\n            .setButtonText(\"Add Frame\")\r\n            .setClass(\"custom-frames-add\")\r\n            .onClick(async () => {\r\n                let option = dropdown.getValue();\r\n                if (option == \"new\") {\r\n                    this.plugin.settings.frames.push({\r\n                        url: \"\",\r\n                        displayName: \"New Frame\",\r\n                        icon: \"\",\r\n                        hideOnMobile: true,\r\n                        addRibbonIcon: false,\r\n                        openInCenter: false,\r\n                        zoomLevel: 1,\r\n                        forceIframe: false,\r\n                        customCss: \"\"\r\n                    });\r\n                } else {\r\n                    this.plugin.settings.frames.push(presets[option]);\r\n                }\r\n                await this.plugin.saveSettings();\r\n                this.display();\r\n            });\r\n\r\n        var disclaimer = this.containerEl.createEl(\"p\", { cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \"Please be advised that, when adding a site as a custom frame, you potentially expose personal information you enter to other plugins you have installed. For more information, see \" });\r\n        disclaimer.createEl(\"a\", { text: \"this discussion\", href: \"https://github.com/Ellpeck/ObsidianCustomFrames/issues/54#issuecomment-1210879685\", cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \".\" });\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"If you like this plugin and want to support its development, you can do so through my website by clicking this fancy image!\" });\r\n        this.containerEl.createEl(\"a\", { href: \"https://ellpeck.de/support\" })\r\n            .createEl(\"img\", { attr: { src: \"https://ellpeck.de/res/generalsupport.png\" }, cls: \"custom-frames-support\" });\r\n    }\r\n}\r\n", "import { ItemView, WorkspaceLeaf, Menu } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getIcon } from \"./settings\";\r\n\r\nexport class CustomFrameView extends ItemView {\r\n\r\n    private static readonly actions: Action[] = [\r\n        {\r\n            name: \"Return to original page\",\r\n            icon: \"home\",\r\n            action: v => v.frame.return()\r\n        }, {\r\n            name: \"Open dev tools\",\r\n            icon: \"binary\",\r\n            action: v => v.frame.toggleDevTools()\r\n        }, {\r\n            name: \"Copy link\",\r\n            icon: \"link\",\r\n            action: v => navigator.clipboard.writeText(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Open in browser\",\r\n            icon: \"globe\",\r\n            action: v => open(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Refresh\",\r\n            icon: \"refresh-cw\",\r\n            action: v => v.frame.refresh()\r\n        }, {\r\n            name: \"Go back\",\r\n            icon: \"arrow-left\",\r\n            action: v => v.frame.goBack()\r\n        }, {\r\n            name: \"Go forward\",\r\n            icon: \"arrow-right\",\r\n            action: v => v.frame.goForward()\r\n        }\r\n    ];\r\n\r\n    private readonly data: CustomFrameSettings;\r\n    private readonly name: string;\r\n    private frame: CustomFrame;\r\n\r\n    constructor(leaf: WorkspaceLeaf, settings: CustomFramesSettings, data: CustomFrameSettings, name: string) {\r\n        super(leaf);\r\n        this.data = data;\r\n        this.name = name;\r\n        this.frame = new CustomFrame(settings, data);\r\n\r\n        for (let action of CustomFrameView.actions)\r\n            this.addAction(action.icon, action.name, () => action.action(this));\r\n    }\r\n\r\n    onload(): void {\r\n        this.contentEl.empty();\r\n        this.contentEl.addClass(\"custom-frames-view\");\r\n        this.contentEl.appendChild(this.frame.create());\r\n    }\r\n\r\n    onPaneMenu(menu: Menu, source: string): void {\r\n        super.onPaneMenu(menu, source);\r\n        for (let action of CustomFrameView.actions) {\r\n            menu.addItem(i => {\r\n                i.setTitle(action.name);\r\n                i.setIcon(action.icon);\r\n                i.onClick(() => action.action(this));\r\n            });\r\n        }\r\n    }\r\n\r\n    getViewType(): string {\r\n        return this.name;\r\n    }\r\n\r\n    getDisplayText(): string {\r\n        return this.data.displayName;\r\n    }\r\n\r\n    getIcon(): string {\r\n        return getIcon(this.data);\r\n    }\r\n\r\n    focus(): void {\r\n        this.frame.focus();\r\n    }\r\n}\r\n\r\ninterface Action {\r\n    name: string;\r\n    icon: string;\r\n    action: (view: CustomFrameView) => any;\r\n}\r\n"],
  "mappings": "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",
  "names": []
}
 +//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/frame.ts", "src/settings.ts", "src/settings-tab.ts", "src/view.ts"],
  "sourcesContent": ["import { Plugin, Platform, WorkspaceLeaf } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFramesSettings, defaultSettings, getIcon, getId } from \"./settings\";\r\nimport { CustomFramesSettingTab } from \"./settings-tab\";\r\nimport { CustomFrameView } from \"./view\";\r\n\r\nexport default class CustomFramesPlugin extends Plugin {\r\n\r\n\tsettings: CustomFramesSettings;\r\n\r\n\tasync onload(): Promise<void> {\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tfor (let frame of this.settings.frames) {\r\n\t\t\tif (!frame.url || !frame.displayName)\r\n\t\t\t\tcontinue;\r\n\t\t\tlet name = `custom-frames-${getId(frame)}`;\r\n\t\t\tif (Platform.isMobileApp && frame.hideOnMobile) {\r\n\t\t\t\tconsole.log(`Skipping frame ${name} which is hidden on mobile`);\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\t\t\ttry {\r\n\t\t\t\tconsole.log(`Registering frame ${name} for URL ${frame.url}`);\r\n\r\n\t\t\t\tthis.registerView(name, l => new CustomFrameView(l, this.settings, frame, name));\r\n\t\t\t\tthis.addCommand({\r\n\t\t\t\t\tid: `open-${name}`,\r\n\t\t\t\t\tname: `Open ${frame.displayName}`,\r\n\t\t\t\t\tcallback: () => this.openLeaf(name, frame.openInCenter, false),\r\n\t\t\t\t});\r\n\r\n\t\t\t\tif (frame.addRibbonIcon)\r\n\t\t\t\t\tthis.addRibbonIcon(getIcon(frame), `Open ${frame.displayName}`,\r\n\t\t\t\t\t\te => this.openLeaf(name, frame.openInCenter, Platform.isMacOS ? e.metaKey : e.ctrlKey));\r\n\t\t\t} catch {\r\n\t\t\t\tconsole.error(`Couldn't register frame ${name}, is there already one with the same name?`);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tthis.addSettingTab(new CustomFramesSettingTab(this.app, this));\r\n\r\n\t\tthis.registerMarkdownCodeBlockProcessor(\"custom-frames\", (s, e) => {\r\n\t\t\te.empty();\r\n\t\t\te.addClass(\"custom-frames-view-file\");\r\n\r\n\t\t\tlet frameMatch = /frame:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet frameName = frameMatch && frameMatch[1].trim();\r\n\t\t\tif (!frameName) {\r\n\t\t\t\te.createSpan({ text: \"Couldn't parse frame name\" });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tlet data = this.settings.frames.find(f => f.displayName == frameName);\r\n\t\t\tif (!data) {\r\n\t\t\t\te.createSpan({ text: `Couldn't find a frame with name ${frameName}` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (Platform.isMobileApp && data.hideOnMobile) {\r\n\t\t\t\te.createSpan({ text: `${frameName} is hidden on mobile` });\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tlet styleMatch = /style:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet style = styleMatch && styleMatch[1].trim();\r\n\t\t\tstyle ||= \"height: 600px;\";\r\n\r\n\t\t\tlet urlSuffixMatch = /urlsuffix:([^\\n]+)/gi.exec(s);\r\n\t\t\tlet urlSuffix = urlSuffixMatch && urlSuffixMatch[1].trim();\r\n\t\t\turlSuffix ||= \"\";\r\n\r\n\t\t\tlet frame = new CustomFrame(this.settings, data);\r\n\t\t\tframe.create(e, style, urlSuffix);\r\n\t\t});\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tthis.settings = Object.assign({}, defaultSettings, await this.loadData());\r\n\t}\r\n\r\n\tasync saveSettings() {\r\n\t\tawait this.saveData(this.settings);\r\n\t}\r\n\r\n\tprivate async openLeaf(name: string, center: boolean, split: boolean): Promise<void> {\r\n\t\tlet leaf: WorkspaceLeaf;\r\n\t\tif (center) {\r\n\t\t\tleaf = this.app.workspace.getLeaf(split);\r\n\t\t\tawait leaf.setViewState({ type: name, active: true });\r\n\t\t} else {\r\n\t\t\tif (!this.app.workspace.getLeavesOfType(name).length)\r\n\t\t\t\tawait this.app.workspace.getRightLeaf(false).setViewState({ type: name, active: true });\r\n\t\t\tleaf = this.app.workspace.getLeavesOfType(name)[0];\r\n\t\t\tthis.app.workspace.revealLeaf(leaf);\r\n\t\t}\r\n\t\tif (leaf.view instanceof CustomFrameView)\r\n\t\t\tleaf.view.focus();\r\n\t}\r\n}\r\n", "import { Platform } from \"obsidian\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getId } from \"./settings\";\r\n\r\nexport class CustomFrame {\r\n\r\n    private readonly settings: CustomFramesSettings;\r\n    private readonly data: CustomFrameSettings;\r\n    private frame: HTMLIFrameElement | any;\r\n\r\n    constructor(settings: CustomFramesSettings, data: CustomFrameSettings) {\r\n        this.settings = settings;\r\n        this.data = data;\r\n    }\r\n\r\n    create(parent: HTMLElement, additionalStyle: string = undefined, urlSuffix: string = undefined): void {\r\n        let style = `padding: ${this.settings.padding}px;`;\r\n        if (additionalStyle)\r\n            style += additionalStyle;\r\n        if (Platform.isDesktopApp && !this.data.forceIframe) {\r\n            let frameDoc = parent.doc;\r\n            this.frame = frameDoc.createElement(\"webview\");\r\n            parent.appendChild(this.frame);\r\n            this.frame.setAttribute(\"allowpopups\", \"\");\r\n            this.frame.addEventListener(\"dom-ready\", () => {\r\n                this.frame.setZoomFactor(this.data.zoomLevel);\r\n                this.frame.insertCSS(this.data.customCss);\r\n            });\r\n            this.frame.addEventListener(\"destroyed\", () => {\r\n                // recreate the webview if it was moved to a new window\r\n                if (frameDoc != parent.doc) {\r\n                    this.frame.detach();\r\n                    this.create(parent, additionalStyle, urlSuffix);\r\n                }\r\n            });\r\n        } else {\r\n            this.frame = parent.doc.createElement(\"iframe\");\r\n            parent.appendChild(this.frame);\r\n            this.frame.setAttribute(\"sandbox\", \"allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts allow-top-navigation-by-user-activation\");\r\n            this.frame.setAttribute(\"allow\", \"encrypted-media; fullscreen; oversized-images; picture-in-picture; sync-xhr; geolocation;\");\r\n            style += `transform: scale(${this.data.zoomLevel}); transform-origin: 0 0;`;\r\n        }\r\n        this.frame.addClass(\"custom-frames-frame\");\r\n        this.frame.addClass(`custom-frames-${getId(this.data)}`);\r\n        this.frame.setAttribute(\"style\", style);\r\n\r\n        let src = this.data.url;\r\n        if (urlSuffix) {\r\n            if (!urlSuffix.startsWith(\"/\"))\r\n                src += \"/\";\r\n            src += urlSuffix;\r\n        }\r\n        this.frame.setAttribute(\"src\", src);\r\n    }\r\n\r\n    refresh(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.location.reload();\r\n        } else {\r\n            this.frame.reload();\r\n        }\r\n    }\r\n\r\n    return(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.open(this.data.url);\r\n        } else {\r\n            this.frame.loadURL(this.data.url);\r\n        }\r\n    }\r\n\r\n    goBack(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.back();\r\n        } else {\r\n            this.frame.goBack();\r\n        }\r\n    }\r\n\r\n    goForward(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.history.forward();\r\n        } else {\r\n            this.frame.goForward();\r\n        }\r\n    }\r\n\r\n    toggleDevTools(): void {\r\n        if (!(this.frame instanceof HTMLIFrameElement)) {\r\n            if (!this.frame.isDevToolsOpened()) {\r\n                this.frame.openDevTools();\r\n            } else {\r\n                this.frame.closeDevTools();\r\n            }\r\n        }\r\n    }\r\n\r\n    getCurrentUrl(): string {\r\n        return this.frame instanceof HTMLIFrameElement ? this.frame.contentWindow.location.href : this.frame.getURL();\r\n    }\r\n\r\n    focus(): void {\r\n        if (this.frame instanceof HTMLIFrameElement) {\r\n            this.frame.contentWindow.focus();\r\n        } else {\r\n            this.frame.focus();\r\n        }\r\n    }\r\n}\r\n", "export const defaultSettings: CustomFramesSettings = {\r\n    frames: [],\r\n    padding: 5\r\n};\r\nexport const presets: Record<string, CustomFrameSettings> = {\r\n    \"obsidian\": {\r\n        url: \"https://forum.obsidian.md/\",\r\n        displayName: \"Obsidian Forum\",\r\n        icon: \"edit\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"detexify\": {\r\n        url: \"https://detexify.kirelabs.org/classify.html\",\r\n        displayName: \"Detexify\",\r\n        icon: \"type\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: false,\r\n        zoomLevel: .95,\r\n        forceIframe: false,\r\n        customCss: `/* hide info clutter and ad banner */\r\n#classify--info-area,\r\n.adsbygoogle {\r\n\tdisplay: none !important\r\n}`\r\n    },\r\n    \"calendar\": {\r\n        url: \"https://calendar.google.com/calendar\",\r\n        displayName: \"Google Calendar\",\r\n        icon: \"calendar\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar, \"Keep\" text, and logo */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child[class*=\" \"],\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child,\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child > div:first-child {\r\ndisplay: none !important;\r\n}`\r\n    },\r\n    \"keep\": {\r\n        url: \"https://keep.google.com\",\r\n        displayName: \"Google Keep\",\r\n        icon: \"files\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the menu bar and the \"Keep\" text */\r\nhtml > body > div:nth-child(2) > div:nth-child(2) > div:first-child,\r\nhtml > body > div:first-child > header:first-child > div > div:first-child > div > div:first-child > a:first-child > span {\r\n\tdisplay: none !important;\r\n}`\r\n    },\r\n    \"todoist\": {\r\n        url: \"https://todoist.com\",\r\n        displayName: \"Todoist\",\r\n        icon: \"list-checks\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: `/* hide the help, home, search, and productivity overview buttons, create extra space, and prevent toast pop-up from acting weird */\r\n[aria-label=\"Go to Home view\"], #quick_find, [aria-label=\"Productivity\"], [aria-label=\"Help & Feedback\"] {\r\n\tdisplay: none !important;\r\n}\r\n\r\n.view_content {\r\n\tpadding-left: 15px;\r\n}\r\n\r\n.view_header {\r\n\tpadding-left: 15px;\r\n\tpadding-top: 10px;\r\n}\r\n\r\n.undo_toast {\r\n\twidth: 95%;\r\n}`\r\n    },\r\n    \"notion\": {\r\n        url: \"https://www.notion.so/\",\r\n        displayName: \"Notion\",\r\n        icon: \"box\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: true,\r\n        openInCenter: true,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    },\r\n    \"twitter\": {\r\n        url: \"https://twitter.com\",\r\n        displayName: \"Twitter\",\r\n        icon: \"twitter\",\r\n        hideOnMobile: true,\r\n        addRibbonIcon: false,\r\n        openInCenter: false,\r\n        zoomLevel: 1,\r\n        forceIframe: false,\r\n        customCss: \"\"\r\n    }\r\n};\r\n\r\nexport interface CustomFramesSettings {\r\n    frames: CustomFrameSettings[];\r\n    padding: number;\r\n}\r\n\r\nexport interface CustomFrameSettings {\r\n    url: string;\r\n    displayName: string;\r\n    icon: string;\r\n    hideOnMobile: boolean;\r\n    addRibbonIcon: boolean;\r\n    openInCenter: boolean;\r\n    zoomLevel: number;\r\n    forceIframe: boolean;\r\n    customCss: string;\r\n}\r\n\r\nexport function getIcon(settings: CustomFrameSettings) {\r\n    return settings.icon ? `lucide-${settings.icon}` : \"documents\";\r\n}\r\n\r\nexport function getId(settings: CustomFrameSettings) {\r\n    return settings.displayName.toLowerCase().replace(/\\s/g, \"-\");\r\n}\r\n", "import { App, ButtonComponent, DropdownComponent, PluginSettingTab, Setting } from \"obsidian\";\r\nimport { defaultSettings, presets } from \"./settings\";\r\nimport CustomFramesPlugin from \"./main\";\r\n\r\nexport class CustomFramesSettingTab extends PluginSettingTab {\r\n\r\n    plugin: CustomFramesPlugin;\r\n\r\n    constructor(app: App, plugin: CustomFramesPlugin) {\r\n        super(app, plugin);\r\n        this.plugin = plugin;\r\n    }\r\n\r\n    display(): void {\r\n        this.containerEl.empty();\r\n        this.containerEl.createEl(\"h2\", { text: \"Custom Frames Settings\" });\r\n        this.containerEl.createEl(\"p\", { text: \"Please note that Obsidian has to be restarted or reloaded for most of these settings to take effect.\", cls: \"mod-warning\" });\r\n\r\n        new Setting(this.containerEl)\r\n            .setName(\"Frame Padding\")\r\n            .setDesc(\"The padding that should be left around the inside of custom frame panes, in pixels.\")\r\n            .addText(t => {\r\n                t.inputEl.type = \"number\";\r\n                t.setValue(String(this.plugin.settings.padding));\r\n                t.onChange(async v => {\r\n                    this.plugin.settings.padding = v.length ? Number(v) : defaultSettings.padding;\r\n                    await this.plugin.saveSettings();\r\n                });\r\n            });\r\n\r\n        for (let frame of this.plugin.settings.frames) {\r\n            let heading = this.containerEl.createEl(\"h3\", { text: frame.displayName || \"Unnamed Frame\" });\r\n            let toggle = new ButtonComponent(this.containerEl)\r\n                .setButtonText(\"Show Settings\")\r\n                .setClass(\"custom-frames-show\")\r\n                .onClick(async () => {\r\n                    content.hidden = !content.hidden;\r\n                    toggle.setButtonText(content.hidden ? \"Show Settings\" : \"Hide Settings\");\r\n                });\r\n            let content = this.containerEl.createDiv();\r\n            content.hidden = true;\r\n\r\n            new Setting(content)\r\n                .setName(\"Display Name\")\r\n                .setDesc(\"The display name that this frame should have.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.displayName);\r\n                    t.onChange(async v => {\r\n                        frame.displayName = v;\r\n                        heading.setText(frame.displayName || \"Unnamed Frame\");\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Icon\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"The icon that this frame's pane should have. The names of any \" });\r\n                    f.createEl(\"a\", { text: \"Lucide icons\", href: \"https://lucide.dev/\" });\r\n                    f.createSpan({ text: \" can be used.\" });\r\n                }))\r\n                .addText(t => {\r\n                    t.setValue(frame.icon);\r\n                    t.onChange(async v => {\r\n                        frame.icon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"URL\")\r\n                .setDesc(\"The URL that should be opened in this frame.\")\r\n                .addText(t => {\r\n                    t.setValue(frame.url);\r\n                    t.onChange(async v => {\r\n                        frame.url = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Disable on Mobile\")\r\n                .setDesc(\"Custom Frames is a lot more restricted on mobile devices and doesn't allow for the same types of content to be displayed. If a frame doesn't work as expected on mobile, it can be disabled.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.hideOnMobile);\r\n                    t.onChange(async v => {\r\n                        frame.hideOnMobile = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Add Ribbon Icon\")\r\n                .setDesc(\"Whether a button to open this frame should be added to the ribbon.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.addRibbonIcon);\r\n                    t.onChange(async v => {\r\n                        frame.addRibbonIcon = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Open in Center\")\r\n                .setDesc(\"Whether this frame should be opened in the unpinned center editor rather than one of the panes on the side. This is useful for sites that don't work well in a narrow view, or sites that don't require a note to be open when viewed.\")\r\n                .addToggle(t => {\r\n                    t.setValue(frame.openInCenter);\r\n                    t.onChange(async v => {\r\n                        frame.openInCenter = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Force iframe\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"Whether this frame should use iframes on desktop as opposed to Electron webviews.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Only enable this setting if the frame is causing issues or frequent crashes. This setting causes all Desktop-only settings to be ignored.\" });\r\n                }))\r\n                .addToggle(t => {\r\n                    t.setValue(frame.forceIframe);\r\n                    t.onChange(async v => {\r\n                        frame.forceIframe = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Page Zoom\")\r\n                .setDesc(\"The zoom that this frame's page should be displayed with, as a percentage.\")\r\n                .addText(t => {\r\n                    t.inputEl.type = \"number\";\r\n                    t.setValue(String(frame.zoomLevel * 100));\r\n                    t.onChange(async v => {\r\n                        frame.zoomLevel = v.length ? Number(v) / 100 : 1;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new Setting(content)\r\n                .setName(\"Additional CSS\")\r\n                .setDesc(createFragment(f => {\r\n                    f.createSpan({ text: \"A snippet of additional CSS that should be applied to this frame.\" });\r\n                    f.createEl(\"br\");\r\n                    f.createEl(\"em\", { text: \"Note that this is only applied on Desktop.\" });\r\n                }))\r\n                .addTextArea(t => {\r\n                    t.inputEl.rows = 5;\r\n                    t.inputEl.cols = 50;\r\n                    t.setValue(frame.customCss);\r\n                    t.onChange(async v => {\r\n                        frame.customCss = v;\r\n                        await this.plugin.saveSettings();\r\n                    });\r\n                });\r\n            new ButtonComponent(content)\r\n                .setButtonText(\"Remove Frame\")\r\n                .onClick(async () => {\r\n                    this.plugin.settings.frames.remove(frame);\r\n                    await this.plugin.saveSettings();\r\n                    this.display();\r\n                });\r\n        }\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"Create a new frame, either from a preset shipped with the plugin, or a custom one that you can edit yourself. Each frame's pane can be opened using the \\\"Custom Frames: Open\\\" command.\" });\r\n\r\n        let addDiv = this.containerEl.createDiv();\r\n        let dropdown = new DropdownComponent(addDiv);\r\n        dropdown.addOption(\"new\", \"Custom\");\r\n        for (let key of Object.keys(presets))\r\n            dropdown.addOption(key, presets[key].displayName);\r\n        new ButtonComponent(addDiv)\r\n            .setButtonText(\"Add Frame\")\r\n            .setClass(\"custom-frames-add\")\r\n            .onClick(async () => {\r\n                let option = dropdown.getValue();\r\n                if (option == \"new\") {\r\n                    this.plugin.settings.frames.push({\r\n                        url: \"\",\r\n                        displayName: \"New Frame\",\r\n                        icon: \"\",\r\n                        hideOnMobile: true,\r\n                        addRibbonIcon: false,\r\n                        openInCenter: false,\r\n                        zoomLevel: 1,\r\n                        forceIframe: false,\r\n                        customCss: \"\"\r\n                    });\r\n                } else {\r\n                    this.plugin.settings.frames.push(presets[option]);\r\n                }\r\n                await this.plugin.saveSettings();\r\n                this.display();\r\n            });\r\n\r\n        var disclaimer = this.containerEl.createEl(\"p\", { cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \"Please be advised that, when adding a site as a custom frame, you potentially expose personal information you enter to other plugins you have installed. For more information, see \" });\r\n        disclaimer.createEl(\"a\", { text: \"this discussion\", href: \"https://github.com/Ellpeck/ObsidianCustomFrames/issues/54#issuecomment-1210879685\", cls: \"mod-warning\" });\r\n        disclaimer.createSpan({ text: \".\" });\r\n\r\n        this.containerEl.createEl(\"hr\");\r\n        this.containerEl.createEl(\"p\", { text: \"If you like this plugin and want to support its development, you can do so through my website by clicking this fancy image!\" });\r\n        this.containerEl.createEl(\"a\", { href: \"https://ellpeck.de/support\" })\r\n            .createEl(\"img\", { attr: { src: \"https://ellpeck.de/res/generalsupport.png\" }, cls: \"custom-frames-support\" });\r\n    }\r\n}\r\n", "import { ItemView, WorkspaceLeaf, Menu } from \"obsidian\";\r\nimport { CustomFrame } from \"./frame\";\r\nimport { CustomFrameSettings, CustomFramesSettings, getIcon } from \"./settings\";\r\n\r\nexport class CustomFrameView extends ItemView {\r\n\r\n    private static readonly actions: Action[] = [\r\n        {\r\n            name: \"Return to original page\",\r\n            icon: \"home\",\r\n            action: v => v.frame.return()\r\n        }, {\r\n            name: \"Open dev tools\",\r\n            icon: \"binary\",\r\n            action: v => v.frame.toggleDevTools()\r\n        }, {\r\n            name: \"Copy link\",\r\n            icon: \"link\",\r\n            action: v => navigator.clipboard.writeText(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Open in browser\",\r\n            icon: \"globe\",\r\n            action: v => open(v.frame.getCurrentUrl())\r\n        }, {\r\n            name: \"Refresh\",\r\n            icon: \"refresh-cw\",\r\n            action: v => v.frame.refresh()\r\n        }, {\r\n            name: \"Go back\",\r\n            icon: \"arrow-left\",\r\n            action: v => v.frame.goBack()\r\n        }, {\r\n            name: \"Go forward\",\r\n            icon: \"arrow-right\",\r\n            action: v => v.frame.goForward()\r\n        }\r\n    ];\r\n\r\n    private readonly data: CustomFrameSettings;\r\n    private readonly name: string;\r\n    private frame: CustomFrame;\r\n\r\n    constructor(leaf: WorkspaceLeaf, settings: CustomFramesSettings, data: CustomFrameSettings, name: string) {\r\n        super(leaf);\r\n        this.data = data;\r\n        this.name = name;\r\n        this.frame = new CustomFrame(settings, data);\r\n\r\n        for (let action of CustomFrameView.actions)\r\n            this.addAction(action.icon, action.name, () => action.action(this));\r\n    }\r\n\r\n    onload(): void {\r\n        this.contentEl.empty();\r\n        this.contentEl.addClass(\"custom-frames-view\");\r\n        this.frame.create(this.contentEl);\r\n    }\r\n\r\n    onPaneMenu(menu: Menu, source: string): void {\r\n        super.onPaneMenu(menu, source);\r\n        for (let action of CustomFrameView.actions) {\r\n            menu.addItem(i => {\r\n                i.setTitle(action.name);\r\n                i.setIcon(action.icon);\r\n                i.onClick(() => action.action(this));\r\n            });\r\n        }\r\n    }\r\n\r\n    getViewType(): string {\r\n        return this.name;\r\n    }\r\n\r\n    getDisplayText(): string {\r\n        return this.data.displayName;\r\n    }\r\n\r\n    getIcon(): string {\r\n        return getIcon(this.data);\r\n    }\r\n\r\n    focus(): void {\r\n        this.frame.focus();\r\n    }\r\n}\r\n\r\ninterface Action {\r\n    name: string;\r\n    icon: string;\r\n    action: (view: CustomFrameView) => any;\r\n}\r\n"],
  "mappings": "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",
  "names": []
}
