/* THIS IS A GENERATED/BUNDLED FILE BY ESBUILD if you want to view the source, please visit the github repository of this plugin */ var __create = Object.create; var __defProp = Object.defineProperty; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropNames = Object.getOwnPropertyNames; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __markAsModule = (target) => __defProp(target, "__esModule", { value: true }); 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()); }); }; // main.ts __export(exports, { default: () => main_default }); // src/App.ts var import_obsidian3 = __toModule(require("obsidian")); // src/Modal.ts var import_obsidian = __toModule(require("obsidian")); var CameraModal = class extends import_obsidian.Modal { constructor(app2, cameraSettings) { super(app2); this.videoStream = null; this.chosenFolderPath = cameraSettings.chosenFolderPath; } onOpen() { return __async(this, null, function* () { const { contentEl } = this; const webCamContainer = contentEl.createDiv(); const statusMsg = webCamContainer.createEl("span", { text: "Loading.." }); const videoEl = webCamContainer.createEl("video"); const buttonsDiv = webCamContainer.createDiv(); const firstRow = buttonsDiv.createDiv(); const secondRow = buttonsDiv.createDiv(); const recordVideoButton = firstRow.createEl("button", { text: "Start recording" }); const switchCameraButton = firstRow.createEl("button", { text: "Switch Camera" }); const snapPhotoButton = firstRow.createEl("button", { text: "Take a snap" }); firstRow.style.display = "none"; secondRow.style.display = "none"; const filePicker = secondRow.createEl("input", { placeholder: "Choose image file from system", type: "file" }); filePicker.id = "filepicker"; filePicker.accept = "image/*,video/*"; filePicker.capture = "camera"; filePicker.style.display = "none"; const label = secondRow.createEl("label"); label.textContent = "Upload"; label.style.cursor = "pointer"; label.style.display = "inline-block"; label.style.margin = "5px 0px"; label.style.padding = "5px"; label.style.border = "0.5px solid #555"; label.htmlFor = "filepicker"; label.innerHTML = "⇧ Upload"; label.appendChild(filePicker); secondRow.appendChild(label); videoEl.autoplay = true; videoEl.muted = true; const chunks = []; let recorder = null; this.videoStream = null; const cameras = (yield navigator.mediaDevices.enumerateDevices()).filter((d) => d.kind === "videoinput"); if (cameras.length <= 1) switchCameraButton.style.display = "none"; let cameraIndex = 0; const getVideoStream = () => __async(this, null, function* () { try { return yield navigator.mediaDevices.getUserMedia({ video: { deviceId: cameras[cameraIndex].deviceId }, audio: true }); } catch (error) { console.log(error); return null; } }); this.videoStream = yield getVideoStream(); if (this.videoStream) { firstRow.style.display = "block"; secondRow.style.display = "block"; statusMsg.style.display = "none"; } else { secondRow.style.display = "block"; statusMsg.textContent = "Error in loading videostream in your device.."; } const handleImageSelectChange = (file, isImage = true) => __async(this, null, function* () { const chosenFile = file; const bufferFile = yield chosenFile.arrayBuffer(); saveFile(bufferFile, isImage, chosenFile.name.split(" ").join("-")); }); filePicker.onchange = () => { var _a; if (!((_a = filePicker.files) == null ? void 0 : _a.length)) return; const selectedFile = filePicker.files[0]; label.textContent = `Selected: ${selectedFile.name}`; const isImage = selectedFile.type.startsWith("image/"); handleImageSelectChange(selectedFile, isImage); }; const view = this.app.workspace.getActiveViewOfType(import_obsidian.MarkdownView); const saveFile = (file, isImage = false, fileName = "") => __async(this, null, function* () { if (!fileName) { const dateString = (new Date() + "").slice(4, 28).split(" ").join("_").split(":").join("-"); fileName = isImage ? `image_${dateString}.png` : `video_${dateString}.webm`; } new import_obsidian.Notice(`Adding new ${isImage ? "Image" : "Video"} to vault...`); const filePath = this.chosenFolderPath + "/" + fileName; const folderExists = app.vault.getAbstractFileByPath(this.chosenFolderPath); if (!folderExists) yield app.vault.createFolder(this.chosenFolderPath); const fileExists = app.vault.getAbstractFileByPath(filePath); if (!fileExists) yield app.vault.createBinary(filePath, file); if (!view) return new import_obsidian.Notice(`Saved to ${filePath}`); const cursor = view.editor.getCursor(); view.editor.replaceRange(isImage ? `![${fileName}](${filePath}) ` : ` ![[${filePath}]] `, cursor); this.close(); }); switchCameraButton.onclick = () => __async(this, null, function* () { cameraIndex = (cameraIndex + 1) % cameras.length; this.videoStream = yield navigator.mediaDevices.getUserMedia({ video: { deviceId: cameras[cameraIndex].deviceId }, audio: true }); videoEl.srcObject = this.videoStream; videoEl.play(); }); snapPhotoButton.onclick = () => { const canvas = webCamContainer.createEl("canvas"); canvas.style.display = "none"; const { videoHeight, videoWidth } = videoEl; canvas.height = videoHeight; canvas.width = videoWidth; canvas.getContext("2d").drawImage(videoEl, 0, 0, videoWidth, videoHeight); canvas.toBlob((blob) => __async(this, null, function* () { const bufferFile = yield blob.arrayBuffer(); saveFile(bufferFile, true); }), "image/png"); }; videoEl.srcObject = this.videoStream; recordVideoButton.onclick = () => __async(this, null, function* () { switchCameraButton.disabled = true; if (!recorder) { recorder = new MediaRecorder(this.videoStream, { mimeType: "video/webm" }); } let isRecording = recorder && recorder.state === "recording"; if (isRecording) { recorder.stop(); } else { recorder.start(); } isRecording = !isRecording; recordVideoButton.innerText = isRecording ? "Stop Recording" : "Start Recording"; recorder.ondataavailable = (e) => chunks.push(e.data); recorder.onstop = (_) => __async(this, null, function* () { const blob = new Blob(chunks, { type: "audio/ogg; codecs=opus" }); const bufferFile = yield blob.arrayBuffer(); saveFile(bufferFile, false); }); }); }); } onClose() { var _a; const { contentEl } = this; (_a = this.videoStream) == null ? void 0 : _a.getTracks().forEach((track) => { track.stop(); }); contentEl.empty(); } }; var Modal_default = CameraModal; // src/SettingsTab.ts var import_obsidian2 = __toModule(require("obsidian")); var DEFAULT_SETTINGS = { chosenFolderPath: "attachments/snaps" }; var CameraSettingsTab = class extends import_obsidian2.PluginSettingTab { constructor(app2, plugin) { super(app2, plugin); this.plugin = plugin; } display() { const { containerEl } = this; containerEl.empty(); containerEl.createEl("h2", { text: "Obsidian-Camera settings" }); new import_obsidian2.Setting(containerEl).setName("Folder Path").setDesc("Folder where the videos and snaps should be saved").addText((text) => text.setPlaceholder("Enter your secret").setValue(this.plugin.settings.chosenFolderPath).onChange((value) => __async(this, null, function* () { this.plugin.settings.chosenFolderPath = value; yield this.plugin.saveSettings(); }))); } }; // src/App.ts var ObsidianCamera = class extends import_obsidian3.Plugin { onload() { return __async(this, null, function* () { yield this.loadSettings(); this.addRibbonIcon("camera", "Obsidian Camera", (evt) => { new Modal_default(this.app, this.settings).open(); }); this.addSettingTab(new CameraSettingsTab(this.app, this)); this.addCommand({ id: "Open camera modal", name: "Open camera modal / File Picker", callback: () => { new Modal_default(this.app, this.settings).open(); } }); }); } 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); }); } }; // main.ts var main_default = ObsidianCamera; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["main.ts", "src/App.ts", "src/Modal.ts", "src/SettingsTab.ts"],
  "sourcesContent": ["import ObsidianCamera from \"src/App\";\n\nexport default ObsidianCamera", "import { Plugin } from \"obsidian\";\nimport CameraModal from \"./Modal\";\nimport CameraSettingsTab, { DEFAULT_SETTINGS, CameraPluginSettings } from \"./SettingsTab\";\n\nexport default class ObsidianCamera extends Plugin {\n  settings: CameraPluginSettings;\n  async onload() {\n    await this.loadSettings();\n    this.addRibbonIcon(\"camera\", \"Obsidian Camera\", (evt: MouseEvent) => {\n      new CameraModal(this.app, this.settings).open();\n    });\n    this.addSettingTab(new CameraSettingsTab(this.app, this));\n\n    this.addCommand({\n      id: \"Open camera modal\",\n      name: \"Open camera modal / File Picker\",\n      callback: () => {\n        new CameraModal(this.app, this.settings).open();\n      },\n    });\n  }\n\n\n  async loadSettings() {\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\n  }\n\n  async saveSettings() {\n    await this.saveData(this.settings);\n  }\n}\n", "import { App, MarkdownView, Modal, Notice } from \"obsidian\";\nimport { CameraPluginSettings } from \"./SettingsTab\";\n\nclass CameraModal extends Modal {\n\tchosenFolderPath: string;\n\tvideoStream: MediaStream = null;\n\tconstructor(app: App, cameraSettings: CameraPluginSettings) {\n\t\tsuper(app);\n\t\tthis.chosenFolderPath = cameraSettings.chosenFolderPath;\n\t}\n\n\tasync onOpen() {\n\t\tconst { contentEl } = this;\n\t\tconst webCamContainer = contentEl.createDiv();\n\n\t\tconst statusMsg = webCamContainer.createEl(\"span\", {\n\t\t\ttext: \"Loading..\",\n\t\t});\n\t\tconst videoEl = webCamContainer.createEl(\"video\");\n\t\tconst buttonsDiv = webCamContainer.createDiv();\n\t\tconst firstRow = buttonsDiv.createDiv();\n\t\tconst secondRow = buttonsDiv.createDiv();\n\t\tconst recordVideoButton = firstRow.createEl(\"button\", {\n\t\t\ttext: \"Start recording\",\n\t\t});\n\t\tconst switchCameraButton = firstRow.createEl(\"button\", {\n\t\t\ttext: \"Switch Camera\",\n\t\t});\n\t\tconst snapPhotoButton = firstRow.createEl(\"button\", {\n\t\t\ttext: \"Take a snap\",\n\t\t});\n\t\tfirstRow.style.display = \"none\";\n\t\tsecondRow.style.display = \"none\";\n\n\t\tconst filePicker = secondRow.createEl(\"input\", {\n\t\t\tplaceholder: \"Choose image file from system\",\n\t\t\ttype: \"file\",\n\t\t});\n\t\tfilePicker.id = \"filepicker\";\n\t\tfilePicker.accept = \"image/*,video/*\";\n\t\tfilePicker.capture = \"camera\"; // back camera by default for mobile screens\n\n\t\tfilePicker.style.display = \"none\";\n\n\t\tconst label = secondRow.createEl(\"label\");\n\t\tlabel.textContent = \"Upload\";\n\t\tlabel.style.cursor = \"pointer\";\n\t\tlabel.style.display = \"inline-block\";\n\t\tlabel.style.margin = \"5px 0px\";\n\t\tlabel.style.padding = \"5px\";\n\t\tlabel.style.border = \"0.5px solid #555\";\n\t\tlabel.htmlFor = \"filepicker\";\n\t\tlabel.innerHTML = \"&#8679; Upload\";\n\n\t\tlabel.appendChild(filePicker);\n\n\t\tsecondRow.appendChild(label);\n\n\t\tvideoEl.autoplay = true;\n\t\tvideoEl.muted = true;\n\t\tconst chunks: BlobPart[] = [];\n\t\tlet recorder: MediaRecorder = null;\n\t\tthis.videoStream = null;\n\n\t\tconst cameras = (\n\t\t\tawait navigator.mediaDevices.enumerateDevices()\n\t\t).filter((d) => d.kind === \"videoinput\");\n\n\t\tif (cameras.length <= 1) switchCameraButton.style.display = \"none\";\n\t\tlet cameraIndex = 0;\n\n\t\tconst getVideoStream = async () => {\n\t\t\ttry {\n\t\t\t\treturn await navigator.mediaDevices.getUserMedia({\n\t\t\t\t\tvideo: { deviceId: cameras[cameraIndex].deviceId },\n\t\t\t\t\taudio: true,\n\t\t\t\t});\n\t\t\t} catch (error) {\n\t\t\t\tconsole.log(error);\n\t\t\t\treturn null;\n\t\t\t}\n\t\t};\n\n\t\tthis.videoStream = await getVideoStream();\n\t\tif (this.videoStream) {\n\t\t\tfirstRow.style.display = \"block\";\n\t\t\tsecondRow.style.display = \"block\";\n\t\t\tstatusMsg.style.display = \"none\";\n\t\t} else {\n\t\t\tsecondRow.style.display = \"block\";\n\t\t\tstatusMsg.textContent =\n\t\t\t\t\"Error in loading videostream in your device..\";\n\t\t}\n\n\t\tconst handleImageSelectChange = async (\n\t\t\tfile: File,\n\t\t\tisImage: boolean = true\n\t\t) => {\n\t\t\tconst chosenFile = file;\n\t\t\tconst bufferFile = await chosenFile.arrayBuffer();\n\t\t\tsaveFile(bufferFile, isImage, chosenFile.name.split(\" \").join(\"-\"));\n\t\t};\n\n\t\tfilePicker.onchange = () => {\n\t\t\tif (!filePicker.files?.length) return;\n\t\t\tconst selectedFile = filePicker.files[0];\n\t\t\tlabel.textContent = `Selected: ${selectedFile.name}`;\n\t\t\tconst isImage = selectedFile.type.startsWith(\"image/\");\n\t\t\thandleImageSelectChange(selectedFile, isImage);\n\t\t};\n\n\t\tconst view = this.app.workspace.getActiveViewOfType(MarkdownView);\n\n\t\tconst saveFile = async (\n\t\t\tfile: ArrayBuffer,\n\t\t\tisImage = false,\n\t\t\tfileName = \"\"\n\t\t) => {\n\t\t\tif (!fileName) {\n\t\t\t\tconst dateString = (new Date() + \"\")\n\t\t\t\t\t.slice(4, 28)\n\t\t\t\t\t.split(\" \")\n\t\t\t\t\t.join(\"_\")\n\t\t\t\t\t.split(\":\")\n\t\t\t\t\t.join(\"-\");\n\t\t\t\tfileName = isImage\n\t\t\t\t\t? `image_${dateString}.png`\n\t\t\t\t\t: `video_${dateString}.webm`;\n\t\t\t}\n\t\t\tnew Notice(`Adding new ${isImage ? \"Image\" : \"Video\"} to vault...`);\n\n\t\t\tconst filePath = this.chosenFolderPath + \"/\" + fileName;\n\t\t\tconst folderExists = app.vault.getAbstractFileByPath(\n\t\t\t\tthis.chosenFolderPath\n\t\t\t);\n\t\t\tif (!folderExists)\n\t\t\t\tawait app.vault.createFolder(this.chosenFolderPath);\n\t\t\tconst fileExists = app.vault.getAbstractFileByPath(filePath);\n\t\t\tif (!fileExists) await app.vault.createBinary(filePath, file);\n\n\t\t\tif (!view) return new Notice(`Saved to ${filePath}`);\n\n\t\t\tconst cursor = view.editor.getCursor();\n\t\t\tview.editor.replaceRange(\n\t\t\t\tisImage\n\t\t\t\t\t? `![${fileName}](${filePath})\\n`\n\t\t\t\t\t: `\\n![[${filePath}]]\\n`,\n\t\t\t\tcursor\n\t\t\t);\n\t\t\tthis.close();\n\t\t};\n\n\t\tswitchCameraButton.onclick = async () => {\n\t\t\tcameraIndex = (cameraIndex + 1) % cameras.length;\n\t\t\tthis.videoStream = await navigator.mediaDevices.getUserMedia({\n\t\t\t\tvideo: { deviceId: cameras[cameraIndex].deviceId },\n\t\t\t\taudio: true,\n\t\t\t});\n\t\t\tvideoEl.srcObject = this.videoStream;\n\t\t\tvideoEl.play();\n\t\t};\n\n\t\tsnapPhotoButton.onclick = () => {\n\t\t\tconst canvas = webCamContainer.createEl(\"canvas\");\n\t\t\tcanvas.style.display = \"none\";\n\t\t\tconst { videoHeight, videoWidth } = videoEl;\n\t\t\tcanvas.height = videoHeight;\n\t\t\tcanvas.width = videoWidth;\n\n\t\t\tcanvas\n\t\t\t\t.getContext(\"2d\")\n\t\t\t\t.drawImage(videoEl, 0, 0, videoWidth, videoHeight);\n\t\t\tcanvas.toBlob(async (blob) => {\n\t\t\t\tconst bufferFile = await blob.arrayBuffer();\n\t\t\t\tsaveFile(bufferFile, true);\n\t\t\t}, \"image/png\");\n\t\t};\n\n\t\tvideoEl.srcObject = this.videoStream;\n\n\t\trecordVideoButton.onclick = async () => {\n\t\t\tswitchCameraButton.disabled = true;\n\t\t\tif (!recorder) {\n\t\t\t\trecorder = new MediaRecorder(this.videoStream, {\n\t\t\t\t\tmimeType: \"video/webm\",\n\t\t\t\t});\n\t\t\t}\n\n\t\t\tlet isRecording: boolean =\n\t\t\t\trecorder && recorder.state === \"recording\";\n\t\t\tif (isRecording) {\n\t\t\t\trecorder.stop();\n\t\t\t} else {\n\t\t\t\trecorder.start();\n\t\t\t}\n\t\t\tisRecording = !isRecording;\n\t\t\trecordVideoButton.innerText = isRecording\n\t\t\t\t? \"Stop Recording\"\n\t\t\t\t: \"Start Recording\";\n\n\t\t\trecorder.ondataavailable = (e) => chunks.push(e.data);\n\t\t\trecorder.onstop = async (_) => {\n\t\t\t\tconst blob = new Blob(chunks, {\n\t\t\t\t\ttype: \"audio/ogg; codecs=opus\",\n\t\t\t\t});\n\t\t\t\tconst bufferFile = await blob.arrayBuffer();\n\t\t\t\tsaveFile(bufferFile, false);\n\t\t\t};\n\t\t};\n\t}\n\n\tonClose() {\n\t\tconst { contentEl } = this;\n\t\tthis.videoStream?.getTracks().forEach((track) => {\n\t\t\ttrack.stop();\n\t\t});\n\t\tcontentEl.empty();\n\t}\n}\n\nexport default CameraModal;\n", "import ObsidianCamera from \"main\";\nimport { App, PluginSettingTab, Setting } from \"obsidian\";\n\nexport interface CameraPluginSettings {\n\tchosenFolderPath: string;\n}\n\nexport const DEFAULT_SETTINGS: CameraPluginSettings = {\n\tchosenFolderPath: \"attachments/snaps\",\n};\n\nexport default class CameraSettingsTab extends PluginSettingTab {\n\tplugin: ObsidianCamera;\n\n\tconstructor(app: App, plugin: ObsidianCamera) {\n\t\tsuper(app, plugin);\n\t\tthis.plugin = plugin;\n\t}\n\n\tdisplay(): void {\n\t\tconst { containerEl } = this;\n\t\tcontainerEl.empty();\n\t\tcontainerEl.createEl(\"h2\", { text: \"Obsidian-Camera settings\" });\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName(\"Folder Path\")\n\t\t\t.setDesc(\"Folder where the videos and snaps should be saved\")\n\t\t\t.addText((text) =>\n\t\t\t\ttext\n\t\t\t\t\t.setPlaceholder(\"Enter your secret\")\n\t\t\t\t\t.setValue(this.plugin.settings.chosenFolderPath)\n\t\t\t\t\t.onChange(async (value) => {\n\t\t\t\t\t\tthis.plugin.settings.chosenFolderPath = value;\n\t\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t\t})\n\t\t\t);\n\t}\n}\n"],
  "mappings": "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",
  "names": []
}
