tech-tips/.obsidian/plugins/obsidian-paste-png-to-jpeg/main.js

359 lines
40 KiB
JavaScript

/* 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": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA,sBAGO;;;ACFP,kBAAyB;AAElB,IAAM,QAAQ,CAAE,eAA0B;AAE1C,qBAAqB,MAAa;AACxC,MAAI,OAAO;AACV,YAAQ,IAAK,IAAI,OAAQ,cAAc,MAAM,IAAI,KAAK,GAAG;AAAA;AAAA;AA0BpD,IAAM,OAAO;AAAA,EAEnB,QAAQ,cAAgC;AAEvC,QAAI,QAAkB;AACtB,aAAS,IAAI,GAAG,IAAI,aAAa,QAAQ,IAAI,GAAG,KAAK;AACpD,cAAQ,MAAM,OAAO,aAAa,GAAG,MAAM;AAAA;AAG5C,UAAM,WAAW;AACjB,aAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,IAAI,GAAG,KAAK;AAC7C,YAAM,OAAO,MAAM;AAGnB,UAAI,CAAC,QAAQ,SAAS;AAAK;AAAA;AAEtB,iBAAS,KAAK;AAAA;AAGpB,QAAI,MAAM,OAAO;AAAI,eAAS,QAAQ;AAEtC,WAAO,SAAS,KAAK;AAAA;AAAA,EAItB,SAAS,UAA0B;AAClC,UAAM,KAAK,SAAS,MAAM;AAC1B,WAAO,GAAG,GAAG,SAAS;AAAA;AAAA,EAGvB,SAAS,UACT;AACC,QAAI,WAAW,0BAAS;AAExB,WAAO,SAAS,UAAU,GAAE,SAAS,QAAQ;AAAA;AAAA,EAI9C,UAAU,UAA0B;AACnC,UAAM,YAAY,CAAC,GAAG,SAAS,SAAS,IAAI,OAAO,OAAO,QAAQ,IAAI,OAAK,EAAE;AAC7E,WAAO,SAAS,MAAM,UAAU,UAAU,SAAS,KAAK;AAAA;AAAA;AAwDnD,sBAAsB,MAAW,SAAqC;AACzE,SAAO,IAAI,QAAQ,CAAC,SAAS,WAChC;AACO,QAAI,SAAS,IAAI;AACjB,WAAO,YAAY,SAAU,GACnC;AACU,UAAI,QAAQ,IAAI;AAChB,YAAM,SAAS,WACxB;AACa,YAAI,SAAS,SAAS,cAAc;AACpC,YAAI,UAAU,OAAO,WAAW;AAChC,YAAI,aAAa,MAAM;AACvB,YAAI,cAAc,MAAM;AACxB,YAAI,OAAO;AAEX,eAAO,QAAQ;AACf,eAAO,SAAS;AAE5B,gBAAQ,YAAY;AACpB,gBAAQ,SAAS,GAAG,GAAG,YAAY;AACnC,gBAAQ;AAER,gBAAQ,UAAU,aAAa,GAAG,cAAc;AACpC,gBAAQ,UAAU,OAAO,GAAG,GAAG,YAAY,aAAY,CAAC,aAAW,GAAE,CAAC,cAAY,GAAE,YAAW;AAC3G,gBAAQ;AAEI,eAAO,OAAO,UAAU,cAAa;AAErC,YAAI,cAAc,oBAAoB;AACtC,gBAAQ;AAAA;AAGrB,YAAM,MAAM,EAAE,OAAO,OAAO;AAAA;AAG7B,WAAO,cAAc;AAAA;AAAA;AAIvB,6BAA8B,MAC9B;AACI,QAAM,QAAQ,KAAK,MAAM;AACzB,QAAM,cAAc,MAAM,GAAG,MAAM,KAAK;AACxC,QAAM,UAAU,YAAY,MAAM,KAAK;AACvC,QAAM,MAAM,OAAO,KAAK,MAAM;AAC9B,QAAM,YAAY,IAAI;AAEtB,QAAM,aAAa,IAAI,WAAW;AAElC,WAAS,IAAI,GAAG,IAAI,WAAW,EAAE,GACpC;AACO,eAAW,KAAK,IAAI,WAAW;AAAA;AAEnC,SAAO,WAAW;AAAA;;;ADjKtB,IAAM,mBAAmC;AAAA,EACxC,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,oBAAoB;AAAA,EACpB,YAAY;AAAA,EACZ,UAAS;AAAA,EACT,WAAU;AAAA,EACV,SAAQ;AAAA,EACR,SAAQ;AAAA;AAKT,yCAAkD,uBAAO;AAAA,EAGlD,SACN;AAAA;AACC,YAAM,MAAM;AACZ,cAAQ,IAAI,mBAAmB,IAAI,QAAQ,IAAI;AAC/C,YAAM,KAAK;AAEX,WAAK,cACJ,KAAK,IAAI,MAAM,GAAG,UAAU,CAAC,SAC7B;AACC,YAAI,CAAE,iBAAgB;AACrB;AAED,cAAM,YAAY,KAAK,QAAQ,KAAK,KAAK;AAGzC,YAAI,YAAY;AACf;AAED,YAAI,QAAQ,OACZ;AACC,mBAAS,wBAAwB;AACjC,eAAK,WAAW;AAAA;AAAA;AAMnB,WAAK,cAAc,IAAI,WAAW,KAAK,KAAK;AAAA;AAAA;AAAA,EAGvC,WAAW,MACjB;AAAA;AACC,YAAM,aAAa,KAAK;AACxB,UAAI,CAAC,YACL;AACC,YAAI,uBAAO;AACX;AAAA;AAID,UAAI,UAAiB,MAAM,KAAK,YAAY,MAAM;AAClD,UAAI,KAAK,SAAS,YAAY;AACvB,kBAAU,MAAM,KAAK,gBAAgB,MAAM;AAAA;AAElD,YAAM,aAAoB,WAAW;AAErC,UAAI,UAAU;AACd,UAAI,KAAK,SAAS,UAClB;AACC,cAAM,YAAY,KAAK,IAAI,MAAM,UAAU,0BAA0B,MAAM,KAAK,SAAS;AACzF,cAAM,WAAW,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO;AACrD,YAAI,CAAC,UACL;AACC,gBAAM,KAAK,IAAI,MAAM,aAAa;AAAA;AAGnC,kBAAU;AAAA,aAGX;AACC,kBAAU,KAAK,OAAO,OAAO,CAAE,MAAM,KAAK,SAAS;AAAA;AAGpD,YAAM,aAAa,KAAK;AACxB,UAAI,KAAK,SAAS,WAClB;AACC,YAAI,SAAqB,MAAM,KAAK,IAAI,MAAM,WAAW;AACzD,YAAI,UAAe,IAAI,KAAM,CAAC;AAC9B,YAAI,cAA0B,MAAM,aAAa,SAAS,OAAQ,KAAK,SAAS;AAChF,cAAM,KAAK,IAAI,MAAM,aAAa,MAAK;AAAA;AAIxC,YAAM,WAAW,KAAK,aAAa,MAAM;AAGzC,gBAAS,KAAK,KAAK,SAAS;AAC5B,UACA;AACC,cAAM,KAAK,IAAI,MAAM,OAAO,MAAM;AAAA,eAE5B,KAAP;AAEC,YAAI,uBAAO,oBAAoB,YAAY;AAC3C,cAAM;AAAA;AAGP,YAAM,cAAc,KAAK,aAAa,MAAM;AAC5C,eAAS,gBAAgB,UAAU;AAInC,YAAM,SAAS,KAAK,gBAAiB;AACrC,UAAI,CAAC,QACL;AACC,YAAI,uBAAO,oBAAoB;AAC/B;AAAA;AAGD,YAAM,SAAS,OAAO;AACtB,YAAM,OAAO,OAAO,QAAQ,OAAO;AACnC,eAAS,gBAAgB;AAEzB,aAAO,YAAY;AAAA,QAClB,SAAS;AAAA,UACR;AAAA,YACC,MAAM,iCAAI,SAAJ,EAAY,IAAI;AAAA,YACtB,IAAI,iCAAI,SAAJ,EAAY,IAAI,KAAK;AAAA,YACzB,MAAM,KAAK,QAAQ,UAAU;AAAA;AAAA;AAAA;AAKhC,UAAI,uBAAO,WAAW,iBAAiB;AAAA;AAAA;AAAA,EAGxC,aAAc,MAAa,YAAoB,SAC/C;AACC,WAAO,KAAK,IAAI,YAAY,qBAAqB,MAAM,YAAW;AAAA;AAAA,EAI7D,gBAAgB,MAAa,YACnC;AAAA;AACC,YAAM,UAAU,WAAW,WAAW,MAAM,KAAK;AACjD,YAAM,YAAY,KAAK,SAAS,YAAY,SAAS,KAAK;AAE1D,aAAO,GAAG,WAAW;AAAA;AAAA;AAAA,EAIhB,YAAY,MAAa,YAC/B;AAAA;AACC,YAAM,UAAU,KAAK;AACrB,YAAM,YAAY,KAAK,SAAS,YAAY,SAAS,KAAK;AAE1D,aAAO,GAAG,WAAW;AAAA;AAAA;AAAA,EAGtB,gBACA;AACC,UAAM,OAAO,KAAK,IAAI,UAAU,oBAAoB;AACpD,UAAM,OAAO,6BAAM;AACnB,aAAS,eAAe,6BAAM;AAC9B,WAAO;AAAA;AAAA,EAGR,gBAAgB,YAChB;AACC,UAAM,OAAO,KAAK,IAAI,UAAU,oBAAoB;AACpD,QAAI,MACJ;AACC,UAAI,KAAK,KAAK,QAAQ,YACtB;AACC,eAAO,KAAK;AAAA;AAAA;AAGd,WAAO;AAAA;AAAA,EAGF,eAAe;AAAA;AACpB,WAAK,WAAW,OAAO,OAAO,IAAI,kBAAkB,MAAM,KAAK;AAAA;AAAA;AAAA,EAG1D,eAAe;AAAA;AACpB,YAAM,KAAK,SAAS,KAAK;AAAA;AAAA;AAAA;AAa3B,IAAM,aAAa;AAAA,EAClB;AAAA,EAAO;AAAA,EAAQ;AAAA;AAGhB,iBAAiB,MAA8B;AAC9C,MAAI,gBAAgB,uBAAO;AAC1B,QAAI,WAAW,SAAS,KAAK,UAAU,gBAAgB;AACtD,aAAO;AAAA;AAAA;AAGT,SAAO;AAAA;AAGR,+BAAyB,iCAAiB;AAAA,EAGzC,YAAY,KAAU,QAA8B;AACnD,UAAM,KAAK;AACX,SAAK,SAAS;AAAA;AAAA,EAGf,UAAgB;AACf,UAAM,EAAE,gBAAgB;AACxB,gBAAY;AAEZ,QAAI,wBAAQ,aACV,QAAQ,eACR,QAAQ,8LACR,UAAU,YAAU,OACnB,SAAS,KAAK,OAAO,SAAS,WAC9B,SAAS,CAAO,UAAU;AAC1B,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO;AAAA;AAIrB,QAAI,wBAAQ,aACV,QAAQ,WACR,QAAQ,+DACR,YAAY,YAAU,OACrB,WAAW,EAAC,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,OAAM,SAC9H,SAAS,KAAK,OAAO,SAAS,SAC9B,SAAS,CAAO,UAAU;AAC1B,WAAK,OAAO,SAAS,UAAU;AAC/B,YAAM,KAAK,OAAO;AAAA;AAIrB,QAAI,wBAAQ,aACV,QAAQ,eACR,QAAQ,2LACR,UAAU,YAAU,OACnB,SAAS,KAAK,OAAO,SAAS,YAC9B,SAAS,CAAO,UAAU;AAC1B,WAAK,OAAO,SAAS,aAAa;AAClC,YAAM,KAAK,OAAO;AAAA;AAIrB,QAAI,wBAAQ,aACV,QAAQ,mBACR,QAAQ,mTACR,UAAU,YAAU,OACnB,SAAS,KAAK,OAAO,SAAS,UAC9B,SAAS,CAAO,UAAU;AAC1B,WAAK,OAAO,SAAS,WAAW;AAChC,YAAM,KAAK,OAAO;AAAA;AAAA;AAAA;",
  "names": []
}
