obsidian-dev-utils 26.1.3-beta.9 → 26.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -16,21 +16,85 @@ import {
16
16
  ToggleComponent
17
17
  } from "obsidian";
18
18
  import { CssClass } from "../../../CssClass.mjs";
19
+ import {
20
+ onAncestorScrollOrResize,
21
+ toPx
22
+ } from "../../../HTMLElement.mjs";
19
23
  class OverlayValidatorComponent {
24
+ constructor(el) {
25
+ this.el = el;
26
+ if (!el.parentElement) {
27
+ throw new Error("Element must be attached to the DOM");
28
+ }
29
+ this._validatorEl = el.parentElement.createEl("input", {
30
+ attr: {
31
+ tabindex: -1
32
+ },
33
+ cls: [CssClass.LibraryName, CssClass.OverlayValidator]
34
+ });
35
+ this._validatorEl.addEventListener("focus", () => {
36
+ this.el.focus();
37
+ });
38
+ this._validatorEl.isActiveElement = this.isElementOrDescendantActive.bind(this);
39
+ let tabIndexEl = this.el.querySelector("[tabindex]");
40
+ if (!tabIndexEl) {
41
+ if (this.el.getAttr("tabindex") === null) {
42
+ this.el.tabIndex = -1;
43
+ }
44
+ tabIndexEl = this.el;
45
+ }
46
+ this.el.addEventListener("focusin", () => {
47
+ this.forceBlurValidatorEl();
48
+ });
49
+ this.el.addEventListener("click", () => {
50
+ tabIndexEl.focus();
51
+ });
52
+ this.el.addEventListener("focusout", () => {
53
+ setTimeout(() => {
54
+ if (this.isElementOrDescendantActive()) {
55
+ return;
56
+ }
57
+ this.forceBlurValidatorEl();
58
+ }, 0);
59
+ });
60
+ const unregisterScrollOrResizeHandlers = onAncestorScrollOrResize(this.el, this.updatePosition.bind(this));
61
+ const observer = new MutationObserver((mutations) => {
62
+ for (const mutation of mutations) {
63
+ for (const removedNode of Array.from(mutation.removedNodes)) {
64
+ if (removedNode === this._validatorEl) {
65
+ unregisterScrollOrResizeHandlers();
66
+ observer.disconnect();
67
+ return;
68
+ }
69
+ }
70
+ }
71
+ });
72
+ observer.observe(document.body, {
73
+ childList: true,
74
+ subtree: true
75
+ });
76
+ }
20
77
  get validatorEl() {
21
78
  return this._validatorEl;
22
79
  }
23
80
  _validatorEl;
24
- constructor(el) {
25
- const rect = el.getBoundingClientRect();
26
- this._validatorEl = document.body.createEl("input", {
27
- cls: [CssClass.LibraryName, CssClass.OverlayValidator]
28
- });
81
+ forceBlurValidatorEl() {
82
+ this._validatorEl.dispatchEvent(new Event("blur"));
83
+ }
84
+ isElementOrDescendantActive() {
85
+ return this.el.contains(document.activeElement);
86
+ }
87
+ updatePosition() {
88
+ if (!this.el.offsetParent) {
89
+ return;
90
+ }
91
+ const rect = this.el.getBoundingClientRect();
92
+ const parentRect = this.el.offsetParent.getBoundingClientRect();
29
93
  this._validatorEl.setCssStyles({
30
- height: `${rect.height.toString()}px`,
31
- left: `${(rect.left + window.scrollX).toString()}px`,
32
- top: `${(rect.top + window.scrollY).toString()}px`,
33
- width: `${rect.width.toString()}px`
94
+ height: toPx(rect.height),
95
+ left: toPx(rect.left - parentRect.left + this.el.offsetParent.scrollLeft),
96
+ top: toPx(rect.top - parentRect.top + this.el.offsetParent.scrollTop),
97
+ width: toPx(rect.width)
34
98
  });
35
99
  }
36
100
  }
@@ -75,4 +139,4 @@ function isValidatorComponent(obj) {
75
139
  export {
76
140
  getValidatorComponent
77
141
  };
78
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL0NvbXBvbmVudHMvU2V0dGluZ0NvbXBvbmVudHMvVmFsaWRhdG9yQ29tcG9uZW50LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEBwYWNrYWdlRG9jdW1lbnRhdGlvblxuICpcbiAqIENvbnRhaW5zIGEgY29tcG9uZW50IHRoYXQgaGFzIGEgdmFsaWRhdG9yIGVsZW1lbnQuXG4gKi9cblxuaW1wb3J0IHtcbiAgQ29sb3JDb21wb25lbnQsXG4gIERyb3Bkb3duQ29tcG9uZW50LFxuICBQcm9ncmVzc0JhckNvbXBvbmVudCxcbiAgU2VhcmNoQ29tcG9uZW50LFxuICBTbGlkZXJDb21wb25lbnQsXG4gIFRleHRBcmVhQ29tcG9uZW50LFxuICBUZXh0Q29tcG9uZW50LFxuICBUb2dnbGVDb21wb25lbnRcbn0gZnJvbSAnb2JzaWRpYW4nO1xuXG5pbXBvcnQgdHlwZSB7IFZhbGlkYXRvckVsZW1lbnQgfSBmcm9tICcuLi8uLi8uLi9IVE1MRWxlbWVudC50cyc7XG5cbmltcG9ydCB7IENzc0NsYXNzIH0gZnJvbSAnLi4vLi4vLi4vQ3NzQ2xhc3MudHMnO1xuXG4vKipcbiAqIEEgY29tcG9uZW50IHRoYXQgaGFzIGEgdmFsaWRhdG9yIGVsZW1lbnQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVmFsaWRhdG9yQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIFRoZSB2YWxpZGF0b3IgZWxlbWVudCBvZiB0aGUgY29tcG9uZW50LlxuICAgKi9cbiAgcmVhZG9ubHkgdmFsaWRhdG9yRWw6IFZhbGlkYXRvckVsZW1lbnQ7XG59XG5cbmNsYXNzIE92ZXJsYXlWYWxpZGF0b3JDb21wb25lbnQgaW1wbGVtZW50cyBWYWxpZGF0b3JDb21wb25lbnQge1xuICBwdWJsaWMgZ2V0IHZhbGlkYXRvckVsKCk6IFZhbGlkYXRvckVsZW1lbnQge1xuICAgIHJldHVybiB0aGlzLl92YWxpZGF0b3JFbDtcbiAgfVxuXG4gIHByaXZhdGUgcmVhZG9ubHkgX3ZhbGlkYXRvckVsOiBWYWxpZGF0b3JFbGVtZW50O1xuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcihlbDogSFRNTEVsZW1lbnQpIHtcbiAgICBjb25zdCByZWN0ID0gZWwuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KCk7XG5cbiAgICB0aGlzLl92YWxpZGF0b3JFbCA9IGRvY3VtZW50LmJvZHkuY3JlYXRlRWwoJ2lucHV0Jywge1xuICAgICAgY2xzOiBbQ3NzQ2xhc3MuTGlicmFyeU5hbWUsIENzc0NsYXNzLk92ZXJsYXlWYWxpZGF0b3JdXG4gICAgfSk7XG5cbiAgICB0aGlzLl92YWxpZGF0b3JFbC5zZXRDc3NTdHlsZXMoe1xuICAgICAgaGVpZ2h0OiBgJHtyZWN0LmhlaWdodC50b1N0cmluZygpfXB4YCxcbiAgICAgIGxlZnQ6IGAkeyhyZWN0LmxlZnQgKyB3aW5kb3cuc2Nyb2xsWCkudG9TdHJpbmcoKX1weGAsXG4gICAgICB0b3A6IGAkeyhyZWN0LnRvcCArIHdpbmRvdy5zY3JvbGxZKS50b1N0cmluZygpfXB4YCxcbiAgICAgIHdpZHRoOiBgJHtyZWN0LndpZHRoLnRvU3RyaW5nKCl9cHhgXG4gICAgfSk7XG4gIH1cbn1cblxuY2xhc3MgVmFsaWRhdG9yRWxlbWVudFdyYXBwZXIgaW1wbGVtZW50cyBWYWxpZGF0b3JDb21wb25lbnQge1xuICBwdWJsaWMgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHZhbGlkYXRvckVsOiBWYWxpZGF0b3JFbGVtZW50KSB7fVxufVxuXG4vKipcbiAqIEdldHMgYSB2YWxpZGF0b3IgY29tcG9uZW50IHJlbGF0ZWQgdG8gdGhlIGdpdmVuIG9iamVjdC5cbiAqXG4gKiBAcGFyYW0gb2JqIC0gQW55IG9iamVjdC5cbiAqIEByZXR1cm5zIFRoZSByZWxhdGVkIHZhbGlkYXRvciBjb21wb25lbnQgb3IgYG51bGxgIGlmIG5vIHJlbGF0ZWQgdmFsaWRhdG9yIGNvbXBvbmVudCBpcyBmb3VuZC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFZhbGlkYXRvckNvbXBvbmVudChvYmo6IHVua25vd24pOiBudWxsIHwgVmFsaWRhdG9yQ29tcG9uZW50IHtcbiAgaWYgKGlzVmFsaWRhdG9yQ29tcG9uZW50KG9iaikpIHtcbiAgICByZXR1cm4gb2JqO1xuICB9XG5cbiAgaWYgKG9iaiBpbnN0YW5jZW9mIENvbG9yQ29tcG9uZW50KSB7XG4gICAgcmV0dXJuIG5ldyBWYWxpZGF0b3JFbGVtZW50V3JhcHBlcihvYmouY29sb3JQaWNrZXJFbCk7XG4gIH1cblxuICBpZiAob2JqIGluc3RhbmNlb2YgRHJvcGRvd25Db21wb25lbnQpIHtcbiAgICByZXR1cm4gbmV3IFZhbGlkYXRvckVsZW1lbnRXcmFwcGVyKG9iai5zZWxlY3RFbCk7XG4gIH1cblxuICBpZiAob2JqIGluc3RhbmNlb2YgUHJvZ3Jlc3NCYXJDb21wb25lbnQpIHtcbiAgICByZXR1cm4gbmV3IE92ZXJsYXlWYWxpZGF0b3JDb21wb25lbnQob2JqLnByb2dyZXNzQmFyKTtcbiAgfVxuXG4gIGlmIChvYmogaW5zdGFuY2VvZiBTZWFyY2hDb21wb25lbnQpIHtcbiAgICByZXR1cm4gbmV3IFZhbGlkYXRvckVsZW1lbnRXcmFwcGVyKG9iai5pbnB1dEVsKTtcbiAgfVxuXG4gIGlmIChvYmogaW5zdGFuY2VvZiBTbGlkZXJDb21wb25lbnQpIHtcbiAgICByZXR1cm4gbmV3IFZhbGlkYXRvckVsZW1lbnRXcmFwcGVyKG9iai5zbGlkZXJFbCk7XG4gIH1cblxuICBpZiAob2JqIGluc3RhbmNlb2YgVGV4dEFyZWFDb21wb25lbnQpIHtcbiAgICByZXR1cm4gbmV3IFZhbGlkYXRvckVsZW1lbnRXcmFwcGVyKG9iai5pbnB1dEVsKTtcbiAgfVxuXG4gIGlmIChvYmogaW5zdGFuY2VvZiBUZXh0Q29tcG9uZW50KSB7XG4gICAgcmV0dXJuIG5ldyBWYWxpZGF0b3JFbGVtZW50V3JhcHBlcihvYmouaW5wdXRFbCk7XG4gIH1cblxuICBpZiAob2JqIGluc3RhbmNlb2YgVG9nZ2xlQ29tcG9uZW50KSB7XG4gICAgcmV0dXJuIG5ldyBPdmVybGF5VmFsaWRhdG9yQ29tcG9uZW50KG9iai50b2dnbGVFbCk7XG4gIH1cblxuICByZXR1cm4gbnVsbDtcbn1cblxuZnVuY3Rpb24gaXNWYWxpZGF0b3JDb21wb25lbnQob2JqOiB1bmtub3duKTogb2JqIGlzIFZhbGlkYXRvckNvbXBvbmVudCB7XG4gIHJldHVybiAhIShvYmogYXMgUGFydGlhbDxWYWxpZGF0b3JDb21wb25lbnQ+KS52YWxpZGF0b3JFbDtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFNQTtBQUFBLEVBQ0U7QUFBQSxFQUNBO0FBQUEsRUFDQTtBQUFBLEVBQ0E7QUFBQSxFQUNBO0FBQUEsRUFDQTtBQUFBLEVBQ0E7QUFBQSxFQUNBO0FBQUEsT0FDSztBQUlQLFNBQVMsZ0JBQWdCO0FBWXpCLE1BQU0sMEJBQXdEO0FBQUEsRUFDNUQsSUFBVyxjQUFnQztBQUN6QyxXQUFPLEtBQUs7QUFBQSxFQUNkO0FBQUEsRUFFaUI7QUFBQSxFQUVWLFlBQVksSUFBaUI7QUFDbEMsVUFBTSxPQUFPLEdBQUcsc0JBQXNCO0FBRXRDLFNBQUssZUFBZSxTQUFTLEtBQUssU0FBUyxTQUFTO0FBQUEsTUFDbEQsS0FBSyxDQUFDLFNBQVMsYUFBYSxTQUFTLGdCQUFnQjtBQUFBLElBQ3ZELENBQUM7QUFFRCxTQUFLLGFBQWEsYUFBYTtBQUFBLE1BQzdCLFFBQVEsR0FBRyxLQUFLLE9BQU8sU0FBUyxDQUFDO0FBQUEsTUFDakMsTUFBTSxJQUFJLEtBQUssT0FBTyxPQUFPLFNBQVMsU0FBUyxDQUFDO0FBQUEsTUFDaEQsS0FBSyxJQUFJLEtBQUssTUFBTSxPQUFPLFNBQVMsU0FBUyxDQUFDO0FBQUEsTUFDOUMsT0FBTyxHQUFHLEtBQUssTUFBTSxTQUFTLENBQUM7QUFBQSxJQUNqQyxDQUFDO0FBQUEsRUFDSDtBQUNGO0FBRUEsTUFBTSx3QkFBc0Q7QUFBQSxFQUNuRCxZQUE0QixhQUErQjtBQUEvQjtBQUFBLEVBQWdDO0FBQ3JFO0FBUU8sU0FBUyxzQkFBc0IsS0FBeUM7QUFDN0UsTUFBSSxxQkFBcUIsR0FBRyxHQUFHO0FBQzdCLFdBQU87QUFBQSxFQUNUO0FBRUEsTUFBSSxlQUFlLGdCQUFnQjtBQUNqQyxXQUFPLElBQUksd0JBQXdCLElBQUksYUFBYTtBQUFBLEVBQ3REO0FBRUEsTUFBSSxlQUFlLG1CQUFtQjtBQUNwQyxXQUFPLElBQUksd0JBQXdCLElBQUksUUFBUTtBQUFBLEVBQ2pEO0FBRUEsTUFBSSxlQUFlLHNCQUFzQjtBQUN2QyxXQUFPLElBQUksMEJBQTBCLElBQUksV0FBVztBQUFBLEVBQ3REO0FBRUEsTUFBSSxlQUFlLGlCQUFpQjtBQUNsQyxXQUFPLElBQUksd0JBQXdCLElBQUksT0FBTztBQUFBLEVBQ2hEO0FBRUEsTUFBSSxlQUFlLGlCQUFpQjtBQUNsQyxXQUFPLElBQUksd0JBQXdCLElBQUksUUFBUTtBQUFBLEVBQ2pEO0FBRUEsTUFBSSxlQUFlLG1CQUFtQjtBQUNwQyxXQUFPLElBQUksd0JBQXdCLElBQUksT0FBTztBQUFBLEVBQ2hEO0FBRUEsTUFBSSxlQUFlLGVBQWU7QUFDaEMsV0FBTyxJQUFJLHdCQUF3QixJQUFJLE9BQU87QUFBQSxFQUNoRDtBQUVBLE1BQUksZUFBZSxpQkFBaUI7QUFDbEMsV0FBTyxJQUFJLDBCQUEwQixJQUFJLFFBQVE7QUFBQSxFQUNuRDtBQUVBLFNBQU87QUFDVDtBQUVBLFNBQVMscUJBQXFCLEtBQXlDO0FBQ3JFLFNBQU8sQ0FBQyxDQUFFLElBQW9DO0FBQ2hEOyIsCiAgIm5hbWVzIjogW10KfQo=
142
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/Components/SettingComponents/ValidatorComponent.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains a component that has a validator element.\n */\n\nimport {\n  ColorComponent,\n  DropdownComponent,\n  ProgressBarComponent,\n  SearchComponent,\n  SliderComponent,\n  TextAreaComponent,\n  TextComponent,\n  ToggleComponent\n} from 'obsidian';\n\nimport type { ValidatorElement } from '../../../HTMLElement.ts';\n\nimport { CssClass } from '../../../CssClass.ts';\nimport {\n  onAncestorScrollOrResize,\n  toPx\n} from '../../../HTMLElement.ts';\n\n/**\n * A component that has a validator element.\n */\nexport interface ValidatorComponent {\n  /**\n   * The validator element of the component.\n   */\n  readonly validatorEl: ValidatorElement;\n}\n\nclass OverlayValidatorComponent implements ValidatorComponent {\n  public get validatorEl(): ValidatorElement {\n    return this._validatorEl;\n  }\n\n  private readonly _validatorEl: ValidatorElement;\n\n  public constructor(private readonly el: HTMLElement) {\n    if (!el.parentElement) {\n      throw new Error('Element must be attached to the DOM');\n    }\n\n    this._validatorEl = el.parentElement.createEl('input', {\n      attr: {\n        tabindex: -1\n      },\n      cls: [CssClass.LibraryName, CssClass.OverlayValidator]\n    });\n\n    this._validatorEl.addEventListener('focus', () => {\n      this.el.focus();\n    });\n\n    this._validatorEl.isActiveElement = this.isElementOrDescendantActive.bind(this);\n\n    let tabIndexEl = this.el.querySelector<HTMLElement>('[tabindex]');\n    if (!tabIndexEl) {\n      if (this.el.getAttr('tabindex') === null) {\n        this.el.tabIndex = -1;\n      }\n      tabIndexEl = this.el;\n    }\n\n    this.el.addEventListener('focusin', () => {\n      this.forceBlurValidatorEl();\n    });\n    this.el.addEventListener('click', () => {\n      tabIndexEl.focus();\n    });\n    this.el.addEventListener('focusout', () => {\n      setTimeout(() => {\n        if (this.isElementOrDescendantActive()) {\n          return;\n        }\n\n        this.forceBlurValidatorEl();\n      }, 0);\n    });\n\n    const unregisterScrollOrResizeHandlers = onAncestorScrollOrResize(this.el, this.updatePosition.bind(this));\n\n    const observer = new MutationObserver((mutations) => {\n      for (const mutation of mutations) {\n        for (const removedNode of Array.from(mutation.removedNodes)) {\n          if (removedNode === this._validatorEl) {\n            unregisterScrollOrResizeHandlers();\n            observer.disconnect();\n            return;\n          }\n        }\n      }\n    });\n\n    observer.observe(document.body, {\n      childList: true,\n      subtree: true\n    });\n  }\n\n  private forceBlurValidatorEl(): void {\n    this._validatorEl.dispatchEvent(new Event('blur'));\n  }\n\n  private isElementOrDescendantActive(): boolean {\n    return this.el.contains(document.activeElement);\n  }\n\n  private updatePosition(): void {\n    if (!this.el.offsetParent) {\n      return;\n    }\n    const rect = this.el.getBoundingClientRect();\n    const parentRect = this.el.offsetParent.getBoundingClientRect();\n\n    this._validatorEl.setCssStyles({\n      height: toPx(rect.height),\n      left: toPx(rect.left - parentRect.left + this.el.offsetParent.scrollLeft),\n      top: toPx(rect.top - parentRect.top + this.el.offsetParent.scrollTop),\n      width: toPx(rect.width)\n    });\n  }\n}\n\nclass ValidatorElementWrapper implements ValidatorComponent {\n  public constructor(public readonly validatorEl: ValidatorElement) {}\n}\n\n/**\n * Gets a validator component related to the given object.\n *\n * @param obj - Any object.\n * @returns The related validator component or `null` if no related validator component is found.\n */\nexport function getValidatorComponent(obj: unknown): null | ValidatorComponent {\n  if (isValidatorComponent(obj)) {\n    return obj;\n  }\n\n  if (obj instanceof ColorComponent) {\n    return new ValidatorElementWrapper(obj.colorPickerEl);\n  }\n\n  if (obj instanceof DropdownComponent) {\n    return new ValidatorElementWrapper(obj.selectEl);\n  }\n\n  if (obj instanceof ProgressBarComponent) {\n    return new OverlayValidatorComponent(obj.progressBar);\n  }\n\n  if (obj instanceof SearchComponent) {\n    return new ValidatorElementWrapper(obj.inputEl);\n  }\n\n  if (obj instanceof SliderComponent) {\n    return new ValidatorElementWrapper(obj.sliderEl);\n  }\n\n  if (obj instanceof TextAreaComponent) {\n    return new ValidatorElementWrapper(obj.inputEl);\n  }\n\n  if (obj instanceof TextComponent) {\n    return new ValidatorElementWrapper(obj.inputEl);\n  }\n\n  if (obj instanceof ToggleComponent) {\n    return new OverlayValidatorComponent(obj.toggleEl);\n  }\n\n  return null;\n}\n\nfunction isValidatorComponent(obj: unknown): obj is ValidatorComponent {\n  return !!(obj as Partial<ValidatorComponent>).validatorEl;\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -5,6 +5,11 @@ if you want to view the source, please visit the github repository of this plugi
5
5
 
6
6
  (function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
7
7
 
8
+ import {
9
+ autoUpdate,
10
+ computePosition,
11
+ offset
12
+ } from "@floating-ui/dom";
8
13
  import {
9
14
  debounce,
10
15
  PluginSettingTab,
@@ -20,6 +25,7 @@ import {
20
25
  noop,
21
26
  noopAsync
22
27
  } from "../../Function.mjs";
28
+ import { toPx } from "../../HTMLElement.mjs";
23
29
  import { deepEqual } from "../../Object.mjs";
24
30
  import { AsyncEventsComponent } from "../Components/AsyncEventsComponent.mjs";
25
31
  import { getTextBasedComponentValue } from "../Components/SettingComponents/TextBasedComponent.mjs";
@@ -89,16 +95,46 @@ class PluginSettingsTabBase extends PluginSettingTab {
89
95
  shouldShowValidationMessage: true
90
96
  };
91
97
  const optionsExt = { ...DEFAULT_OPTIONS, ...options };
92
- const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;
98
+ const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;
93
99
  const textBasedComponent = getTextBasedComponentValue(valueComponent);
94
100
  const readonlyValue = this.pluginSettings[propertyName];
95
101
  const defaultValue = this.plugin.settingsManager.defaultSettings[propertyName];
96
102
  textBasedComponent?.setPlaceholderValue(optionsExt.pluginSettingsToComponentValueConverter(defaultValue));
97
- this.asyncEventsComponent.registerAsyncEvent(this.on("validationMessageChanged", (anotherPropertyName, validationMessage) => {
103
+ let validationMessage;
104
+ let tooltipEl = null;
105
+ let tooltipContentEl = null;
106
+ if (validatorEl?.parentElement) {
107
+ tooltipEl = validatorEl.parentElement.createDiv({ cls: [CssClass.LibraryName, CssClass.Tooltip, CssClass.TooltipValidator] });
108
+ tooltipContentEl = tooltipEl.createSpan();
109
+ tooltipEl.createDiv({ cls: [CssClass.LibraryName, CssClass.TooltipArrow] });
110
+ tooltipEl.hide();
111
+ autoUpdate(
112
+ validatorEl,
113
+ tooltipEl,
114
+ convertAsyncToSync(async () => {
115
+ if (!tooltipEl) {
116
+ return;
117
+ }
118
+ const OFFSET = 8;
119
+ const { x, y } = await computePosition(validatorEl, tooltipEl, {
120
+ middleware: [
121
+ offset(OFFSET)
122
+ ],
123
+ placement: "bottom"
124
+ });
125
+ tooltipEl.setCssProps({
126
+ left: toPx(x),
127
+ top: toPx(y)
128
+ });
129
+ })
130
+ );
131
+ }
132
+ this.asyncEventsComponent.registerAsyncEvent(this.on("validationMessageChanged", (anotherPropertyName, anotherValidationMessage) => {
98
133
  if (propertyName !== anotherPropertyName) {
99
134
  return;
100
135
  }
101
- updateValidatorElement(validationMessage);
136
+ validationMessage = anotherValidationMessage;
137
+ updateValidatorEl();
102
138
  }));
103
139
  let shouldEmptyOnBlur = false;
104
140
  if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {
@@ -107,6 +143,12 @@ class PluginSettingsTabBase extends PluginSettingTab {
107
143
  valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));
108
144
  }
109
145
  let shouldSkipOnChange = false;
146
+ const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;
147
+ const updateValidatorElDebounced = debounce(() => {
148
+ requestAnimationFrame(() => {
149
+ updateValidatorEl();
150
+ });
151
+ }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);
110
152
  valueComponent.onChange(async (uiValue) => {
111
153
  if (shouldSkipOnChange) {
112
154
  shouldSkipOnChange = false;
@@ -115,7 +157,6 @@ class PluginSettingsTabBase extends PluginSettingTab {
115
157
  shouldEmptyOnBlur = false;
116
158
  const oldValue = this.pluginSettings[propertyName];
117
159
  let newValue = void 0;
118
- let validationMessage;
119
160
  if (textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty) {
120
161
  newValue = defaultValue;
121
162
  validationMessage = "";
@@ -131,42 +172,50 @@ class PluginSettingsTabBase extends PluginSettingTab {
131
172
  }
132
173
  }
133
174
  }
134
- updateValidatorElement(validationMessage);
175
+ updateValidatorElDebounced();
135
176
  if (newValue !== void 0) {
136
177
  await optionsExt.onChanged(newValue, oldValue);
137
178
  }
138
179
  this.saveSettingsDebounced();
139
180
  });
140
- validatorElement?.addEventListener("focus", () => {
141
- updateValidatorElement();
181
+ validatorEl?.addEventListener("focus", () => {
182
+ updateValidatorElDebounced();
142
183
  });
143
- validatorElement?.addEventListener("blur", () => {
144
- updateValidatorElement();
184
+ validatorEl?.addEventListener("blur", () => {
185
+ updateValidatorElDebounced();
145
186
  });
146
- updateValidatorElement(this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName]);
187
+ validatorEl?.addEventListener("click", () => {
188
+ requestAnimationFrame(() => {
189
+ updateValidatorElDebounced();
190
+ });
191
+ });
192
+ validationMessage = this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName] ?? "";
193
+ updateValidatorEl();
147
194
  return valueComponent;
148
- function updateValidatorElement(validationMessage) {
149
- if (shouldEmptyOnBlur && !validatorElement?.isActiveElement()) {
195
+ function updateValidatorEl() {
196
+ if (shouldEmptyOnBlur && !validatorEl?.isActiveElement()) {
150
197
  shouldEmptyOnBlur = false;
151
198
  if (!textBasedComponent?.isEmpty()) {
152
199
  shouldSkipOnChange = true;
153
200
  textBasedComponent?.empty();
154
201
  }
155
202
  }
156
- if (!validatorElement) {
203
+ if (!validatorEl) {
157
204
  return;
158
205
  }
159
206
  if (validationMessage === "") {
160
- validatorElement.setCustomValidity("");
161
- validatorElement.checkValidity();
162
- validationMessage = validatorElement.validationMessage;
163
- }
164
- if (validationMessage !== void 0) {
165
- validatorElement.setCustomValidity(validationMessage);
166
- setTooltip(validatorElement, validationMessage);
207
+ validatorEl.setCustomValidity("");
208
+ validatorEl.checkValidity();
209
+ validationMessage = validatorEl.validationMessage;
167
210
  }
168
- if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {
169
- validatorElement.reportValidity();
211
+ validatorEl.setCustomValidity(validationMessage);
212
+ if (optionsExt.shouldShowValidationMessage) {
213
+ tooltipEl?.toggle(!!validationMessage);
214
+ if (tooltipContentEl) {
215
+ tooltipContentEl.textContent = validationMessage;
216
+ }
217
+ } else if (validationMessage) {
218
+ setTooltip(validatorEl, validationMessage);
170
219
  }
171
220
  }
172
221
  }
@@ -229,4 +278,4 @@ export {
229
278
  PluginSettingsTabBase,
230
279
  SAVE_TO_FILE_CONTEXT
231
280
  };
232
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type { Debouncer } from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../AsyncEvents.ts';\nimport type { StringKeys } from '../../Type.ts';\nimport type { ValueComponentWithChangeTracking } from '../Components/SettingComponents/ValueComponentWithChangeTracking.ts';\nimport type { ValidationMessageHolder } from '../ValidationMessage.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nimport type { PluginSettingsManagerBase } from './PluginSettingsManagerBase.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../Function.ts';\nimport { deepEqual } from '../../Object.ts';\nimport { AsyncEventsComponent } from '../Components/AsyncEventsComponent.ts';\nimport { getTextBasedComponentValue } from '../Components/SettingComponents/TextBasedComponent.ts';\nimport { getValidatorComponent } from '../Components/SettingComponents/ValidatorComponent.ts';\nimport { isValidationMessageHolder } from '../ValidationMessage.ts';\nimport { getPluginId } from './PluginId.ts';\n\n/**\n * The context passed to the {@link PluginSettingsManagerBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for binding a value component to a plugin setting.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  onChanged?(newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>): Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for binding a value component to a plugin setting.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsTabBase<PluginTypes extends PluginTypesBase> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * The debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEvents: AsyncEvents;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ExtractPluginSettings<PluginTypes> {\n    return this.plugin.settingsManager.settingsWrapper.settings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public override plugin: ExtractPlugin<PluginTypes>) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n    this.asyncEvents = new AsyncEvents();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<ExtractPluginSettings<PluginTypes>, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<PluginTypes>, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<PluginTypes>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.pluginSettings[propertyName] as ReadonlyDeep<PropertyType>;\n    const defaultValue = (this.plugin.settingsManager.defaultSettings as PluginSettings)[propertyName] as PropertyType;\n    textBasedComponent?.setPlaceholderValue(optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>));\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, validationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      updateValidatorElement(validationMessage);\n    }));\n\n    let shouldEmptyOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.pluginSettings[propertyName];\n      let newValue: PropertyType | undefined = undefined;\n      let validationMessage: string;\n      if (textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty) {\n        newValue = defaultValue;\n        validationMessage = '';\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n        } else {\n          newValue = convertedValue;\n          validationMessage = await this.plugin.settingsManager.setProperty(propertyName, newValue);\n          if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n            shouldEmptyOnBlur = true;\n          }\n        }\n      }\n      updateValidatorElement(validationMessage);\n      if (newValue !== undefined) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue as ReadonlyDeep<PropertyType>);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorElement?.addEventListener('focus', () => {\n      updateValidatorElement();\n    });\n    validatorElement?.addEventListener('blur', () => {\n      updateValidatorElement();\n    });\n\n    updateValidatorElement(this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName]);\n    return valueComponent;\n\n    function updateValidatorElement(validationMessage?: string): void {\n      if (shouldEmptyOnBlur && !validatorElement?.isActiveElement()) {\n        shouldEmptyOnBlur = false;\n        if (!textBasedComponent?.isEmpty()) {\n          shouldSkipOnChange = true;\n          textBasedComponent?.empty();\n        }\n      }\n\n      if (!validatorElement) {\n        return;\n      }\n\n      if (validationMessage === '') {\n        validatorElement.setCustomValidity('');\n        validatorElement.checkValidity();\n        validationMessage = validatorElement.validationMessage;\n      }\n\n      if (validationMessage !== undefined) {\n        validatorElement.setCustomValidity(validationMessage);\n        setTooltip(validatorElement, validationMessage);\n      }\n      if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {\n        validatorElement.reportValidity();\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT));\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedSettings - The loaded settings.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    this.refresh();\n    await noopAsync();\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      for (const [propertyName, validationMessage] of Object.entries(newSettings.validationMessages as Record<string, string>)) {\n        await this.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n      }\n      return;\n    }\n\n    this.refresh();\n  }\n\n  private refresh(): void {\n    this.containerEl.empty();\n    this.display();\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

281
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type { Debouncer } from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  autoUpdate,\n  computePosition,\n  offset\n} from '@floating-ui/dom';\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../AsyncEvents.ts';\nimport type { StringKeys } from '../../Type.ts';\nimport type { ValueComponentWithChangeTracking } from '../Components/SettingComponents/ValueComponentWithChangeTracking.ts';\nimport type { ValidationMessageHolder } from '../ValidationMessage.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nimport type { PluginSettingsManagerBase } from './PluginSettingsManagerBase.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../Function.ts';\nimport { toPx } from '../../HTMLElement.ts';\nimport { deepEqual } from '../../Object.ts';\nimport { AsyncEventsComponent } from '../Components/AsyncEventsComponent.ts';\nimport { getTextBasedComponentValue } from '../Components/SettingComponents/TextBasedComponent.ts';\nimport { getValidatorComponent } from '../Components/SettingComponents/ValidatorComponent.ts';\nimport { isValidationMessageHolder } from '../ValidationMessage.ts';\nimport { getPluginId } from './PluginId.ts';\n\n/**\n * The context passed to the {@link PluginSettingsManagerBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for binding a value component to a plugin setting.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  onChanged?(newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>): Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for binding a value component to a plugin setting.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsTabBase<PluginTypes extends PluginTypesBase> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * The debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEvents: AsyncEvents;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ExtractPluginSettings<PluginTypes> {\n    return this.plugin.settingsManager.settingsWrapper.settings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public override plugin: ExtractPlugin<PluginTypes>) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n    this.asyncEvents = new AsyncEvents();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<ExtractPluginSettings<PluginTypes>, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<PluginTypes>, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<PluginTypes>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.pluginSettings[propertyName] as ReadonlyDeep<PropertyType>;\n    const defaultValue = (this.plugin.settingsManager.defaultSettings as PluginSettings)[propertyName] as PropertyType;\n    textBasedComponent?.setPlaceholderValue(optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>));\n\n    let validationMessage: string;\n    let tooltipEl: HTMLElement | null = null;\n    let tooltipContentEl: HTMLElement | null = null;\n    if (validatorEl?.parentElement) {\n      tooltipEl = validatorEl.parentElement.createDiv({ cls: [CssClass.LibraryName, CssClass.Tooltip, CssClass.TooltipValidator] });\n      tooltipContentEl = tooltipEl.createSpan();\n      tooltipEl.createDiv({ cls: [CssClass.LibraryName, CssClass.TooltipArrow] });\n      tooltipEl.hide();\n\n      autoUpdate(\n        validatorEl,\n        tooltipEl,\n        convertAsyncToSync(async () => {\n          if (!tooltipEl) {\n            return;\n          }\n\n          const OFFSET = 8;\n          const { x, y } = await computePosition(validatorEl, tooltipEl, {\n            middleware: [\n              offset(OFFSET)\n            ],\n            placement: 'bottom'\n          });\n          tooltipEl.setCssProps({\n            left: toPx(x),\n            top: toPx(y)\n          });\n        })\n      );\n    }\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, anotherValidationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      validationMessage = anotherValidationMessage;\n      updateValidatorEl();\n    }));\n\n    let shouldEmptyOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n    const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;\n    const updateValidatorElDebounced = debounce(() => {\n      requestAnimationFrame(() => {\n        updateValidatorEl();\n      });\n    }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.pluginSettings[propertyName];\n      let newValue: PropertyType | undefined = undefined;\n      if (textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty) {\n        newValue = defaultValue;\n        validationMessage = '';\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n        } else {\n          newValue = convertedValue;\n          validationMessage = await this.plugin.settingsManager.setProperty(propertyName, newValue);\n          if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n            shouldEmptyOnBlur = true;\n          }\n        }\n      }\n\n      updateValidatorElDebounced();\n      if (newValue !== undefined) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue as ReadonlyDeep<PropertyType>);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorEl?.addEventListener('focus', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('blur', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('click', () => {\n      requestAnimationFrame(() => {\n        updateValidatorElDebounced();\n      });\n    });\n\n    validationMessage = this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName] ?? '';\n    updateValidatorEl();\n\n    return valueComponent;\n\n    function updateValidatorEl(): void {\n      if (shouldEmptyOnBlur && !validatorEl?.isActiveElement()) {\n        shouldEmptyOnBlur = false;\n        if (!textBasedComponent?.isEmpty()) {\n          shouldSkipOnChange = true;\n          textBasedComponent?.empty();\n        }\n      }\n\n      if (!validatorEl) {\n        return;\n      }\n\n      if (validationMessage === '') {\n        validatorEl.setCustomValidity('');\n        validatorEl.checkValidity();\n        validationMessage = validatorEl.validationMessage;\n      }\n\n      validatorEl.setCustomValidity(validationMessage);\n      if (optionsExt.shouldShowValidationMessage) {\n        tooltipEl?.toggle(!!validationMessage);\n        if (tooltipContentEl) {\n          tooltipContentEl.textContent = validationMessage;\n        }\n      } else if (validationMessage) {\n        setTooltip(validatorEl, validationMessage);\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT));\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedSettings - The loaded settings.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    this.refresh();\n    await noopAsync();\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      for (const [propertyName, validationMessage] of Object.entries(newSettings.validationMessages as Record<string, string>)) {\n        await this.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n      }\n      return;\n    }\n\n    this.refresh();\n  }\n\n  private refresh(): void {\n    this.containerEl.empty();\n    this.display();\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

package/dist/styles.css CHANGED
@@ -172,7 +172,36 @@ body:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {
172
172
  .obsidian-dev-utils :invalid {
173
173
  box-shadow: 0 0 0 2px var(--text-error);
174
174
  }
175
+ .obsidian-dev-utils input.metadata-input-text:active:invalid, .obsidian-dev-utils input.metadata-input-text:focus-visible:invalid, .obsidian-dev-utils input.metadata-input-text:focus:invalid,
176
+ .obsidian-dev-utils input[type=date]:active:invalid,
177
+ .obsidian-dev-utils input[type=date]:focus-visible:invalid,
178
+ .obsidian-dev-utils input[type=date]:focus:invalid,
179
+ .obsidian-dev-utils input[type=datetime-local]:active:invalid,
180
+ .obsidian-dev-utils input[type=datetime-local]:focus-visible:invalid,
181
+ .obsidian-dev-utils input[type=datetime-local]:focus:invalid,
182
+ .obsidian-dev-utils input[type=email]:active:invalid,
183
+ .obsidian-dev-utils input[type=email]:focus-visible:invalid,
184
+ .obsidian-dev-utils input[type=email]:focus:invalid,
185
+ .obsidian-dev-utils input[type=number]:active:invalid,
186
+ .obsidian-dev-utils input[type=number]:focus-visible:invalid,
187
+ .obsidian-dev-utils input[type=number]:focus:invalid,
188
+ .obsidian-dev-utils input[type=password]:active:invalid,
189
+ .obsidian-dev-utils input[type=password]:focus-visible:invalid,
190
+ .obsidian-dev-utils input[type=password]:focus:invalid,
191
+ .obsidian-dev-utils input[type=search]:active:invalid,
192
+ .obsidian-dev-utils input[type=search]:focus-visible:invalid,
193
+ .obsidian-dev-utils input[type=search]:focus:invalid,
194
+ .obsidian-dev-utils input[type=text]:active:invalid,
195
+ .obsidian-dev-utils input[type=text]:focus-visible:invalid,
196
+ .obsidian-dev-utils input[type=text]:focus:invalid,
197
+ .obsidian-dev-utils textarea:active:invalid,
198
+ .obsidian-dev-utils textarea:focus-visible:invalid,
199
+ .obsidian-dev-utils textarea:focus:invalid {
200
+ box-shadow: 0 0 0 2px var(--text-error);
201
+ }
175
202
  .obsidian-dev-utils.overlay-validator {
203
+ caret-color: transparent;
204
+ cursor: default;
176
205
  position: absolute;
177
206
  background-color: transparent;
178
207
  border: none;
@@ -180,5 +209,10 @@ body:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {
180
209
  pointer-events: none;
181
210
  z-index: 9999;
182
211
  }
212
+ .obsidian-dev-utils.tooltip.tooltip-validator {
213
+ position: absolute;
214
+ animation: none;
215
+ transform: none;
216
+ }
183
217
 
184
- /*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/tri-state-checkbox-component.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AACE;EACE;;AAGF;AAAA;AAAA;AAAA;EAIE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC5DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;EACE;;;ACDF;EACE;;;ACFJ;EACE;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%20%20%20%20%20%20padding-top:%203px;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cr%5Cn%20%20&.tri-state-checkbox-component%20%7B%5Cr%5Cn%20%20%20%20input%5Btype='checkbox'%5D:indeterminate%20%7B%5Cr%5Cn%20%20%20%20%20%20appearance:%20checkbox;%5Cr%5Cn%20%20%20%20%7D%5Cr%5Cn%20%20%7D%5Cr%5Cn%7D%5Cr%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%20%20%7D%5Cn%5Cn%20%20&.overlay-validator%20%7B%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20background-color:%20transparent;%5Cn%20%20%20%20border:%20none;%5Cn%20%20%20%20outline:%20none;%5Cn%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20z-index:%209999;%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */
218
+ /*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/tri-state-checkbox-component.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AACE;EACE;;AAGF;AAAA;AAAA;AAAA;EAIE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC5DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;EACE;;;ACDF;EACE;;;ACEJ;EAJA;;AAoBI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EApBJ;;AA0BA;EACE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%20%20%20%20%20%20padding-top:%203px;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cr%5Cn%20%20&.tri-state-checkbox-component%20%7B%5Cr%5Cn%20%20%20%20input%5Btype='checkbox'%5D:indeterminate%20%7B%5Cr%5Cn%20%20%20%20%20%20appearance:%20checkbox;%5Cr%5Cn%20%20%20%20%7D%5Cr%5Cn%20%20%7D%5Cr%5Cn%7D%5Cr%5Cn%22,%22@mixin%20invalid%20%7B%5Cn%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%7D%5Cn%5Cn.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20@include%20invalid;%5Cn%20%20%7D%5Cn%5Cn%20%20input.metadata-input-text,%5Cn%20%20input%5Btype='date'%5D,%5Cn%20%20input%5Btype='datetime-local'%5D,%5Cn%20%20input%5Btype='email'%5D,%5Cn%20%20input%5Btype='number'%5D,%5Cn%20%20input%5Btype='password'%5D,%5Cn%20%20input%5Btype='search'%5D,%5Cn%20%20input%5Btype='text'%5D,%5Cn%20%20textarea%20%7B%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus-visible,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20&:invalid%20%7B%5Cn%20%20%20%20%20%20%20%20@include%20invalid;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20&.overlay-validator%20%7B%5Cn%20%20%20%20caret-color:%20transparent;%5Cn%20%20%20%20cursor:%20default;%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20background-color:%20transparent;%5Cn%20%20%20%20border:%20none;%5Cn%20%20%20%20outline:%20none;%5Cn%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20z-index:%209999;%5Cn%20%20%7D%5Cn%5Cn%20%20&.tooltip.tooltip-validator%20%7B%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20animation:%20none;%5Cn%20%20%20%20transform:%20none;%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "obsidian-dev-utils",
3
- "version": "26.1.3-beta.9",
3
+ "version": "26.2.0",
4
4
  "description": "This is the collection of useful functions that you can use for your Obsidian plugin development",
5
5
  "keywords": [
6
6
  "obsidian"
@@ -443,6 +443,7 @@
443
443
  },
444
444
  "dependencies": {
445
445
  "@eslint/js": "^9.24.0",
446
+ "@floating-ui/dom": "^1.6.13",
446
447
  "@guardian/eslint-plugin-tsdoc-required": "^0.1.3",
447
448
  "@lezer/common": "^1.2.3",
448
449
  "@stylistic/eslint-plugin": "^4.2.0",
@@ -457,10 +458,10 @@
457
458
  "@types/path-browserify": "^1.0.3",
458
459
  "@types/picomatch": "^4.0.0",
459
460
  "@types/pug": "^2.0.10",
460
- "@types/react": "^19.1.1",
461
+ "@types/react": "^19.1.2",
461
462
  "@types/shell-quote": "^1.7.5",
462
- "@typescript-eslint/eslint-plugin": "^8.29.1",
463
- "@typescript-eslint/parser": "^8.29.1",
463
+ "@typescript-eslint/eslint-plugin": "^8.30.1",
464
+ "@typescript-eslint/parser": "^8.30.1",
464
465
  "better-typescript-lib": "^2.11.0",
465
466
  "commander": "^13.1.0",
466
467
  "compare-versions": "^6.1.1",
@@ -503,7 +504,7 @@
503
504
  "tsx": "^4.19.3",
504
505
  "type-fest": "^4.39.1",
505
506
  "typescript": "^5.8.3",
506
- "typescript-eslint": "^8.29.1"
507
+ "typescript-eslint": "^8.30.1"
507
508
  },
508
509
  "overrides": {
509
510
  "esbuild": "$esbuild"