/* THIS IS A GENERATED/BUNDLED FILE BY ESBUILD */ var __create = Object.create; var __defProp = Object.defineProperty; var __defProps = Object.defineProperties; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropDescs = Object.getOwnPropertyDescriptors; var __getOwnPropNames = Object.getOwnPropertyNames; var __getOwnPropSymbols = Object.getOwnPropertySymbols; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __propIsEnum = Object.prototype.propertyIsEnumerable; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __spreadValues = (a, b) => { for (var prop in b || (b = {})) if (__hasOwnProp.call(b, prop)) __defNormalProp(a, prop, b[prop]); if (__getOwnPropSymbols) for (var prop of __getOwnPropSymbols(b)) { if (__propIsEnum.call(b, prop)) __defNormalProp(a, prop, b[prop]); } return a; }; var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b)); var __markAsModule = (target) => __defProp(target, "__esModule", { value: true }); var __commonJS = (cb, mod) => function __require() { return mod || (0, cb[Object.keys(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports; }; var __export = (target, all) => { __markAsModule(target); for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __reExport = (target, module2, desc) => { if (module2 && typeof module2 === "object" || typeof module2 === "function") { for (let key of __getOwnPropNames(module2)) if (!__hasOwnProp.call(target, key) && key !== "default") __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable }); } return target; }; var __toModule = (module2) => { return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2); }; var __async = (__this, __arguments, generator) => { return new Promise((resolve, reject) => { var fulfilled = (value) => { try { step(generator.next(value)); } catch (e) { reject(e); } }; var rejected = (value) => { try { step(generator.throw(value)); } catch (e) { reject(e); } }; var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected); step((generator = generator.apply(__this, __arguments)).next()); }); }; // package.json var require_package = __commonJS({ "package.json"(exports, module2) { module2.exports = { name: "obsidian-paste-png-to-jpeg", version: "1.2.2", main: "main.js", scripts: { dev: "node esbuild.config.mjs", build: "tsc -noEmit -skipLibCheck && BUILD_ENV=production node esbuild.config.mjs && cp manifest.json build", version: "node version-bump.mjs && git add manifest.json versions.json", release: "yarn build && gh release create ${npm_package_version} build/*" }, keywords: [], author: "Reorx", license: "MIT", devDependencies: { "@types/node": "^16.11.6", "@typescript-eslint/eslint-plugin": "^5.2.0", "@typescript-eslint/parser": "^5.2.0", "builtin-modules": "^3.2.0", esbuild: "0.13.12", obsidian: "^0.13.26", tslib: "2.3.1", typescript: "4.4.4" }, dependencies: { "cash-dom": "^8.1.1" } }; } }); // src/main.ts __export(exports, { default: () => PastePngToJpegPlugin }); var import_obsidian = __toModule(require("obsidian")); // src/utils.ts var import_path = __toModule(require("path")); var DEBUG = !(undefined === "production"); function debugLog(...args) { if (DEBUG) { console.log(new Date().toISOString().slice(11, 23), ...args); } } var path = { join(...partSegments) { let parts = []; for (let i = 0, l = partSegments.length; i < l; i++) { parts = parts.concat(partSegments[i].split("/")); } const newParts = []; for (let i = 0, l = parts.length; i < l; i++) { const part = parts[i]; if (!part || part === ".") continue; else newParts.push(part); } if (parts[0] === "") newParts.unshift(""); return newParts.join("/"); }, basename(fullpath) { const sp = fullpath.split("/"); return sp[sp.length - 1]; }, filename(fullpath) { let filename = (0, import_path.basename)(fullpath); return filename.substring(0, filename.indexOf(".")); }, extension(fullpath) { const positions = [...fullpath.matchAll(new RegExp("\\.", "gi"))].map((a) => a.index); return fullpath.slice(positions[positions.length - 1] + 1); } }; function ConvertImage(file, quality) { return new Promise((resolve, reject) => { let reader = new FileReader(); reader.onloadend = function(e) { let image = new Image(); image.onload = function() { let canvas = document.createElement("canvas"); let context = canvas.getContext("2d"); let imageWidth = image.width; let imageHeight = image.height; let data = ""; canvas.width = imageWidth; canvas.height = imageHeight; context.fillStyle = "#fff"; context.fillRect(0, 0, imageWidth, imageHeight); context.save(); context.translate(imageWidth / 2, imageHeight / 2); context.drawImage(image, 0, 0, imageWidth, imageHeight, -imageWidth / 2, -imageHeight / 2, imageWidth, imageHeight); context.restore(); data = canvas.toDataURL("image/jpeg", quality); var arrayBuffer = base64ToArrayBuffer(data); resolve(arrayBuffer); }; image.src = e.target.result.toString(); }; reader.readAsDataURL(file); }); } function base64ToArrayBuffer(code) { const parts = code.split(";base64,"); const contentType = parts[0].split(":")[1]; const fileExt = contentType.split("/")[1]; const raw = window.atob(parts[1]); const rawLength = raw.length; const uInt8Array = new Uint8Array(rawLength); for (let i = 0; i < rawLength; ++i) { uInt8Array[i] = raw.charCodeAt(i); } return uInt8Array.buffer; } // src/main.ts var DEFAULT_SETTINGS = { imageNamePattern: "{{fileName}}", dupNumberAtStart: false, dupNumberDelimiter: "-", autoRename: true, autoMove: true, pngToJpeg: true, quality: "0.6", dirpath: "image/" }; var PastePngToJpegPlugin = class extends import_obsidian.Plugin { onload() { return __async(this, null, function* () { const pkg = require_package(); console.log(`Plugin loading: ${pkg.name} ${pkg.version}`); yield this.loadSettings(); this.registerEvent(this.app.vault.on("create", (file) => { if (!(file instanceof import_obsidian.TFile)) return; const timeGapMs = Date.now() - file.stat.ctime; if (timeGapMs > 1e3) return; if (isImage(file)) { debugLog("pasted image created", file); this.renameFile(file); } })); this.addSettingTab(new SettingTab(this.app, this)); }); } renameFile(file) { return __async(this, null, function* () { const activeFile = this.getActiveFile(); if (!activeFile) { new import_obsidian.Notice("Error: No active file found."); return; } let newName = yield this.keepOrgName(file, activeFile); if (this.settings.autoRename) { newName = yield this.generateNewName(file, activeFile); } const sourcePath = activeFile.path; let newPath = ""; if (this.settings.autoMove) { const imagePath = this.app.vault.getConfig("attachmentFolderPath") + "/" + this.settings.dirpath; const isCreate = yield this.app.vault.adapter.exists(imagePath); if (!isCreate) { yield this.app.vault.createFolder(imagePath); } newPath = imagePath; } else { newPath = file.parent.path + +"/" + this.settings.dirpath; } const originName = file.name; if (this.settings.pngToJpeg) { let binary = yield this.app.vault.readBinary(file); let imgBlob = new Blob([binary]); let arrayBuffer = yield ConvertImage(imgBlob, Number(this.settings.quality)); yield this.app.vault.modifyBinary(file, arrayBuffer); } const linkText = this.makeLinkText(file, sourcePath); newPath = path.join(newPath, newName); try { yield this.app.vault.rename(file, newPath); } catch (err) { new import_obsidian.Notice(`Failed to rename ${newName}: ${err}`); throw err; } const newLinkText = this.makeLinkText(file, sourcePath); debugLog("replace text", linkText, newLinkText); const editor = this.getActiveEditor(sourcePath); if (!editor) { new import_obsidian.Notice(`Failed to rename ${newName}: no active editor`); return; } const cursor = editor.getCursor(); const line = editor.getLine(cursor.line); debugLog("current line", line); editor.transaction({ changes: [ { from: __spreadProps(__spreadValues({}, cursor), { ch: 0 }), to: __spreadProps(__spreadValues({}, cursor), { ch: line.length }), text: line.replace(linkText, newLinkText) } ] }); new import_obsidian.Notice(`Renamed ${originName} to ${newName}`); }); } makeLinkText(file, sourcePath, subpath) { return this.app.fileManager.generateMarkdownLink(file, sourcePath, subpath); } generateNewName(file, activeFile) { return __async(this, null, function* () { const newName = activeFile.basename + "-" + Date.now(); const extension = this.settings.pngToJpeg ? "jpeg" : file.extension; return `${newName}.${extension}`; }); } keepOrgName(file, activeFile) { return __async(this, null, function* () { const newName = file.basename; const extension = this.settings.pngToJpeg ? "jpeg" : file.extension; return `${newName}.${extension}`; }); } getActiveFile() { const view = this.app.workspace.getActiveViewOfType(import_obsidian.MarkdownView); const file = view == null ? void 0 : view.file; debugLog("active file", file == null ? void 0 : file.path); return file; } getActiveEditor(sourcePath) { const view = this.app.workspace.getActiveViewOfType(import_obsidian.MarkdownView); if (view) { if (view.file.path == sourcePath) { return view.editor; } } return null; } loadSettings() { return __async(this, null, function* () { this.settings = Object.assign({}, DEFAULT_SETTINGS, yield this.loadData()); }); } saveSettings() { return __async(this, null, function* () { yield this.saveData(this.settings); }); } }; var IMAGE_EXTS = [ "jpg", "jpeg", "png" ]; function isImage(file) { if (file instanceof import_obsidian.TFile) { if (IMAGE_EXTS.contains(file.extension.toLowerCase())) { return true; } } return false; } var SettingTab = class extends import_obsidian.PluginSettingTab { constructor(app, plugin) { super(app, plugin); this.plugin = plugin; } display() { const { containerEl } = this; containerEl.empty(); new import_obsidian.Setting(containerEl).setName("Png to Jpeg").setDesc(`Paste images from ClipBoard to notes by copying them through various screenshot software,turn on this feature will automatically convert png to jpeg, and more quality compression volume.`).addToggle((toggle) => toggle.setValue(this.plugin.settings.pngToJpeg).onChange((value) => __async(this, null, function* () { this.plugin.settings.pngToJpeg = value; yield this.plugin.saveSettings(); }))); new import_obsidian.Setting(containerEl).setName("Quality").setDesc(`The smaller the Quality, the greater the compression ratio.`).addDropdown((toggle) => toggle.addOptions({ "0.1": "0.1", "0.2": "0.2", "0.3": "0.3", "0.4": "0.4", "0.5": "0.5", "0.6": "0.6", "0.7": "0.7", "0.8": "0.8", "0.9": "0.9", "1.0": "1.0" }).setValue(this.plugin.settings.quality).onChange((value) => __async(this, null, function* () { this.plugin.settings.quality = value; yield this.plugin.saveSettings(); }))); new import_obsidian.Setting(containerEl).setName("Auto Rename").setDesc(`Automatically names the image with the name of the previous note +'-'+ the current timestamp + '.' + file type, for example, the image in test.md will be named test-1652261724173.jpeg`).addToggle((toggle) => toggle.setValue(this.plugin.settings.autoRename).onChange((value) => __async(this, null, function* () { this.plugin.settings.autoRename = value; yield this.plugin.saveSettings(); }))); new import_obsidian.Setting(containerEl).setName("Auto Move Image").setDesc(`Automatically move images to the image directory,If you do not set the default directory for attachments, then it will be stored directly under the image/ folder in the same directory as the notes, if there is a default directory, then it will be stored under the image/ folder in the Magician directory`).addToggle((toggle) => toggle.setValue(this.plugin.settings.autoMove).onChange((value) => __async(this, null, function* () { this.plugin.settings.autoMove = value; yield this.plugin.saveSettings(); }))); } }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/main.ts", "../src/utils.ts"],
  "sourcesContent": ["import {\r\n  App, Plugin, PluginSettingTab, Setting, TFile, TAbstractFile,\r\n\tMarkdownView, Notice, Vault,\r\n} from 'obsidian';\r\n\r\nimport {\r\n   debugLog, path, ConvertImage\r\n} from './utils';\r\n\r\ninterface PluginSettings {\r\n\t// {{imageNameKey}}-{{DATE:YYYYMMDD}}\r\n\timageNamePattern: string\r\n\tdupNumberAtStart: boolean\r\n\tdupNumberDelimiter: string\r\n\tautoRename: boolean\r\n\tautoMove:boolean\r\n\tpngToJpeg: boolean\r\n\tquality: string\r\n\tdirpath: string\r\n}\r\n\r\nconst DEFAULT_SETTINGS: PluginSettings = {\r\n\timageNamePattern: '{{fileName}}',\r\n\tdupNumberAtStart: false,\r\n\tdupNumberDelimiter: '-',\r\n\tautoRename: true,\r\n\tautoMove:true,\r\n\tpngToJpeg:true,\r\n\tquality:'0.6',\r\n\tdirpath:\"image/\" \r\n}\r\n\r\nconst PASTED_IMAGE_PREFIX = 'Pasted image '\r\n\r\nexport default class PastePngToJpegPlugin extends Plugin {\r\n\tsettings: PluginSettings\r\n\r\n\tasync onload() \r\n\t{\r\n\t\tconst pkg = require('../package.json')\r\n\t\tconsole.log(`Plugin loading: ${pkg.name} ${pkg.version}`)\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tthis.registerEvent(\r\n\t\t\tthis.app.vault.on('create', (file) => \r\n\t\t\t{\r\n\t\t\t\tif (!(file instanceof TFile))\r\n\t\t\t\t\treturn\r\n\r\n\t\t\t\tconst timeGapMs = Date.now() - file.stat.ctime;\r\n\r\n\t\t\t\t// if the pasted image is created more than 1 second ago, ignore it\r\n\t\t\t\tif (timeGapMs > 1000)\r\n\t\t\t\t\treturn\r\n\r\n\t\t\t\tif (isImage(file)) \r\n\t\t\t\t{\r\n\t\t\t\t\tdebugLog('pasted image created', file)\r\n\t\t\t\t\tthis.renameFile(file);\r\n\t\t\t\t} \r\n\t\t\t})\r\n\t\t)\r\n\r\n\t\t// add settings tab\r\n\t\tthis.addSettingTab(new SettingTab(this.app, this));\r\n\t}\r\n\r\n\tasync renameFile(file: TFile) \r\n\t{\r\n\t\tconst activeFile = this.getActiveFile()\r\n\t\tif (!activeFile) \r\n\t\t{\r\n\t\t\tnew Notice('Error: No active file found.')\r\n\t\t\treturn\r\n\t\t}\r\n\r\n\t\t// deduplicate name\r\n\t\tlet newName:string = await this.keepOrgName(file, activeFile);\r\n\t\tif (this.settings.autoRename) {\r\n        \tnewName = await this.generateNewName(file, activeFile);\r\n      \t}\r\n\t\tconst sourcePath:string = activeFile.path;\r\n\r\n\t\tlet newPath = \"\";\r\n\t\tif( this.settings.autoMove )\r\n\t\t{\r\n\t\t\tconst imagePath = this.app.vault.getConfig(\"attachmentFolderPath\") + \"/\" + this.settings.dirpath;\r\n\t\t\tconst isCreate = await this.app.vault.adapter.exists(imagePath);\r\n\t\t\tif( !isCreate )\r\n\t\t\t{\r\n\t\t\t\tawait this.app.vault.createFolder(imagePath);\r\n\t\t\t}\r\n\r\n\t\t\tnewPath = imagePath;\r\n\t\t}\r\n\t\telse\r\n\t\t{\r\n\t\t\tnewPath = file.parent.path + + \"/\" + this.settings.dirpath;\r\n\t\t}\r\n\t\t\r\n\t\tconst originName = file.name;\r\n\t\tif( this.settings.pngToJpeg)\r\n\t\t{\r\n\t\t\tlet binary:ArrayBuffer = await this.app.vault.readBinary(file);\r\n\t\t\tlet imgBlob:Blob = new Blob( [binary] );\r\n\t\t\tlet arrayBuffer:ArrayBuffer = await ConvertImage(imgBlob, Number( this.settings.quality ) );\r\n\t\t\tawait this.app.vault.modifyBinary(file,arrayBuffer);\r\n\t\t}\r\n\r\n\t\t// get origin file link before renaming\r\n\t\tconst linkText = this.makeLinkText(file, sourcePath);\r\n\r\n\t\t// file system operation\r\n\t\tnewPath =path.join(newPath, newName)\r\n\t\ttry \r\n\t\t{\r\n\t\t\tawait this.app.vault.rename(file, newPath);\r\n\t\t} \r\n\t\tcatch (err) \r\n\t\t{\r\n\t\t\tnew Notice(`Failed to rename ${newName}: ${err}`)\r\n\t\t\tthrow err\r\n\t\t}\r\n\r\n\t\tconst newLinkText = this.makeLinkText(file, sourcePath);\r\n\t\tdebugLog('replace text', linkText, newLinkText)\r\n\r\n\t\t// in case fileManager.renameFile may not update the internal link in the active file,\r\n\t\t// we manually replace by manipulating the editor\r\n\t\tconst editor = this.getActiveEditor( sourcePath );\r\n\t\tif (!editor) \r\n\t\t{\r\n\t\t\tnew Notice(`Failed to rename ${newName}: no active editor`)\r\n\t\t\treturn\r\n\t\t}\r\n\r\n\t\tconst cursor = editor.getCursor()\r\n\t\tconst line = editor.getLine(cursor.line)\r\n\t\tdebugLog('current line', line)\r\n\t\t// console.log('editor context', cursor, )\r\n\t\teditor.transaction({\r\n\t\t\tchanges: [\r\n\t\t\t\t{\r\n\t\t\t\t\tfrom: {...cursor, ch: 0},\r\n\t\t\t\t\tto: {...cursor, ch: line.length},\r\n\t\t\t\t\ttext: line.replace(linkText, newLinkText),\r\n\t\t\t\t}\r\n\t\t\t]\r\n\t\t})\r\n\r\n\t\tnew Notice(`Renamed ${originName} to ${newName}`)\r\n\t}\r\n\r\n\tmakeLinkText( file: TFile, sourcePath: string, subpath?:string): string \r\n\t{\r\n\t\treturn this.app.fileManager.generateMarkdownLink(file, sourcePath,subpath)\r\n\t}\r\n\r\n\t// returns a new name for the input file, with extension\r\n\tasync generateNewName(file: TFile, activeFile: TFile):Promise<string>\r\n\t{\r\n\t\tconst newName = activeFile.basename + '-' + Date.now();\r\n\t\tconst extension = this.settings.pngToJpeg ? 'jpeg' : file.extension;\r\n\t\t\r\n\t\treturn `${newName}.${extension}`;\r\n\t}\r\n\t\r\n\t// changes only the extension\r\n\tasync keepOrgName(file: TFile, activeFile: TFile):Promise<string>\r\n\t{\r\n\t\tconst newName = file.basename;\r\n\t\tconst extension = this.settings.pngToJpeg ? 'jpeg' : file.extension;\r\n\t\t\r\n\t\treturn `${newName}.${extension}`;\r\n\t}\r\n\r\n\tgetActiveFile() \r\n\t{\r\n\t\tconst view = this.app.workspace.getActiveViewOfType(MarkdownView)\r\n\t\tconst file = view?.file\r\n\t\tdebugLog('active file', file?.path)\r\n\t\treturn file\r\n\t}\r\n\r\n\tgetActiveEditor(sourcePath:string) \r\n\t{\r\n\t\tconst view = this.app.workspace.getActiveViewOfType(MarkdownView)\r\n\t\tif( view )\r\n\t\t{\r\n\t\t\tif( view.file.path == sourcePath )\r\n\t\t\t{\r\n\t\t\t\treturn view.editor\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn null\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tthis.settings = Object.assign({}, DEFAULT_SETTINGS, 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\r\nfunction isPastedImage(file: TAbstractFile): boolean {\r\n\tif (file instanceof TFile) {\r\n\t\tif (file.name.startsWith(PASTED_IMAGE_PREFIX)) {\r\n\t\t\treturn true\r\n\t\t}\r\n\t}\r\n\treturn false\r\n}\r\n\r\nconst IMAGE_EXTS = [\r\n\t'jpg', 'jpeg', 'png'\r\n]\r\n\r\nfunction isImage(file: TAbstractFile): boolean {\r\n\tif (file instanceof TFile) {\r\n\t\tif (IMAGE_EXTS.contains(file.extension.toLowerCase())) {\r\n\t\t\treturn true\r\n\t\t}\r\n\t}\r\n\treturn false\r\n}\r\n\r\nclass SettingTab extends PluginSettingTab {\r\n\tplugin: PastePngToJpegPlugin;\r\n\r\n\tconstructor(app: App, plugin: PastePngToJpegPlugin) {\r\n\t\tsuper(app, plugin);\r\n\t\tthis.plugin = plugin;\r\n\t}\r\n\r\n\tdisplay(): void {\r\n\t\tconst { containerEl } = this;\r\n\t\tcontainerEl.empty();\r\n\r\n\t\tnew Setting(containerEl)\r\n\t\t\t.setName('Png to Jpeg')\r\n\t\t\t.setDesc(`Paste images from ClipBoard to notes by copying them through various screenshot software,turn on this feature will automatically convert png to jpeg, and more quality compression volume.`)\r\n\t\t\t.addToggle(toggle => toggle\r\n\t\t\t\t.setValue(this.plugin.settings.pngToJpeg)\r\n\t\t\t\t.onChange(async (value) => {\r\n\t\t\t\t\tthis.plugin.settings.pngToJpeg = value;\r\n\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t}\r\n\t\t\t));\t\r\n\t\t\t\r\n\t\tnew Setting(containerEl)\r\n\t\t\t.setName('Quality')\r\n\t\t\t.setDesc(`The smaller the Quality, the greater the compression ratio.`)\r\n\t\t\t.addDropdown(toggle => toggle\r\n\t\t\t\t.addOptions({'0.1':'0.1','0.2':'0.2','0.3':'0.3','0.4':'0.4','0.5':'0.5','0.6':'0.6','0.7':'0.7','0.8':'0.8','0.9':'0.9','1.0':'1.0'})\r\n\t\t\t\t.setValue(this.plugin.settings.quality)\r\n\t\t\t\t.onChange(async (value) => {\r\n\t\t\t\t\tthis.plugin.settings.quality = value;\r\n\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t}\r\n\t\t\t));\t\r\n\t\t\t\r\n\t\tnew Setting(containerEl)\r\n\t\t\t.setName('Auto Rename')\r\n\t\t\t.setDesc(`Automatically names the image with the name of the previous note +'-'+ the current timestamp + '.' + file type, for example, the image in test.md will be named test-1652261724173.jpeg`)\r\n\t\t\t.addToggle(toggle => toggle\r\n\t\t\t\t.setValue(this.plugin.settings.autoRename)\r\n\t\t\t\t.onChange(async (value) => {\r\n\t\t\t\t\tthis.plugin.settings.autoRename = value;\r\n\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t}\r\n\t\t\t));\t\r\n\t\t\t\r\n\t\tnew Setting(containerEl)\r\n\t\t\t.setName('Auto Move Image')\r\n\t\t\t.setDesc(`Automatically move images to the image directory,If you do not set the default directory for attachments, then it will be stored directly under the image/ folder in the same directory as the notes, if there is a default directory, then it will be stored under the image/ folder in the Magician directory`)\r\n\t\t\t.addToggle(toggle => toggle\r\n\t\t\t\t.setValue(this.plugin.settings.autoMove)\r\n\t\t\t\t.onChange(async (value) => {\r\n\t\t\t\t\tthis.plugin.settings.autoMove = value;\r\n\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t}\r\n\t\t\t));\t\t\t\t\r\n\t}\r\n}\r\n", "import { App, Vault } from 'obsidian';\r\nimport { basename } from 'path';\r\n\r\nexport const DEBUG = !(process.env.BUILD_ENV === 'production')\r\n\r\nexport function debugLog(...args: any[]) {\r\n\tif (DEBUG) {\r\n\t\tconsole.log((new Date()).toISOString().slice(11, 23), ...args)\r\n\t}\r\n}\r\n\r\ninterface ElementTreeOptions extends DomElementInfo {\r\n\ttag: keyof HTMLElementTagNameMap\r\n\tchildren?: ElementTreeOptions[]\r\n}\r\n\r\ninterface ElementTree {\r\n\tel: HTMLElement\r\n\tchildren: ElementTree[]\r\n}\r\n\r\nexport function createElementTree(rootEl: HTMLElement, opts: ElementTreeOptions): ElementTree {\r\n\tconst result: ElementTree = {\r\n\t\tel: rootEl.createEl(opts.tag, opts as DomElementInfo),\r\n\t\tchildren: [],\r\n\t}\r\n\tconst children = opts.children || []\r\n\tfor (const child of children) {\r\n\t\tresult.children.push(createElementTree(result.el, child))\r\n\t}\r\n\treturn result\r\n}\r\n\r\nexport const path = {\r\n\t// Credit: @creationix/path.js\r\n\tjoin(...partSegments: string[]): string {\r\n\t\t// Split the inputs into a list of path commands.\r\n\t\tlet parts: string[] = []\r\n\t\tfor (let i = 0, l = partSegments.length; i < l; i++) {\r\n\t\t\tparts = parts.concat(partSegments[i].split('/'))\r\n\t\t}\r\n\t\t// Interpret the path commands to get the new resolved path.\r\n\t\tconst newParts = []\r\n\t\tfor (let i = 0, l = parts.length; i < l; i++) {\r\n\t\t\tconst part = parts[i]\r\n\t\t\t// Remove leading and trailing slashes\r\n\t\t\t// Also remove \".\" segments\r\n\t\t\tif (!part || part === '.') continue\r\n\t\t\t// Push new path segments.\r\n\t\t\telse newParts.push(part)\r\n\t\t}\r\n\t\t// Preserve the initial slash if there was one.\r\n\t\tif (parts[0] === '') newParts.unshift('')\r\n\t\t// Turn back into a single string path.\r\n\t\treturn newParts.join('/')\r\n\t},\r\n\r\n\t// returns the last part of a path, e.g. 'foo.jpg'\r\n\tbasename(fullpath: string): string {\r\n\t\tconst sp = fullpath.split('/')\r\n\t\treturn sp[sp.length - 1]\r\n\t},\r\n\r\n\tfilename(fullpath:string):string\r\n\t{\r\n\t\tlet filename = basename(fullpath);\r\n\r\n\t\treturn filename.substring(0,filename.indexOf('.') )\r\n\t},\r\n\r\n\t// return extension without dot, e.g. 'jpg'\r\n\textension(fullpath: string): string {\r\n\t\tconst positions = [...fullpath.matchAll(new RegExp('\\\\.', 'gi'))].map(a => a.index)\r\n\t\treturn fullpath.slice(positions[positions.length - 1] + 1)\r\n\t},\r\n}\r\n\r\nconst filenameNotAllowedChars = /[^a-zA-Z0-9~`!@$&*()\\-_=+{};'\",<.>? ]/g\r\n\r\nexport const sanitizer = {\r\n\tfilename(s: string): string {\r\n\t\treturn s.replace(filenameNotAllowedChars, '').trim()\r\n\t},\r\n\r\n\tdelimiter(s: string): string {\r\n\t\ts = this.filename(s)\r\n\t\t// use default '-' if no valid chars found\r\n\t\tif (!s) s = '-'\r\n\t\treturn s\r\n\t}\r\n}\r\n\r\n// ref: https://stackoverflow.com/a/6969486/596206\r\nexport function escapeRegExp(s: string) {\r\n    return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\r\n}\r\n\r\n\r\ninterface CompositionState {\r\n\tlock: boolean\r\n}\r\n\r\nexport function lockInputMethodComposition(el: HTMLInputElement): CompositionState {\r\n\tconst state: CompositionState = {\r\n\t\tlock: false,\r\n\t}\r\n\tel.addEventListener('compositionstart', () => {\r\n\t\tstate.lock = true\r\n\t})\r\n\tel.addEventListener('compositionend', () => {\r\n\t\tstate.lock = false\r\n\t})\r\n\treturn state\r\n}\r\n\r\n\r\ninterface VaultConfig {\r\n\tuseMarkdownLinks?: boolean\r\n}\r\n\r\ninterface VaultWithConfig extends Vault {\r\n\tconfig?: VaultConfig,\r\n}\r\n\r\nexport function getVaultConfig(app: App): VaultConfig|null {\r\n\tconst vault = app.vault as VaultWithConfig\r\n\treturn vault.config\r\n}\r\n\r\nexport function ConvertImage(file:Blob, quality:number):Promise<ArrayBuffer> {\r\n    return new Promise((resolve, reject) => \r\n\t{\r\n        let reader = new FileReader(); //\u8BFB\u53D6file\r\n        reader.onloadend = function (e) \r\n\t\t{\r\n            let image = new Image() //\u65B0\u5EFA\u4E00\u4E2Aimg\u6807\u7B7E\uFF08\u8FD8\u6CA1\u5D4C\u5165DOM\u8282\u70B9)\r\n            image.onload = function () \r\n\t\t\t{\r\n                let canvas = document.createElement('canvas');\r\n                let context = canvas.getContext('2d');\r\n                let imageWidth = image.width;\r\n                let imageHeight = image.height;\r\n                let data = '';\r\n\r\n                canvas.width = imageWidth;\r\n                canvas.height = imageHeight;\r\n\r\n\t\t\t\tcontext.fillStyle = '#fff';\r\n\t\t\t\tcontext.fillRect(0, 0, imageWidth, imageHeight);\r\n\t\t\t\tcontext.save();\r\n\r\n\t\t\t\tcontext.translate(imageWidth / 2, imageHeight / 2);\r\n                context.drawImage(image, 0, 0, imageWidth, imageHeight,-imageWidth/2,-imageHeight/2,imageWidth,imageHeight);\r\n\t\t\t\tcontext.restore();\r\n\r\n                data = canvas.toDataURL('image/jpeg',quality);\r\n\r\n                var arrayBuffer = base64ToArrayBuffer(data);\r\n                resolve(arrayBuffer)\r\n            }\r\n\r\n\t\t\timage.src = e.target.result.toString() //\u5C06\u56FE\u7247\u7684\u8DEF\u5F84\u8BBE\u6210file\u8DEF\u5F84\r\n        }\r\n\r\n\t\treader.readAsDataURL(file);\r\n    })\r\n}\r\n\r\nfunction base64ToArrayBuffer (code:string):ArrayBuffer\r\n{\r\n    const parts = code.split(\";base64,\");\r\n    const contentType = parts[0].split(\":\")[1];\r\n    const fileExt = contentType.split(\"/\")[1];\r\n    const raw = window.atob(parts[1]);\r\n    const rawLength = raw.length;\r\n\r\n    const uInt8Array = new Uint8Array(rawLength);\r\n\r\n    for (let i = 0; i < rawLength; ++i) \r\n\t{\r\n        uInt8Array[i] = raw.charCodeAt(i);\r\n    }\r\n    return uInt8Array.buffer;\r\n};"],
  "mappings": "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",
  "names": []
}
