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.

211 lines
20 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: () => ObsidianCamera
});
var import_obsidian = __toModule(require("obsidian"));
var ObsidianCamera = class extends import_obsidian.Plugin {
onload() {
return __async(this, null, function* () {
this.addRibbonIcon("camera", "Obsidian Camera", (evt) => {
new CameraModal(this.app).open();
});
this.addCommand({
id: "Open camera modal",
name: "Open camera modal / File Picker",
callback: () => {
new CameraModal(this.app).open();
}
});
});
}
};
var CameraModal = class extends import_obsidian.Modal {
constructor(app2) {
super(app2);
}
onOpen() {
return __async(this, null, function* () {
const { contentEl } = this;
const webCamContainer = contentEl.createDiv();
const videoEl = webCamContainer.createEl("video");
videoEl.autoplay = true;
videoEl.muted = true;
const recordVideoButton = webCamContainer.createEl("button", {
text: "Start recording"
});
const switchCameraButton = webCamContainer.createEl("button", {
text: "Switch Camera"
});
const snapPhotoButton = webCamContainer.createEl("button", {
text: "Take a snap"
});
const filePicker = webCamContainer.createEl("input", {
placeholder: "Choose image file from system",
type: "file"
});
filePicker.accept = "image/*,video/*";
filePicker.capture = "camera";
const filePicker2 = webCamContainer.createEl("input", {
placeholder: "Choose image file from system",
type: "file"
});
filePicker2.accept = "image/*";
filePicker2.capture = "camera";
const chosenFolderPath = "attachments/snaps";
const chunks = [];
let recorder = null;
let 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;
}
});
videoStream = yield getVideoStream();
if (!videoStream) {
videoEl.style.display = "none";
snapPhotoButton.style.display = "none";
recordVideoButton.style.display = "none";
switchCameraButton.style.display = "none";
filePicker2.style.display = "block";
}
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 = chosenFolderPath + "/" + fileName;
const folderExists = app.vault.getAbstractFileByPath(chosenFolderPath);
if (!folderExists)
yield app.vault.createFolder(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);
videoStream && videoStream.getTracks().forEach((track) => {
track.stop();
});
this.close();
});
switchCameraButton.onclick = () => __async(this, null, function* () {
cameraIndex = (cameraIndex + 1) % cameras.length;
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 = 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(videoStream, {
mimeType: "video/webm"
});
}
recorder.ondataavailable = (e) => chunks.push(e.data);
recorder.onstop = (e) => __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() {
const { contentEl } = this;
contentEl.empty();
}
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["main.ts"],
  "sourcesContent": ["import { App, MarkdownView, Modal, Notice, Plugin } from \"obsidian\";\n\nexport default class ObsidianCamera extends Plugin {\n\tasync onload() {\n\t\tthis.addRibbonIcon(\"camera\", \"Obsidian Camera\", (evt: MouseEvent) => {\n\t\t\tnew CameraModal(this.app).open();\n\t\t});\n\n\t\tthis.addCommand({\n\t\t\tid: \"Open camera modal\",\n\t\t\tname: \"Open camera modal / File Picker\",\n\t\t\tcallback: () => {\n\t\t\t\tnew CameraModal(this.app).open();\n\t\t\t},\n\t\t});\n\t}\n}\n\nclass CameraModal extends Modal {\n\tconstructor(app: App) {\n\t\tsuper(app);\n\t}\n\n\tasync onOpen() {\n\t\tconst { contentEl } = this;\n\t\tconst webCamContainer = contentEl.createDiv();\n\t\t// const pText = webCamContainer.createEl('p', { text: 'getVideoStream not yet supported on this device.' });\n\t\t// pText.style.display = 'none'\n\t\tconst videoEl = webCamContainer.createEl(\"video\");\n\t\tvideoEl.autoplay = true;\n\t\tvideoEl.muted = true\n\t\tconst recordVideoButton = webCamContainer.createEl(\"button\", {\n\t\t\ttext: \"Start recording\",\n\t\t});\n\t\tconst switchCameraButton = webCamContainer.createEl(\"button\", {\n\t\t\ttext: \"Switch Camera\",\n\t\t});\n\t\tconst snapPhotoButton = webCamContainer.createEl(\"button\", {\n\t\t\ttext: \"Take a snap\",\n\t\t});\n\n\t\tconst filePicker = webCamContainer.createEl(\"input\", {\n\t\t\tplaceholder: \"Choose image file from system\",\n\t\t\ttype: \"file\",\n\t\t});\n\t\tfilePicker.accept = \"image/*,video/*\";\n\t\tfilePicker.capture = \"camera\"; // back camera by default for mobile screens\n\n\t\tconst filePicker2 = webCamContainer.createEl(\"input\", {\n\t\t\tplaceholder: \"Choose image file from system\",\n\t\t\ttype: \"file\",\n\t\t});\n\t\tfilePicker2.accept = \"image/*\";\n\t\tfilePicker2.capture = \"camera\"; // back camera by default for mobile screens\n\t\t// filePicker2.style.display = 'none'\n\n\n\t\tconst chosenFolderPath = \"attachments/snaps\";\n\t\tconst chunks: BlobPart[] = [];\n\t\tlet recorder: MediaRecorder = null;\n\t\tlet videoStream: MediaStream = 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\tvideoStream = await getVideoStream();\n\t\tif (!videoStream) {\n\t\t\tvideoEl.style.display = 'none'\n\t\t\t// pText.style.display = 'block'\n\t\t\tsnapPhotoButton.style.display = 'none'\n\t\t\trecordVideoButton.style.display = 'none'\n\t\t\tswitchCameraButton.style.display = 'none'\n\t\t\tfilePicker2.style.display = 'block'\n\t\t}\n\n\t\tconst handleImageSelectChange = async (file: File) => {\n\t\t\tconst chosenFile = file;\n\t\t\tconst bufferFile = await chosenFile.arrayBuffer();\n\t\t\tsaveFile(bufferFile, false, chosenFile.name.split(' ').join('-'));\n\t\t};\n\t\tfilePicker.onchange = () => handleImageSelectChange(filePicker.files[0])\n\t\tfilePicker2.onchange = () => handleImageSelectChange(filePicker2.files[0])\n\n\n\t\tconst view = this.app.workspace.getActiveViewOfType(MarkdownView);\n\n\t\tconst saveFile = async (file: ArrayBuffer, isImage = false, fileName = '') => {\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\tif (!isImage) new Notice(\"Adding video to vault...\")\n\n\t\t\tconst filePath = chosenFolderPath + \"/\" + fileName;\n\t\t\tconst folderExists =\n\t\t\t\tapp.vault.getAbstractFileByPath(chosenFolderPath);\n\t\t\tif (!folderExists) await app.vault.createFolder(chosenFolderPath);\n\t\t\tconst fileExists =\n\t\t\t\tapp.vault.getAbstractFileByPath(filePath);\n\t\t\tif (!fileExists)\n\t\t\t\tawait 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\tvideoStream && videoStream.getTracks().forEach((track) => {\n\t\t\t\ttrack.stop();\n\t\t\t});\n\t\t\tthis.close(); // closing the modal\n\t\t};\n\n\n\t\tswitchCameraButton.onclick = async () => {\n\t\t\tcameraIndex = (cameraIndex + 1) % cameras.length;\n\t\t\tvideoStream = await getVideoStream();\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.getContext(\"2d\").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 = videoStream;\n\n\t\trecordVideoButton.onclick = async () => {\n\t\t\tswitchCameraButton.disabled = true;\n\t\t\tlet isRecording: boolean =\n\t\t\t\trecorder && recorder.state === \"recording\";\n\t\t\tif (isRecording) recorder.stop();\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\tif (!recorder) {\n\t\t\t\trecorder = new MediaRecorder(videoStream, {\n\t\t\t\t\tmimeType: \"video/webm\",\n\t\t\t\t});\n\t\t\t}\n\n\t\t\trecorder.ondataavailable = (e) => chunks.push(e.data);\n\t\t\trecorder.onstop = async (e) => {\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\trecorder.start();\n\t\t};\n\n\n\t}\n\n\tonClose() {\n\t\tconst { contentEl } = this;\n\t\tcontentEl.empty();\n\t}\n}\n\n"],
  "mappings": "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",
  "names": []
}
