You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
265 lines
26 KiB
265 lines
26 KiB
/*
|
|
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.accept = "image/*,video/*";
|
|
filePicker.capture = "camera";
|
|
const filePicker2 = secondRow.createEl("input", {
|
|
placeholder: "Choose image file from system",
|
|
type: "file"
|
|
});
|
|
filePicker2.accept = "image/*";
|
|
filePicker2.capture = "camera";
|
|
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) => __async(this, null, function* () {
|
|
const chosenFile = file;
|
|
const bufferFile = yield chosenFile.arrayBuffer();
|
|
saveFile(bufferFile, false, chosenFile.name.split(" ").join("-"));
|
|
});
|
|
filePicker.onchange = () => handleImageSelectChange(filePicker.files[0]);
|
|
filePicker2.onchange = () => handleImageSelectChange(filePicker2.files[0]);
|
|
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`;
|
|
}
|
|
if (!isImage)
|
|
new import_obsidian.Notice("Adding 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 getVideoStream();
|
|
});
|
|
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;
|
|
let isRecording = recorder && recorder.state === "recording";
|
|
if (isRecording)
|
|
recorder.stop();
|
|
isRecording = !isRecording;
|
|
recordVideoButton.innerText = isRecording ? "Stop Recording" : "Start Recording";
|
|
if (!recorder) {
|
|
recorder = new MediaRecorder(this.videoStream, {
|
|
mimeType: "video/webm"
|
|
});
|
|
}
|
|
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);
|
|
});
|
|
recorder.start();
|
|
});
|
|
});
|
|
}
|
|
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* () {
|
|
console.log("Chosen Folder Path: " + value);
|
|
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  chosenFolderPath: string;\n  videoStream: MediaStream = null;\n  constructor(app: App, cameraSettings: CameraPluginSettings) {\n    super(app);\n    this.chosenFolderPath = cameraSettings.chosenFolderPath\n  }\n\n  async onOpen() {\n    const { contentEl } = this;\n    const webCamContainer = contentEl.createDiv();\n\n    const statusMsg = webCamContainer.createEl('span', { text: \"Loading..\" })\n    const videoEl = webCamContainer.createEl(\"video\");\n    const buttonsDiv = webCamContainer.createDiv();\n    const firstRow = buttonsDiv.createDiv();\n    const secondRow = buttonsDiv.createDiv();\n    const recordVideoButton = firstRow.createEl(\"button\", {\n      text: \"Start recording\",\n    });\n    const switchCameraButton = firstRow.createEl(\"button\", {\n      text: \"Switch Camera\",\n    });\n    const snapPhotoButton = firstRow.createEl(\"button\", {\n      text: \"Take a snap\",\n    });\n    firstRow.style.display = 'none';\n    secondRow.style.display = 'none';\n\n    const filePicker = secondRow.createEl(\"input\", {\n      placeholder: \"Choose image file from system\",\n      type: \"file\",\n    });\n    filePicker.accept = \"image/*,video/*\";\n    filePicker.capture = \"camera\"; // back camera by default for mobile screens\n\n    const filePicker2 = secondRow.createEl(\"input\", {\n      placeholder: \"Choose image file from system\",\n      type: \"file\",\n    });\n    filePicker2.accept = \"image/*\";\n    filePicker2.capture = \"camera\"; // back camera by default for mobile screens\n\n\n    videoEl.autoplay = true;\n    videoEl.muted = true\n    const chunks: BlobPart[] = [];\n    let recorder: MediaRecorder = null;\n    this.videoStream = null;\n\n    const cameras = (\n      await navigator.mediaDevices.enumerateDevices()\n    ).filter((d) => d.kind === \"videoinput\");\n\n    if (cameras.length <= 1) switchCameraButton.style.display = \"none\";\n    let cameraIndex = 0;\n\n    const getVideoStream = async () => {\n      try {\n        return await navigator.mediaDevices.getUserMedia({\n          video: { deviceId: cameras[cameraIndex].deviceId },\n          audio: true,\n        });\n      } catch (error) {\n        console.log(error);\n        return null;\n      }\n    };\n\n    this.videoStream = await getVideoStream();\n    if (this.videoStream) {\n      firstRow.style.display = 'block'\n      secondRow.style.display = 'block'\n      statusMsg.style.display = \"none\"\n    } else {\n      secondRow.style.display = 'block'\n      statusMsg.textContent = \"Error in loading videostream in your device..\"\n    }\n\n    const handleImageSelectChange = async (file: File) => {\n      const chosenFile = file;\n      const bufferFile = await chosenFile.arrayBuffer();\n      saveFile(bufferFile, false, chosenFile.name.split(' ').join('-'));\n    };\n    filePicker.onchange = () => handleImageSelectChange(filePicker.files[0])\n    filePicker2.onchange = () => handleImageSelectChange(filePicker2.files[0])\n\n\n    const view = this.app.workspace.getActiveViewOfType(MarkdownView);\n\n    const saveFile = async (file: ArrayBuffer, isImage = false, fileName = '') => {\n      if (!fileName) {\n        const dateString = (new Date() + \"\")\n          .slice(4, 28)\n          .split(\" \")\n          .join(\"_\")\n          .split(\":\")\n          .join(\"-\");\n        fileName = isImage\n          ? `image_${dateString}.png`\n          : `video_${dateString}.webm`;\n      }\n      if (!isImage) new Notice(\"Adding video to vault...\")\n\n      const filePath = this.chosenFolderPath + \"/\" + fileName;\n      const folderExists =\n        app.vault.getAbstractFileByPath(this.chosenFolderPath);\n      if (!folderExists) await app.vault.createFolder(this.chosenFolderPath);\n      const fileExists =\n        app.vault.getAbstractFileByPath(filePath);\n      if (!fileExists)\n        await app.vault.createBinary(filePath, file);\n\n      if (!view) return new Notice(`Saved to ${filePath}`);\n\n      const cursor = view.editor.getCursor();\n      view.editor.replaceRange(\n        isImage\n          ? `![${fileName}](${filePath})\\n`\n          : `\\n![[${filePath}]]\\n`,\n        cursor\n      );\n      this.close(); // closing the modal\n    };\n\n\n    switchCameraButton.onclick = async () => {\n      cameraIndex = (cameraIndex + 1) % cameras.length;\n      this.videoStream = await getVideoStream();\n    };\n\n    snapPhotoButton.onclick = () => {\n      const canvas = webCamContainer.createEl(\"canvas\");\n      canvas.style.display = \"none\";\n      const { videoHeight, videoWidth } = videoEl\n      canvas.height = videoHeight;\n      canvas.width = videoWidth;\n\n      canvas.getContext(\"2d\").drawImage(videoEl, 0, 0, videoWidth, videoHeight);\n      canvas.toBlob(async (blob) => {\n        const bufferFile = await blob.arrayBuffer();\n        saveFile(bufferFile, true);\n      }, \"image/png\");\n    };\n\n    videoEl.srcObject = this.videoStream;\n\n    recordVideoButton.onclick = async () => {\n      switchCameraButton.disabled = true;\n      let isRecording: boolean =\n        recorder && recorder.state === \"recording\";\n      if (isRecording) recorder.stop();\n      isRecording = !isRecording;\n      recordVideoButton.innerText = isRecording\n        ? \"Stop Recording\"\n        : \"Start Recording\";\n\n      if (!recorder) {\n        recorder = new MediaRecorder(this.videoStream, {\n          mimeType: \"video/webm\",\n        });\n      }\n\n      recorder.ondataavailable = (e) => chunks.push(e.data);\n      recorder.onstop = async (_) => {\n        const blob = new Blob(chunks, {\n          type: \"audio/ogg; codecs=opus\",\n        });\n        const bufferFile = await blob.arrayBuffer();\n        saveFile(bufferFile, false);\n      };\n      recorder.start();\n    };\n\n\n  }\n\n  onClose() {\n    const { contentEl } = this;\n    this.videoStream?.getTracks().forEach(track => {\n      track.stop()\n    })\n    contentEl.empty();\n  }\n}\n\nexport default CameraModal", "\nimport ObsidianCamera from \"main\";\nimport { App, PluginSettingTab, Setting } from \"obsidian\";\n\nexport interface CameraPluginSettings {\n  chosenFolderPath: string;\n\n}\n\nexport const DEFAULT_SETTINGS: CameraPluginSettings = {\n  chosenFolderPath: 'attachments/snaps'\n}\n\nexport default class CameraSettingsTab extends PluginSettingTab {\n  plugin: ObsidianCamera;\n\n  constructor(app: App, plugin: ObsidianCamera) {\n    super(app, plugin);\n    this.plugin = plugin;\n  }\n\n  display(): void {\n    const { containerEl } = this;\n    containerEl.empty();\n    containerEl.createEl('h2', { text: 'Obsidian-Camera settings' });\n\n    new Setting(containerEl)\n      .setName('Folder Path')\n      .setDesc('Folder where the videos and snaps should be saved')\n      .addText(text => text\n        .setPlaceholder('Enter your secret')\n        .setValue(this.plugin.settings.chosenFolderPath)\n        .onChange(async (value) => {\n          console.log('Chosen Folder Path: ' + value);\n          this.plugin.settings.chosenFolderPath = value;\n          await this.plugin.saveSettings();\n        }));\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|