obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
114 lines (110 loc) • 12.3 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initEsm() {
if (globalThis.process) {
return;
}
const browserProcess = {
browser: true,
cwd() {
return '/';
},
env: {},
platform: 'android'
};
globalThis.process = browserProcess;
})();
import {
ButtonComponent,
TextComponent
} from "obsidian";
import {
convertAsyncToSync,
invokeAsyncSafely
} from "../../Async.mjs";
import { CssClass } from "../../CssClass.mjs";
import { noop } from "../../Function.mjs";
import { t } from "../i18n/i18n.mjs";
import {
ModalBase,
showModal
} from "./ModalBase.mjs";
class PromptModal extends ModalBase {
isOkClicked = false;
options;
value;
constructor(options, resolve) {
super(options, resolve, CssClass.PromptModal);
const DEFAULT_OPTIONS = {
app: options.app,
cancelButtonText: t(($) => $.obsidianDevUtils.buttons.cancel),
defaultValue: "",
okButtonText: t(($) => $.obsidianDevUtils.buttons.ok),
placeholder: "",
title: "",
valueValidator: noop
};
this.options = { ...DEFAULT_OPTIONS, ...options };
this.value = options.defaultValue ?? "";
}
onClose() {
super.onClose();
this.resolve(this.isOkClicked ? this.value : null);
}
onOpen() {
super.onOpen();
this.titleEl.setText(this.options.title);
const textComponent = new TextComponent(this.contentEl);
const inputEl = textComponent.inputEl;
const validate = async () => {
const errorMessage = await this.options.valueValidator(inputEl.value);
inputEl.setCustomValidity(errorMessage ?? "");
inputEl.reportValidity();
};
textComponent.setValue(this.value);
textComponent.inputEl.select();
textComponent.setPlaceholder(this.options.placeholder);
inputEl.addClass(CssClass.TextBox);
textComponent.onChange((newValue) => {
this.value = newValue;
});
inputEl.addEventListener("keydown", (event) => {
if (event.key === "Enter") {
this.handleOk(event, textComponent);
} else if (event.key === "Escape") {
this.close();
}
});
inputEl.addEventListener("input", convertAsyncToSync(validate));
inputEl.addEventListener("focus", convertAsyncToSync(validate));
invokeAsyncSafely(validate);
const okButton = new ButtonComponent(this.contentEl);
okButton.setButtonText(this.options.okButtonText);
okButton.setCta();
okButton.onClick((event) => {
this.handleOk(event, textComponent);
});
okButton.setClass(CssClass.OkButton);
const cancelButton = new ButtonComponent(this.contentEl);
cancelButton.setButtonText(this.options.cancelButtonText);
cancelButton.onClick(this.close.bind(this));
cancelButton.setClass(CssClass.CancelButton);
}
handleOk(event, textComponent) {
event.preventDefault();
if (!textComponent.inputEl.checkValidity()) {
return;
}
this.isOkClicked = true;
this.close();
}
}
async function prompt(options) {
return await showModal((resolve) => new PromptModal(options, resolve));
}
export {
prompt
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Modals/Prompt.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Utility for displaying a prompt modal in Obsidian.\n *\n * This module exports a function to display a modal that prompts the user for input. The modal includes \"OK\" and \"Cancel\" buttons.\n */\n\nimport type { App } from 'obsidian';\nimport type { Promisable } from 'type-fest';\n\nimport {\n  ButtonComponent,\n  TextComponent\n} from 'obsidian';\n\nimport type { PromiseResolve } from '../../Async.ts';\nimport type { MaybeReturn } from '../../Type.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.ts';\nimport { t } from '../i18n/i18n.ts';\nimport {\n  ModalBase,\n  showModal\n} from './ModalBase.ts';\n\n/**\n * Options for {@link prompt}.\n */\nexport interface PromptOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A text for the \"Cancel\" button.\n   */\n  cancelButtonText?: string;\n\n  /**\n   * A default value to pre-fill the input field.\n   */\n  defaultValue?: string;\n\n  /**\n   * A text for the \"OK\" button.\n   */\n  okButtonText?: string;\n\n  /**\n   * A placeholder text for the input field.\n   */\n  placeholder?: string;\n\n  /**\n   * A title of the modal.\n   */\n  title?: DocumentFragment | string;\n\n  /**\n   * A function to validate the input value.\n   *\n   * @param value - The input value to validate.\n   * @returns an error message if the value is invalid, or null if the value is valid.\n   */\n  valueValidator?(value: string): Promisable<MaybeReturn<string>>;\n}\n\nclass PromptModal extends ModalBase<null | string, PromptOptions> {\n  private isOkClicked = false;\n  private readonly options: Required<PromptOptions>;\n  private value: string;\n\n  public constructor(options: PromptOptions, resolve: PromiseResolve<null | string>) {\n    super(options, resolve, CssClass.PromptModal);\n    const DEFAULT_OPTIONS: Required<PromptOptions> = {\n      app: options.app,\n      cancelButtonText: t(($) => $.obsidianDevUtils.buttons.cancel),\n      defaultValue: '',\n      okButtonText: t(($) => $.obsidianDevUtils.buttons.ok),\n      placeholder: '',\n      title: '',\n      valueValidator: noop\n    };\n    this.options = { ...DEFAULT_OPTIONS, ...options };\n    this.value = options.defaultValue ?? '';\n  }\n\n  public override onClose(): void {\n    super.onClose();\n    this.resolve(this.isOkClicked ? this.value : null);\n  }\n\n  public override onOpen(): void {\n    super.onOpen();\n    this.titleEl.setText(this.options.title);\n    const textComponent = new TextComponent(this.contentEl);\n    const inputEl = textComponent.inputEl;\n\n    const validate = async (): Promise<void> => {\n      const errorMessage = await this.options.valueValidator(inputEl.value) as string | undefined;\n      inputEl.setCustomValidity(errorMessage ?? '');\n      inputEl.reportValidity();\n    };\n\n    textComponent.setValue(this.value);\n    textComponent.inputEl.select();\n    textComponent.setPlaceholder(this.options.placeholder);\n    inputEl.addClass(CssClass.TextBox);\n    textComponent.onChange((newValue) => {\n      this.value = newValue;\n    });\n    inputEl.addEventListener('keydown', (event: KeyboardEvent) => {\n      if (event.key === 'Enter') {\n        this.handleOk(event, textComponent);\n      } else if (event.key === 'Escape') {\n        this.close();\n      }\n    });\n    inputEl.addEventListener('input', convertAsyncToSync(validate));\n    inputEl.addEventListener('focus', convertAsyncToSync(validate));\n    invokeAsyncSafely(validate);\n    const okButton = new ButtonComponent(this.contentEl);\n    okButton.setButtonText(this.options.okButtonText);\n    okButton.setCta();\n    okButton.onClick((event) => {\n      this.handleOk(event, textComponent);\n    });\n    okButton.setClass(CssClass.OkButton);\n    const cancelButton = new ButtonComponent(this.contentEl);\n    cancelButton.setButtonText(this.options.cancelButtonText);\n    cancelButton.onClick(this.close.bind(this));\n    cancelButton.setClass(CssClass.CancelButton);\n  }\n\n  private handleOk(event: Event, textComponent: TextComponent): void {\n    event.preventDefault();\n    if (!textComponent.inputEl.checkValidity()) {\n      return;\n    }\n\n    this.isOkClicked = true;\n    this.close();\n  }\n}\n\n/**\n * Displays a prompt modal in Obsidian to get user input.\n *\n * @param options - The options for the prompt modal.\n * @returns A {@link Promise} that resolves with the user input or null if the prompt was cancelled.\n */\nexport async function prompt(options: PromptOptions): Promise<null | string> {\n  return await showModal<null | string>((resolve) => new PromptModal(options, resolve));\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAWA;AAAA,EACE;AAAA,EACA;AAAA,OACK;AAKP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,gBAAgB;AACzB,SAAS,YAAY;AACrB,SAAS,SAAS;AAClB;AAAA,EACE;AAAA,EACA;AAAA,OACK;AA6CP,MAAM,oBAAoB,UAAwC;AAAA,EACxD,cAAc;AAAA,EACL;AAAA,EACT;AAAA,EAED,YAAY,SAAwB,SAAwC;AACjF,UAAM,SAAS,SAAS,SAAS,WAAW;AAC5C,UAAM,kBAA2C;AAAA,MAC/C,KAAK,QAAQ;AAAA,MACb,kBAAkB,EAAE,CAAC,MAAM,EAAE,iBAAiB,QAAQ,MAAM;AAAA,MAC5D,cAAc;AAAA,MACd,cAAc,EAAE,CAAC,MAAM,EAAE,iBAAiB,QAAQ,EAAE;AAAA,MACpD,aAAa;AAAA,MACb,OAAO;AAAA,MACP,gBAAgB;AAAA,IAClB;AACA,SAAK,UAAU,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAChD,SAAK,QAAQ,QAAQ,gBAAgB;AAAA,EACvC;AAAA,EAEgB,UAAgB;AAC9B,UAAM,QAAQ;AACd,SAAK,QAAQ,KAAK,cAAc,KAAK,QAAQ,IAAI;AAAA,EACnD;AAAA,EAEgB,SAAe;AAC7B,UAAM,OAAO;AACb,SAAK,QAAQ,QAAQ,KAAK,QAAQ,KAAK;AACvC,UAAM,gBAAgB,IAAI,cAAc,KAAK,SAAS;AACtD,UAAM,UAAU,cAAc;AAE9B,UAAM,WAAW,YAA2B;AAC1C,YAAM,eAAe,MAAM,KAAK,QAAQ,eAAe,QAAQ,KAAK;AACpE,cAAQ,kBAAkB,gBAAgB,EAAE;AAC5C,cAAQ,eAAe;AAAA,IACzB;AAEA,kBAAc,SAAS,KAAK,KAAK;AACjC,kBAAc,QAAQ,OAAO;AAC7B,kBAAc,eAAe,KAAK,QAAQ,WAAW;AACrD,YAAQ,SAAS,SAAS,OAAO;AACjC,kBAAc,SAAS,CAAC,aAAa;AACnC,WAAK,QAAQ;AAAA,IACf,CAAC;AACD,YAAQ,iBAAiB,WAAW,CAAC,UAAyB;AAC5D,UAAI,MAAM,QAAQ,SAAS;AACzB,aAAK,SAAS,OAAO,aAAa;AAAA,MACpC,WAAW,MAAM,QAAQ,UAAU;AACjC,aAAK,MAAM;AAAA,MACb;AAAA,IACF,CAAC;AACD,YAAQ,iBAAiB,SAAS,mBAAmB,QAAQ,CAAC;AAC9D,YAAQ,iBAAiB,SAAS,mBAAmB,QAAQ,CAAC;AAC9D,sBAAkB,QAAQ;AAC1B,UAAM,WAAW,IAAI,gBAAgB,KAAK,SAAS;AACnD,aAAS,cAAc,KAAK,QAAQ,YAAY;AAChD,aAAS,OAAO;AAChB,aAAS,QAAQ,CAAC,UAAU;AAC1B,WAAK,SAAS,OAAO,aAAa;AAAA,IACpC,CAAC;AACD,aAAS,SAAS,SAAS,QAAQ;AACnC,UAAM,eAAe,IAAI,gBAAgB,KAAK,SAAS;AACvD,iBAAa,cAAc,KAAK,QAAQ,gBAAgB;AACxD,iBAAa,QAAQ,KAAK,MAAM,KAAK,IAAI,CAAC;AAC1C,iBAAa,SAAS,SAAS,YAAY;AAAA,EAC7C;AAAA,EAEQ,SAAS,OAAc,eAAoC;AACjE,UAAM,eAAe;AACrB,QAAI,CAAC,cAAc,QAAQ,cAAc,GAAG;AAC1C;AAAA,IACF;AAEA,SAAK,cAAc;AACnB,SAAK,MAAM;AAAA,EACb;AACF;AAQA,eAAsB,OAAO,SAAgD;AAC3E,SAAO,MAAM,UAAyB,CAAC,YAAY,IAAI,YAAY,SAAS,OAAO,CAAC;AACtF;",
  "names": []
}
