obsidian-dev-utils 22.1.1-beta.4 → 22.1.1-beta.6

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.
Files changed (73) hide show
  1. package/CHANGELOG.md +21 -0
  2. package/dist/dprint.json +1 -0
  3. package/dist/lib/cjs/Library.cjs +1 -1
  4. package/dist/lib/cjs/Object.cjs +1 -1
  5. package/dist/lib/cjs/Object.d.cts +6 -0
  6. package/dist/lib/cjs/ScriptUtils/CliUtils.cjs +1 -2
  7. package/dist/lib/cjs/ScriptUtils/CliUtils.d.cts +3 -2
  8. package/dist/lib/cjs/ScriptUtils/cli.cjs +1 -1
  9. package/dist/lib/cjs/String.cjs +1 -1
  10. package/dist/lib/cjs/String.d.cts +2 -1
  11. package/dist/lib/cjs/obsidian/Callout.cjs +1 -1
  12. package/dist/lib/cjs/obsidian/Callout.d.cts +2 -1
  13. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.cjs +1 -1
  14. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.d.cts +2 -1
  15. package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.cjs +38 -0
  16. package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.d.cts +19 -0
  17. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.cjs +1 -1
  18. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.d.cts +2 -1
  19. package/dist/lib/cjs/obsidian/Components/index.cjs +4 -1
  20. package/dist/lib/cjs/obsidian/Components/index.d.cts +1 -0
  21. package/dist/lib/cjs/obsidian/FileManager.cjs +1 -1
  22. package/dist/lib/cjs/obsidian/FileManager.d.cts +2 -1
  23. package/dist/lib/cjs/obsidian/Link.cjs +1 -1
  24. package/dist/lib/cjs/obsidian/Link.d.cts +4 -3
  25. package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
  26. package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +2 -1
  27. package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +11 -1
  28. package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +8 -0
  29. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +148 -52
  30. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +33 -18
  31. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +19 -20
  32. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +13 -15
  33. package/dist/lib/cjs/obsidian/ValidationMessage.cjs +38 -0
  34. package/dist/lib/cjs/obsidian/ValidationMessage.d.cts +16 -0
  35. package/dist/lib/cjs/obsidian/index.cjs +4 -1
  36. package/dist/lib/cjs/obsidian/index.d.cts +1 -0
  37. package/dist/lib/esm/Library.mjs +1 -1
  38. package/dist/lib/esm/Object.d.mts +6 -0
  39. package/dist/lib/esm/Object.mjs +1 -1
  40. package/dist/lib/esm/ScriptUtils/CliUtils.d.mts +3 -2
  41. package/dist/lib/esm/ScriptUtils/CliUtils.mjs +1 -2
  42. package/dist/lib/esm/ScriptUtils/cli.mjs +1 -1
  43. package/dist/lib/esm/String.d.mts +2 -1
  44. package/dist/lib/esm/String.mjs +1 -1
  45. package/dist/lib/esm/obsidian/Callout.d.mts +2 -1
  46. package/dist/lib/esm/obsidian/Callout.mjs +1 -1
  47. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.d.mts +2 -1
  48. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.mjs +1 -1
  49. package/dist/lib/esm/obsidian/Components/PlaceholderComponent.d.mts +19 -0
  50. package/dist/lib/esm/obsidian/Components/PlaceholderComponent.mjs +14 -0
  51. package/dist/lib/esm/obsidian/Components/TypedTextComponent.d.mts +2 -1
  52. package/dist/lib/esm/obsidian/Components/TypedTextComponent.mjs +1 -1
  53. package/dist/lib/esm/obsidian/Components/index.d.mts +1 -0
  54. package/dist/lib/esm/obsidian/Components/index.mjs +3 -1
  55. package/dist/lib/esm/obsidian/FileManager.d.mts +2 -1
  56. package/dist/lib/esm/obsidian/FileManager.mjs +1 -1
  57. package/dist/lib/esm/obsidian/Link.d.mts +4 -3
  58. package/dist/lib/esm/obsidian/Link.mjs +1 -1
  59. package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +2 -1
  60. package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
  61. package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +8 -0
  62. package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +15 -2
  63. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +33 -18
  64. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +148 -53
  65. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +13 -15
  66. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +19 -20
  67. package/dist/lib/esm/obsidian/ValidationMessage.d.mts +16 -0
  68. package/dist/lib/esm/obsidian/ValidationMessage.mjs +14 -0
  69. package/dist/lib/esm/obsidian/index.d.mts +1 -0
  70. package/dist/lib/esm/obsidian/index.mjs +3 -1
  71. package/obsidian/Components/PlaceholderComponent/package.json +6 -0
  72. package/obsidian/ValidationMessage/package.json +6 -0
  73. package/package.json +1 -1
@@ -5,79 +5,100 @@ 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 { Plugin } from "obsidian";
8
+ import { Notice } from "obsidian";
9
9
  import { noop } from "../../Function.mjs";
10
10
  import { DateTransformer } from "../../Transformers/DateTransformer.mjs";
11
11
  import { DurationTransformer } from "../../Transformers/DurationTransformer.mjs";
12
12
  import { GroupTransformer } from "../../Transformers/GroupTransformer.mjs";
13
13
  import { SkipPrivatePropertyTransformer } from "../../Transformers/SkipPrivatePropertyTransformer.mjs";
14
+ import { isValidationMessageHolder } from "../ValidationMessage.mjs";
14
15
  const defaultTransformer = new GroupTransformer([
15
16
  new SkipPrivatePropertyTransformer(),
16
17
  new DateTransformer(),
17
18
  new DurationTransformer()
18
19
  ]);
19
- class PluginSettingsProperty {
20
- constructor(defaultValue, validator) {
21
- this.defaultValue = defaultValue;
22
- this.validator = validator;
23
- }
24
- validationMessage = "";
25
- value;
26
- clear() {
27
- this.value = void 0;
28
- this.validationMessage = "";
29
- }
30
- get() {
31
- return this.value ?? this.defaultValue;
32
- }
33
- getSafe() {
34
- return this.validationMessage ? this.defaultValue : this.get();
20
+ class ProxyHandlerBase {
21
+ constructor(properties) {
22
+ this.properties = properties;
23
+ }
24
+ get(target, prop) {
25
+ const record = target;
26
+ if (typeof prop !== "string") {
27
+ return record[prop];
28
+ }
29
+ const property = this.properties.get(prop);
30
+ if (!property) {
31
+ return record[prop];
32
+ }
33
+ return this.getPropertyValue(property);
35
34
  }
36
- async set(value) {
37
- this.value = value;
38
- if (this.value !== void 0) {
39
- this.validationMessage = await this.validator(this.value) ?? "";
35
+ }
36
+ class EditableSettingsProxyHandler extends ProxyHandlerBase {
37
+ validationPromise = Promise.resolve();
38
+ set(target, prop, value) {
39
+ const record = target;
40
+ if (typeof prop !== "string") {
41
+ record[prop] = value;
42
+ return true;
43
+ }
44
+ const property = this.properties.get(prop);
45
+ if (!property) {
46
+ record[prop] = value;
47
+ return true;
40
48
  }
49
+ property.set(value);
50
+ this.validationPromise = this.validationPromise.then(() => property.setAndValidate(value));
51
+ return true;
52
+ }
53
+ getPropertyValue(property) {
54
+ return property.get();
41
55
  }
42
56
  }
43
57
  class PropertiesMap extends Map {
44
- getTyped(key) {
45
- const property = super.get(key);
58
+ getTyped(propertyName) {
59
+ const property = super.get(propertyName);
46
60
  if (!property) {
47
- throw new Error(`Property ${String(key)} not found`);
61
+ throw new Error(`Property ${String(propertyName)} not found`);
48
62
  }
49
63
  return property;
50
64
  }
51
- setTyped(key, value) {
52
- return super.set(key, value);
65
+ setTyped(propertyName, value) {
66
+ return super.set(propertyName, value);
67
+ }
68
+ }
69
+ class SafeSettingsProxyHandler extends ProxyHandlerBase {
70
+ getPropertyValue(property) {
71
+ return property.getSafe();
53
72
  }
54
73
  }
55
74
  class PluginSettingsManagerBase {
56
75
  constructor(plugin) {
57
76
  this.plugin = plugin;
58
- const defaultSettings = this.createDefaultSettings();
77
+ this.defaultSettings = this.createDefaultSettings();
59
78
  this.addValidators();
60
79
  this.properties = new PropertiesMap();
61
- for (const key of Object.keys(defaultSettings)) {
62
- this.properties.set(key, new PluginSettingsProperty(defaultSettings[key], this.validators.get(key) ?? noop));
80
+ for (const propertyName of Object.keys(this.defaultSettings)) {
81
+ this.properties.set(
82
+ propertyName,
83
+ new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? noop)
84
+ );
63
85
  }
64
86
  this.validators.clear();
65
- this.safeSettings = new Proxy(defaultSettings, {
66
- get: (_target, prop) => {
67
- if (typeof prop !== "string") {
68
- return void 0;
69
- }
70
- return this.properties.get(prop);
71
- }
72
- });
73
- this.addValidators();
87
+ this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
74
88
  }
75
89
  safeSettings;
90
+ defaultSettings;
76
91
  properties;
77
92
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
78
93
  validators = /* @__PURE__ */ new Map();
79
- getProperty(property) {
80
- return this.properties.getTyped(property);
94
+ async editAndSave(editor) {
95
+ const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
96
+ await editor(editableSettings);
97
+ await editableSettings.validationPromise;
98
+ await this.saveToFile();
99
+ }
100
+ getProperty(propertyName) {
101
+ return this.properties.getTyped(propertyName);
81
102
  }
82
103
  async loadFromFile() {
83
104
  for (const property of this.properties.values()) {
@@ -94,18 +115,23 @@ class PluginSettingsManagerBase {
94
115
  }
95
116
  let record = data;
96
117
  record = this.getTransformer().transformObjectRecursively(record);
118
+ const beforePrepareJson = JSON.stringify(record);
97
119
  await this.prepareRecord(record);
120
+ const afterPrepareJson = JSON.stringify(record);
98
121
  for (const [key, value] of Object.entries(record)) {
99
- const propertyObj = this.properties.get(key);
100
- if (!propertyObj) {
122
+ const property = this.properties.get(key);
123
+ if (!property) {
101
124
  console.warn(`Unknown property: ${key}`);
102
125
  continue;
103
126
  }
104
- if (typeof value !== typeof propertyObj.defaultValue) {
105
- console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);
127
+ if (typeof value !== typeof property.defaultValue) {
128
+ console.warn(`Invalid value type. Expected ${typeof property.defaultValue}, got: ${typeof value}`);
106
129
  continue;
107
130
  }
108
- await propertyObj.set(value);
131
+ await property.setAndValidate(value);
132
+ }
133
+ if (afterPrepareJson !== beforePrepareJson) {
134
+ await this.saveToFile();
109
135
  }
110
136
  }
111
137
  /**
@@ -114,11 +140,16 @@ class PluginSettingsManagerBase {
114
140
  * @returns A promise that resolves when the settings are saved.
115
141
  */
116
142
  async saveToFile() {
117
- const record = this.getTransformer().transformObjectRecursively(this.getSettings());
143
+ const oldSettings = this.getSavedSettings();
144
+ for (const property of this.properties.values()) {
145
+ property.save();
146
+ }
147
+ const record = this.getTransformer().transformObjectRecursively(this.getSettingsRecord());
118
148
  await this.plugin.saveData(record);
149
+ await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
119
150
  }
120
- addValidator(property, validator) {
121
- this.validators.set(property, validator);
151
+ addValidator(propertyName, validator) {
152
+ this.validators.set(propertyName, validator);
122
153
  }
123
154
  addValidators() {
124
155
  noop();
@@ -129,15 +160,79 @@ class PluginSettingsManagerBase {
129
160
  prepareRecord(_record) {
130
161
  noop();
131
162
  }
132
- getSettings() {
163
+ getSavedSettings() {
164
+ const savedSettings = {};
165
+ for (const [propertyName, property] of this.properties.entries()) {
166
+ savedSettings[propertyName] = property.getSaved();
167
+ }
168
+ const proto = Object.getPrototypeOf(this.defaultSettings);
169
+ Object.setPrototypeOf(savedSettings, proto);
170
+ return savedSettings;
171
+ }
172
+ getSettingsRecord() {
133
173
  const settings = {};
134
- for (const [key, property] of this.properties.entries()) {
135
- settings[key] = property.get();
174
+ for (const [propertyName, property] of this.properties.entries()) {
175
+ settings[propertyName] = property.get();
136
176
  }
137
177
  return settings;
138
178
  }
139
179
  }
180
+ class PluginSettingsProperty {
181
+ constructor(propertyName, defaultValue, validator) {
182
+ this.propertyName = propertyName;
183
+ this.defaultValue = defaultValue;
184
+ this.validator = validator;
185
+ }
186
+ get validationMessage() {
187
+ return this._validationMessage;
188
+ }
189
+ _validationMessage = "";
190
+ savedValue;
191
+ value;
192
+ clear() {
193
+ this.value = void 0;
194
+ this._validationMessage = "";
195
+ }
196
+ get() {
197
+ return this.value ?? this.defaultValue;
198
+ }
199
+ getSafe() {
200
+ return this._validationMessage ? this.defaultValue : this.get();
201
+ }
202
+ getSaved() {
203
+ return this.savedValue;
204
+ }
205
+ save() {
206
+ this.savedValue = this.value;
207
+ }
208
+ set(value) {
209
+ if (isValidationMessageHolder(value)) {
210
+ this._validationMessage = value.validationMessage;
211
+ } else {
212
+ this.value = value;
213
+ }
214
+ }
215
+ async setAndValidate(value) {
216
+ this.set(value);
217
+ if (this.value === void 0) {
218
+ return;
219
+ }
220
+ this._validationMessage = await this.validator(this.value) ?? "";
221
+ if (!this._validationMessage) {
222
+ return;
223
+ }
224
+ const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;
225
+ new Notice(warningMessage);
226
+ console.warn(warningMessage, {
227
+ defaultValue: this.defaultValue,
228
+ propertyName: this.propertyName,
229
+ validationMessage: this._validationMessage,
230
+ value
231
+ });
232
+ }
233
+ }
140
234
  export {
141
- PluginSettingsManagerBase
235
+ PluginSettingsManagerBase,
236
+ PluginSettingsProperty
142
237
  };
143
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport { Plugin } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\n\nimport { noop } from '../../Function.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\n// eslint-disable-next-line @typescript-eslint/no-invalid-void-type\ntype Validator<T> = (value: T) => Promisable<string | void>;\n\nclass PluginSettingsProperty<T> {\n  public validationMessage = '';\n  private value: T | undefined;\n\n  public constructor(public readonly defaultValue: T, private readonly validator: Validator<T>) {}\n\n  public clear(): void {\n    this.value = undefined;\n    this.validationMessage = '';\n  }\n\n  public get(): T {\n    return this.value ?? this.defaultValue;\n  }\n\n  public getSafe(): T {\n    return this.validationMessage ? this.defaultValue : this.get();\n  }\n\n  public async set(value: T | undefined): Promise<void> {\n    this.value = value;\n    if (this.value !== undefined) {\n      this.validationMessage = (await this.validator(this.value) as string | undefined) ?? '';\n    }\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<any>> {\n  public getTyped<Property extends StringKeys<PluginSettings>>(key: Property): PluginSettingsProperty<PluginSettings[Property]> {\n    const property = super.get(key);\n    if (!property) {\n      throw new Error(`Property ${String(key)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings[Property]>;\n  }\n\n  public setTyped<Property extends StringKeys<PluginSettings>>(key: Property, value: PluginSettingsProperty<PluginSettings[Property]>): this {\n    return super.set(key, value);\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private properties: PropertiesMap<PluginSettings>;\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private validators: Map<string, Validator<any>> = new Map<string, Validator<any>>();\n\n  public constructor(private plugin: Plugin) {\n    const defaultSettings = this.createDefaultSettings();\n\n    this.addValidators();\n\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const key of Object.keys(defaultSettings) as StringKeys<PluginSettings>[]) {\n      this.properties.set(key, new PluginSettingsProperty(defaultSettings[key], this.validators.get(key) ?? noop));\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(defaultSettings, {\n      get: (_target, prop): unknown => {\n        if (typeof prop !== 'string') {\n          return undefined;\n        }\n\n        return this.properties.get(prop);\n      }\n    }) as ReadonlyDeep<PluginSettings>;\n\n    this.addValidators();\n  }\n\n  public getProperty<Property extends StringKeys<PluginSettings>>(property: Property): PluginSettingsProperty<PluginSettings[Property]> {\n    return this.properties.getTyped(property);\n  }\n\n  public async loadFromFile(): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.clear();\n    }\n\n    const data = await this.plugin.loadData() as unknown;\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object' || Array.isArray(data)) {\n      const type = Array.isArray(data) ? 'Array' : typeof data;\n      console.error(`Invalid data type. Expected Object, got: ${type}`);\n      return;\n    }\n\n    let record = data as Record<string, unknown>;\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.prepareRecord(record);\n\n    for (const [key, value] of Object.entries(record)) {\n      const propertyObj = this.properties.get(key);\n      if (!propertyObj) {\n        console.warn(`Unknown property: ${key}`);\n        continue;\n      }\n\n      if (typeof value !== typeof propertyObj.defaultValue) {\n        console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);\n        continue;\n      }\n\n      await propertyObj.set(value as PluginSettings[StringKeys<PluginSettings>]);\n    }\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveToFile(): Promise<void> {\n    const record = this.getTransformer().transformObjectRecursively(this.getSettings());\n    await this.plugin.saveData(record);\n  }\n\n  protected addValidator<Property extends StringKeys<PluginSettings>>(property: Property, validator: Validator<PluginSettings[Property]>): void {\n    this.validators.set(property, validator);\n  }\n\n  protected addValidators(): void {\n    noop();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  protected prepareRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  private getSettings(): Record<StringKeys<PluginSettings>, unknown> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [key, property] of this.properties.entries()) {\n      settings[key as StringKeys<PluginSettings>] = property.get() as unknown;\n    }\n\n    return settings;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

238
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type { ValidationMessageHolder } from '../ValidationMessage.ts';\nimport type { PluginBase } from './PluginBase.ts';\n\nimport { noop } from '../../Function.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\nimport { isValidationMessageHolder } from '../ValidationMessage.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\ntype Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;\n\nabstract class ProxyHandlerBase<PluginSettings extends object> implements ProxyHandler<PluginSettings> {\n  public constructor(protected readonly properties: PropertiesMap<PluginSettings>) {}\n\n  public get(target: PluginSettings, prop: string | symbol): unknown {\n    const record = target as Record<string | symbol, unknown>;\n    if (typeof prop !== 'string') {\n      return record[prop];\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return record[prop];\n    }\n\n    return this.getPropertyValue(property);\n  }\n\n  protected abstract getPropertyValue(property: PluginSettingsProperty<unknown>): unknown;\n}\n\nclass EditableSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  private validationPromise = Promise.resolve();\n  public set(target: PluginSettings, prop: string | symbol, value: unknown): boolean {\n    const record = target as Record<string | symbol, unknown>;\n\n    if (typeof prop !== 'string') {\n      record[prop] = value;\n      return true;\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      record[prop] = value;\n      return true;\n    }\n\n    property.set(value);\n    this.validationPromise = this.validationPromise.then(() => property.setAndValidate(value));\n\n    return true;\n  }\n\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.get();\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<any>> {\n  public getTyped<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    const property = super.get(propertyName);\n    if (!property) {\n      throw new Error(`Property ${String(propertyName)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings[PropertyName]>;\n  }\n\n  public setTyped<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettingsProperty<PluginSettings[PropertyName]>\n  ): this {\n    return super.set(propertyName, value);\n  }\n}\n\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.getSafe();\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private defaultSettings: PluginSettings;\n  private properties: PropertiesMap<PluginSettings>;\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private validators: Map<string, Validator<any>> = new Map<string, Validator<any>>();\n\n  public constructor(private plugin: PluginBase<PluginSettings>) {\n    this.defaultSettings = this.createDefaultSettings();\n\n    this.addValidators();\n\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const propertyName of Object.keys(this.defaultSettings) as StringKeys<PluginSettings>[]) {\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? noop)\n      );\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler<PluginSettings>(this.properties)) as ReadonlyDeep<PluginSettings>;\n  }\n\n  public async editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler<PluginSettings>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & PluginSettings;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile();\n  }\n\n  public getProperty<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    return this.properties.getTyped(propertyName);\n  }\n\n  public async loadFromFile(): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.clear();\n    }\n\n    const data = await this.plugin.loadData() as unknown;\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object' || Array.isArray(data)) {\n      const type = Array.isArray(data) ? 'Array' : typeof data;\n      console.error(`Invalid data type. Expected Object, got: ${type}`);\n      return;\n    }\n\n    let record = data as Record<string, unknown>;\n    record = this.getTransformer().transformObjectRecursively(record);\n    const beforePrepareJson = JSON.stringify(record);\n    await this.prepareRecord(record);\n    const afterPrepareJson = JSON.stringify(record);\n\n    for (const [key, value] of Object.entries(record)) {\n      const property = this.properties.get(key);\n      if (!property) {\n        console.warn(`Unknown property: ${key}`);\n        continue;\n      }\n\n      if (typeof value !== typeof property.defaultValue) {\n        console.warn(`Invalid value type. Expected ${typeof property.defaultValue}, got: ${typeof value}`);\n        continue;\n      }\n\n      await property.setAndValidate(value);\n    }\n\n    if (afterPrepareJson !== beforePrepareJson) {\n      await this.saveToFile();\n    }\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveToFile(): Promise<void> {\n    const oldSettings = this.getSavedSettings();\n\n    for (const property of this.properties.values()) {\n      property.save();\n    }\n\n    const record = this.getTransformer().transformObjectRecursively(this.getSettingsRecord());\n    await this.plugin.saveData(record);\n\n    await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);\n  }\n\n  protected addValidator<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: Validator<PluginSettings[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  protected addValidators(): void {\n    noop();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  protected prepareRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  private getSavedSettings(): Partial<PluginSettings> {\n    const savedSettings: Partial<PluginSettings> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName as StringKeys<PluginSettings>] = property.getSaved() as PluginSettings[StringKeys<PluginSettings>] | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.defaultSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n    return savedSettings;\n  }\n\n  private getSettingsRecord(): Record<StringKeys<PluginSettings>, unknown> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName as StringKeys<PluginSettings>] = property.get() as unknown;\n    }\n\n    return settings;\n  }\n}\n\n/**\n * A property of a plugin settings.\n *\n * @typeParam T - The type of the property.\n */\nexport class PluginSettingsProperty<T> {\n  public get validationMessage(): string {\n    return this._validationMessage;\n  }\n\n  private _validationMessage = '';\n\n  private savedValue: T | undefined;\n\n  private value: T | undefined;\n  public constructor(private readonly propertyName: string, public readonly defaultValue: T, private readonly validator: Validator<T>) {}\n\n  public clear(): void {\n    this.value = undefined;\n    this._validationMessage = '';\n  }\n\n  public get(): T {\n    return this.value ?? this.defaultValue;\n  }\n\n  public getSafe(): T {\n    return this._validationMessage ? this.defaultValue : this.get();\n  }\n\n  public getSaved(): T | undefined {\n    return this.savedValue;\n  }\n\n  public save(): void {\n    this.savedValue = this.value;\n  }\n\n  public set(value: T | undefined | ValidationMessageHolder): void {\n    if (isValidationMessageHolder(value)) {\n      this._validationMessage = value.validationMessage;\n    } else {\n      this.value = value;\n    }\n  }\n\n  public async setAndValidate(value: T | undefined | ValidationMessageHolder): Promise<void> {\n    this.set(value);\n    if (this.value === undefined) {\n      return;\n    }\n\n    this._validationMessage = (await this.validator(this.value) as string | undefined) ?? '';\n\n    if (!this._validationMessage) {\n      return;\n    }\n\n    const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;\n    new Notice(warningMessage);\n    console.warn(warningMessage, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -3,19 +3,20 @@
3
3
  * This module defines a base class for creating plugin setting tabs in Obsidian.
4
4
  * It provides a utility method to bind value components to plugin settings and handle changes.
5
5
  */
6
- import type { ConditionalKeys, Promisable, WritableDeep } from 'type-fest';
6
+ import type { ConditionalKeys, Promisable } from 'type-fest';
7
7
  import { PluginSettingTab } from 'obsidian';
8
8
  import type { StringKeys } from '../../Object.mjs';
9
9
  import type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.mjs';
10
+ import type { ValidationMessageHolder } from '../ValidationMessage.mjs';
10
11
  import type { PluginBase } from './PluginBase.mjs';
11
12
  /**
12
13
  * Options for binding a value component to a plugin setting.
13
14
  */
14
- export interface BindOptions {
15
+ export interface BindOptions<T> {
15
16
  /**
16
17
  * A callback function that is called when the value of the component changes.
17
18
  */
18
- onChanged?(): Promisable<void>;
19
+ onChanged?(newValue: T | undefined, oldValue: T): Promisable<void>;
19
20
  /**
20
21
  * If true, shows the validation message when the component value is invalid. Default is `true`.
21
22
  */
@@ -24,24 +25,21 @@ export interface BindOptions {
24
25
  /**
25
26
  * Extended options for binding a value component to a plugin setting.
26
27
  */
27
- export interface BindOptionsExtended<PluginSettings extends object, UIValue, Property extends StringKeys<PluginSettings>> extends BindOptions {
28
+ export interface BindOptionsExtended<PluginSettings extends object, UIValue, PropertyName extends StringKeys<PluginSettings>> extends BindOptions<PluginSettings[PropertyName]> {
28
29
  /**
29
30
  * Converts the UI component's value back to the plugin settings value.
30
31
  * @param uiValue - The value of the UI component.
31
32
  * @returns The value to set on the plugin settings.
32
33
  */
33
- componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[Property] | ValidationMessageHolder;
34
+ componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;
34
35
  /**
35
36
  * Converts the plugin settings value to the value used by the UI component.
36
37
  * @param pluginSettingsValue - The value of the property in the plugin settings.
37
38
  * @returns The value to set on the UI component.
38
39
  */
39
- pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[Property]) => UIValue;
40
- }
41
- type ExtractPluginSettings<T extends PluginBase<any>> = WritableDeep<T['settings']>;
42
- interface ValidationMessageHolder {
43
- validationMessage: string;
40
+ pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[PropertyName]) => UIValue;
44
41
  }
42
+ type ExtractPluginSettings<Plugin extends PluginBase<any>> = Plugin extends PluginBase<infer PluginSettings> ? PluginSettings : never;
45
43
  /**
46
44
  * Base class for creating plugin settings tabs in Obsidian.
47
45
  * Provides a method for binding value components to plugin settings and handling changes.
@@ -57,23 +55,23 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
57
55
  * @typeParam UIValue - The type of the value of the UI component.
58
56
  * @typeParam TValueComponent - The type of the value component.
59
57
  * @param valueComponent - The value component to bind.
60
- * @param property - The property of the plugin settings to bind to.
58
+ * @param propertyName - The property of the plugin settings to bind to.
61
59
  * @param options - The options for binding the value component.
62
60
  * @returns The value component.
63
61
  */
64
- bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions): TValueComponent;
62
+ bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions<UIValue>): TValueComponent;
65
63
  /**
66
64
  * Binds a value component to a plugin setting.
67
65
  *
68
66
  * @typeParam UIValue - The type of the value of the UI component.
69
67
  * @typeParam TValueComponent - The type of the value component.
70
- * @typeParam Property - The property of the plugin settings to bind to.
68
+ * @typeParam PropertyName - The property name of the plugin settings to bind to.
71
69
  * @param valueComponent - The value component to bind.
72
- * @param property - The property of the plugin settings to bind to.
70
+ * @param propertyName - The property name of the plugin settings to bind to.
73
71
  * @param options - The options for binding the value component.
74
72
  * @returns The value component.
75
73
  */
76
- bind<UIValue, TValueComponent, Property extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: Property, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, Property>): TValueComponent;
74
+ bind<UIValue, TValueComponent, PropertyName extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: PropertyName, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, PropertyName>): TValueComponent;
77
75
  hide(): void;
78
76
  }
79
77
  export {};
@@ -9,7 +9,9 @@ import { PluginSettingTab } from "obsidian";
9
9
  import { invokeAsyncSafely } from "../../Async.mjs";
10
10
  import { CssClass } from "../../CssClass.mjs";
11
11
  import { noop } from "../../Function.mjs";
12
+ import { isPlaceholderComponent } from "../Components/PlaceholderComponent.mjs";
12
13
  import { getValidatorComponent } from "../Components/ValidatorComponent.mjs";
14
+ import { isValidationMessageHolder } from "../ValidationMessage.mjs";
13
15
  import { getPluginId } from "./PluginId.mjs";
14
16
  class PluginSettingsTabBase extends PluginSettingTab {
15
17
  constructor(plugin) {
@@ -22,13 +24,13 @@ class PluginSettingsTabBase extends PluginSettingTab {
22
24
  *
23
25
  * @typeParam UIValue - The type of the value of the UI component.
24
26
  * @typeParam TValueComponent - The type of the value component.
25
- * @typeParam Property - The property of the plugin settings to bind to.
27
+ * @typeParam PropertyName - The property name of the plugin settings to bind to.
26
28
  * @param valueComponent - The value component to bind.
27
- * @param property - The property of the plugin settings to bind to.
29
+ * @param propertyName - The property name of the plugin settings to bind to.
28
30
  * @param options - The options for binding the value component.
29
31
  * @returns The value component.
30
32
  */
31
- bind(valueComponent, property, options) {
33
+ bind(valueComponent, propertyName, options) {
32
34
  const DEFAULT_OPTIONS = {
33
35
  // eslint-disable-next-line @typescript-eslint/no-unsafe-return
34
36
  componentToPluginSettingsValueConverter: (value) => value,
@@ -38,17 +40,17 @@ class PluginSettingsTabBase extends PluginSettingTab {
38
40
  };
39
41
  const optionsExt = { ...DEFAULT_OPTIONS, ...options };
40
42
  const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;
41
- const propertyObj = this.plugin.settingsManager.getProperty(property);
42
- valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(propertyObj.get())).onChange(async (uiValue) => {
43
+ const property = this.plugin.settingsManager.getProperty(propertyName);
44
+ valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(property.get())).onChange(async (uiValue) => {
45
+ const oldValue = property.get();
43
46
  const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);
44
- if (isValidationMessageHolder(convertedValue)) {
45
- propertyObj.validationMessage = convertedValue.validationMessage;
46
- await propertyObj.set(void 0);
47
- } else {
48
- await propertyObj.set(convertedValue);
49
- }
50
- await optionsExt.onChanged();
47
+ await property.setAndValidate(convertedValue);
48
+ const newValue = isValidationMessageHolder(convertedValue) ? void 0 : convertedValue;
49
+ await optionsExt.onChanged(newValue, oldValue);
51
50
  });
51
+ if (isPlaceholderComponent(valueComponent)) {
52
+ valueComponent.setPlaceholder(optionsExt.pluginSettingsToComponentValueConverter(property.defaultValue));
53
+ }
52
54
  validatorElement?.addEventListener("focus", validate);
53
55
  validatorElement?.addEventListener("blur", validate);
54
56
  validate();
@@ -57,13 +59,13 @@ class PluginSettingsTabBase extends PluginSettingTab {
57
59
  if (!validatorElement) {
58
60
  return;
59
61
  }
60
- if (!propertyObj.validationMessage) {
62
+ if (!property.validationMessage) {
61
63
  validatorElement.setCustomValidity("");
62
64
  validatorElement.checkValidity();
63
- propertyObj.validationMessage = validatorElement.validationMessage;
65
+ property.set(validatorElement);
64
66
  }
65
- validatorElement.setCustomValidity(propertyObj.validationMessage);
66
- validatorElement.title = propertyObj.validationMessage;
67
+ validatorElement.setCustomValidity(property.validationMessage);
68
+ validatorElement.title = property.validationMessage;
67
69
  if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {
68
70
  validatorElement.reportValidity();
69
71
  }
@@ -74,10 +76,7 @@ class PluginSettingsTabBase extends PluginSettingTab {
74
76
  invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile());
75
77
  }
76
78
  }
77
- function isValidationMessageHolder(value) {
78
- return !!value.validationMessage;
79
- }
80
79
  export {
81
80
  PluginSettingsTabBase
82
81
  };
83
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginSettingsTabBase\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 {\n  ConditionalKeys,\n  Promisable,\n  WritableDeep\n} from 'type-fest';\n\nimport { PluginSettingTab } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.ts';\nimport type { PluginBase } from './PluginBase.ts';\n\nimport { invokeAsyncSafely } from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.ts';\nimport { getValidatorComponent } from '../Components/ValidatorComponent.ts';\nimport { getPluginId } from './PluginId.ts';\n\n/**\n * Options for binding a value component to a plugin setting.\n */\nexport interface BindOptions {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  onChanged?(): Promisable<void>;\n\n  /**\n   * If true, shows 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<PluginSettings extends object, UIValue, Property extends StringKeys<PluginSettings>> extends BindOptions {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\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[Property] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\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: PluginSettings[Property]) => UIValue;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype ExtractPluginSettings<T extends PluginBase<any>> = WritableDeep<T['settings']>;\n\ninterface ValidationMessageHolder {\n  validationMessage: string;\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 TPlugin - The type of the plugin that extends PluginBase.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport abstract class PluginSettingsTabBase<TPlugin extends PluginBase<any>> extends PluginSettingTab {\n  public constructor(public override plugin: TPlugin) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\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 property - 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    property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>,\n    options?: BindOptions\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 Property - The property of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param property - 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    Property extends StringKeys<ExtractPluginSettings<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    property: Property,\n    options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, Property>\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 Property - The property of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param property - 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    Property extends StringKeys<ExtractPluginSettings<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    property: Property,\n    options?: BindOptions\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<TPlugin>;\n    type PropertyType = PluginSettings[Property];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, Property>> = {\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: PropertyType): UIValue => value as UIValue,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, Property>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const propertyObj = this.plugin.settingsManager.getProperty(property);\n\n    valueComponent\n      .setValue(optionsExt.pluginSettingsToComponentValueConverter(propertyObj.get() as PropertyType))\n      .onChange(async (uiValue) => {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          propertyObj.validationMessage = convertedValue.validationMessage;\n          await propertyObj.set(undefined);\n        } else {\n          await propertyObj.set(convertedValue);\n        }\n        await optionsExt.onChanged();\n      });\n\n    validatorElement?.addEventListener('focus', validate);\n    validatorElement?.addEventListener('blur', validate);\n\n    validate();\n    return valueComponent;\n\n    function validate(): void {\n      if (!validatorElement) {\n        return;\n      }\n\n      if (!propertyObj.validationMessage) {\n        validatorElement.setCustomValidity('');\n        validatorElement.checkValidity();\n        propertyObj.validationMessage = validatorElement.validationMessage;\n      }\n\n      validatorElement.setCustomValidity(propertyObj.validationMessage);\n      validatorElement.title = propertyObj.validationMessage;\n      if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {\n        validatorElement.reportValidity();\n      }\n    }\n  }\n\n  public override hide(): void {\n    super.hide();\n    invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile());\n  }\n}\n\nfunction isValidationMessageHolder(value: unknown): value is ValidationMessageHolder {\n  return !!(value as Partial<ValidationMessageHolder>).validationMessage;\n}\n"],
  "mappings": ";;;;;;;AAYA,SAAS,wBAAwB;AAMjC,SAAS,yBAAyB;AAClC,SAAS,gBAAgB;AACzB,SAAS,YAAY;AACrB,SAAS,6BAA6B;AACtC,SAAS,mBAAmB;AAkDrB,MAAe,8BAA+D,iBAAiB;AAAA,EAC7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,SAAS,aAAa,YAAY,GAAG,SAAS,iBAAiB;AAAA,EAC3F;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmDO,KAKL,gBACA,UACA,SACiB;AAGjB,UAAM,kBAAoF;AAAA;AAAA,MAExF,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,WAAW;AAAA,MACX,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,6BAA6B;AAAA,IAC/B;AAEA,UAAM,aAA+E,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAEtH,UAAM,mBAAmB,sBAAsB,cAAc,GAAG;AAEhE,UAAM,cAAc,KAAK,OAAO,gBAAgB,YAAY,QAAQ;AAEpE,mBACG,SAAS,WAAW,wCAAwC,YAAY,IAAI,CAAiB,CAAC,EAC9F,SAAS,OAAO,YAAY;AAC3B,YAAM,iBAAiB,WAAW,wCAAwC,OAAO;AACjF,UAAI,0BAA0B,cAAc,GAAG;AAC7C,oBAAY,oBAAoB,eAAe;AAC/C,cAAM,YAAY,IAAI,MAAS;AAAA,MACjC,OAAO;AACL,cAAM,YAAY,IAAI,cAAc;AAAA,MACtC;AACA,YAAM,WAAW,UAAU;AAAA,IAC7B,CAAC;AAEH,sBAAkB,iBAAiB,SAAS,QAAQ;AACpD,sBAAkB,iBAAiB,QAAQ,QAAQ;AAEnD,aAAS;AACT,WAAO;AAEP,aAAS,WAAiB;AACxB,UAAI,CAAC,kBAAkB;AACrB;AAAA,MACF;AAEA,UAAI,CAAC,YAAY,mBAAmB;AAClC,yBAAiB,kBAAkB,EAAE;AACrC,yBAAiB,cAAc;AAC/B,oBAAY,oBAAoB,iBAAiB;AAAA,MACnD;AAEA,uBAAiB,kBAAkB,YAAY,iBAAiB;AAChE,uBAAiB,QAAQ,YAAY;AACrC,UAAI,iBAAiB,gBAAgB,KAAK,WAAW,6BAA6B;AAChF,yBAAiB,eAAe;AAAA,MAClC;AAAA,IACF;AAAA,EACF;AAAA,EAEgB,OAAa;AAC3B,UAAM,KAAK;AACX,sBAAkB,MAAM,KAAK,OAAO,gBAAgB,WAAW,CAAC;AAAA,EAClE;AACF;AAEA,SAAS,0BAA0B,OAAkD;AACnF,SAAO,CAAC,CAAE,MAA2C;AACvD;",
  "names": []
}

82
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginSettingsTabBase\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 {\n  ConditionalKeys,\n  Promisable\n} from 'type-fest';\n\nimport { PluginSettingTab } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.ts';\nimport type { ValidationMessageHolder } from '../ValidationMessage.ts';\nimport type { PluginBase } from './PluginBase.ts';\nimport type { PluginSettingsProperty } from './PluginSettingsManagerBase.ts';\n\nimport { invokeAsyncSafely } from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.ts';\nimport { isPlaceholderComponent } from '../Components/PlaceholderComponent.ts';\nimport { getValidatorComponent } from '../Components/ValidatorComponent.ts';\nimport { isValidationMessageHolder } from '../ValidationMessage.ts';\nimport { getPluginId } from './PluginId.ts';\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: T | undefined, oldValue: T): Promisable<void>;\n\n  /**\n   * If true, shows 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   * @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   * @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: PluginSettings[PropertyName]) => UIValue;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype ExtractPluginSettings<Plugin extends PluginBase<any>> = Plugin extends PluginBase<infer PluginSettings> ? PluginSettings : never;\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 TPlugin - The type of the plugin that extends PluginBase.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport abstract class PluginSettingsTabBase<TPlugin extends PluginBase<any>> extends PluginSettingTab {\n  public constructor(public override plugin: TPlugin) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\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<TPlugin>, 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<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, 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<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<TPlugin>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<TPlugin>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: PropertyType): UIValue => value as UIValue,\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 property = this.plugin.settingsManager.getProperty(propertyName) as PluginSettingsProperty<PropertyType>;\n\n    valueComponent\n      .setValue(optionsExt.pluginSettingsToComponentValueConverter(property.get()))\n      .onChange(async (uiValue) => {\n        const oldValue = property.get();\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        await property.setAndValidate(convertedValue);\n        const newValue = isValidationMessageHolder(convertedValue) ? undefined : convertedValue;\n        await optionsExt.onChanged(newValue, oldValue);\n      });\n\n    if (isPlaceholderComponent(valueComponent)) {\n      valueComponent.setPlaceholder(optionsExt.pluginSettingsToComponentValueConverter(property.defaultValue) as string);\n    }\n\n    validatorElement?.addEventListener('focus', validate);\n    validatorElement?.addEventListener('blur', validate);\n\n    validate();\n    return valueComponent;\n\n    function validate(): void {\n      if (!validatorElement) {\n        return;\n      }\n\n      if (!property.validationMessage) {\n        validatorElement.setCustomValidity('');\n        validatorElement.checkValidity();\n        property.set(validatorElement);\n      }\n\n      validatorElement.setCustomValidity(property.validationMessage);\n      validatorElement.title = property.validationMessage;\n      if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {\n        validatorElement.reportValidity();\n      }\n    }\n  }\n\n  public override hide(): void {\n    super.hide();\n    invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile());\n  }\n}\n"],
  "mappings": ";;;;;;;AAWA,SAAS,wBAAwB;AAQjC,SAAS,yBAAyB;AAClC,SAAS,gBAAgB;AACzB,SAAS,YAAY;AACrB,SAAS,8BAA8B;AACvC,SAAS,6BAA6B;AACtC,SAAS,iCAAiC;AAC1C,SAAS,mBAAmB;AAkDrB,MAAe,8BAA+D,iBAAiB;AAAA,EAC7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,SAAS,aAAa,YAAY,GAAG,SAAS,iBAAiB;AAAA,EAC3F;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmDO,KAKL,gBACA,cACA,SACiB;AAGjB,UAAM,kBAAwF;AAAA;AAAA,MAE5F,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,WAAW;AAAA,MACX,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,6BAA6B;AAAA,IAC/B;AAEA,UAAM,aAAmF,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAE1H,UAAM,mBAAmB,sBAAsB,cAAc,GAAG;AAEhE,UAAM,WAAW,KAAK,OAAO,gBAAgB,YAAY,YAAY;AAErE,mBACG,SAAS,WAAW,wCAAwC,SAAS,IAAI,CAAC,CAAC,EAC3E,SAAS,OAAO,YAAY;AAC3B,YAAM,WAAW,SAAS,IAAI;AAC9B,YAAM,iBAAiB,WAAW,wCAAwC,OAAO;AACjF,YAAM,SAAS,eAAe,cAAc;AAC5C,YAAM,WAAW,0BAA0B,cAAc,IAAI,SAAY;AACzE,YAAM,WAAW,UAAU,UAAU,QAAQ;AAAA,IAC/C,CAAC;AAEH,QAAI,uBAAuB,cAAc,GAAG;AAC1C,qBAAe,eAAe,WAAW,wCAAwC,SAAS,YAAY,CAAW;AAAA,IACnH;AAEA,sBAAkB,iBAAiB,SAAS,QAAQ;AACpD,sBAAkB,iBAAiB,QAAQ,QAAQ;AAEnD,aAAS;AACT,WAAO;AAEP,aAAS,WAAiB;AACxB,UAAI,CAAC,kBAAkB;AACrB;AAAA,MACF;AAEA,UAAI,CAAC,SAAS,mBAAmB;AAC/B,yBAAiB,kBAAkB,EAAE;AACrC,yBAAiB,cAAc;AAC/B,iBAAS,IAAI,gBAAgB;AAAA,MAC/B;AAEA,uBAAiB,kBAAkB,SAAS,iBAAiB;AAC7D,uBAAiB,QAAQ,SAAS;AAClC,UAAI,iBAAiB,gBAAgB,KAAK,WAAW,6BAA6B;AAChF,yBAAiB,eAAe;AAAA,MAClC;AAAA,IACF;AAAA,EACF;AAAA,EAEgB,OAAa;AAC3B,UAAM,KAAK;AACX,sBAAkB,MAAM,KAAK,OAAO,gBAAgB,WAAW,CAAC;AAAA,EAClE;AACF;",
  "names": []
}

@@ -0,0 +1,16 @@
1
+ /**
2
+ * Holds a validation message.
3
+ */
4
+ export interface ValidationMessageHolder {
5
+ /**
6
+ * The validation message.
7
+ */
8
+ validationMessage: string;
9
+ }
10
+ /**
11
+ * Type guard to check if a value is a validation message holder.
12
+ *
13
+ * @param value - The value to check.
14
+ * @returns `true` if the value is a validation message holder, `false` otherwise.
15
+ */
16
+ export declare function isValidationMessageHolder(value: unknown): value is ValidationMessageHolder;
@@ -0,0 +1,14 @@
1
+ /*
2
+ THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
3
+ if you want to view the source, please visit the github repository of this plugin
4
+ */
5
+
6
+ (function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
7
+
8
+ function isValidationMessageHolder(value) {
9
+ return value.validationMessage !== void 0;
10
+ }
11
+ export {
12
+ isValidationMessageHolder
13
+ };
14
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1ZhbGlkYXRpb25NZXNzYWdlLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEhvbGRzIGEgdmFsaWRhdGlvbiBtZXNzYWdlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgLyoqXG4gICAqIFRoZSB2YWxpZGF0aW9uIG1lc3NhZ2UuXG4gICAqL1xuICB2YWxpZGF0aW9uTWVzc2FnZTogc3RyaW5nO1xufVxuXG4vKipcbiAqIFR5cGUgZ3VhcmQgdG8gY2hlY2sgaWYgYSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIuXG4gKlxuICogQHBhcmFtIHZhbHVlIC0gVGhlIHZhbHVlIHRvIGNoZWNrLlxuICogQHJldHVybnMgYHRydWVgIGlmIHRoZSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIsIGBmYWxzZWAgb3RoZXJ3aXNlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNWYWxpZGF0aW9uTWVzc2FnZUhvbGRlcih2YWx1ZTogdW5rbm93bik6IHZhbHVlIGlzIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgcmV0dXJuICh2YWx1ZSBhcyBQYXJ0aWFsPFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyPikudmFsaWRhdGlvbk1lc3NhZ2UgIT09IHVuZGVmaW5lZDtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFnQk8sU0FBUywwQkFBMEIsT0FBa0Q7QUFDMUYsU0FBUSxNQUEyQyxzQkFBc0I7QUFDM0U7IiwKICAibmFtZXMiOiBbXQp9Cg==
@@ -27,5 +27,6 @@ export * as Reference from './Reference.mjs';
27
27
  export * as RenameDeleteHandler from './RenameDeleteHandler.mjs';
28
28
  export * as ResourceUrl from './ResourceUrl.mjs';
29
29
  export * as SettingEx from './SettingEx.mjs';
30
+ export * as ValidationMessage from './ValidationMessage.mjs';
30
31
  export * as Vault from './Vault.mjs';
31
32
  export * as VaultEx from './VaultEx.mjs';
@@ -34,6 +34,7 @@ import * as Reference from "./Reference.mjs";
34
34
  import * as RenameDeleteHandler from "./RenameDeleteHandler.mjs";
35
35
  import * as ResourceUrl from "./ResourceUrl.mjs";
36
36
  import * as SettingEx from "./SettingEx.mjs";
37
+ import * as ValidationMessage from "./ValidationMessage.mjs";
37
38
  import * as Vault from "./Vault.mjs";
38
39
  import * as VaultEx from "./VaultEx.mjs";
39
40
  export {
@@ -66,7 +67,8 @@ export {
66
67
  RenameDeleteHandler,
67
68
  ResourceUrl,
68
69
  SettingEx,
70
+ ValidationMessage,
69
71
  Vault,
70
72
  VaultEx
71
73
  };
72
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKiBUSElTIElTIEEgR0VORVJBVEVEL0JVTkRMRUQgRklMRSBCWSBCVUlMRCBTQ1JJUFQgKi9cblxuZXhwb3J0ICogYXMgQXBwIGZyb20gJy4vQXBwLnRzJztcbmV4cG9ydCAqIGFzIEF0dGFjaG1lbnRQYXRoIGZyb20gJy4vQXR0YWNobWVudFBhdGgudHMnO1xuZXhwb3J0ICogYXMgQmFja2xpbmsgZnJvbSAnLi9CYWNrbGluay50cyc7XG5leHBvcnQgKiBhcyBDYWxsb3V0IGZyb20gJy4vQ2FsbG91dC50cyc7XG5leHBvcnQgKiBhcyBDb21wb25lbnRzIGZyb20gJy4vQ29tcG9uZW50cy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBEYXRhdmlldyBmcm9tICcuL0RhdGF2aWV3LnRzJztcbmV4cG9ydCAqIGFzIERhdGF2aWV3TGluayBmcm9tICcuL0RhdGF2aWV3TGluay50cyc7XG5leHBvcnQgKiBhcyBGaWxlQ2hhbmdlIGZyb20gJy4vRmlsZUNoYW5nZS50cyc7XG5leHBvcnQgKiBhcyBGaWxlTWFuYWdlciBmcm9tICcuL0ZpbGVNYW5hZ2VyLnRzJztcbmV4cG9ydCAqIGFzIEZpbGVTeXN0ZW0gZnJvbSAnLi9GaWxlU3lzdGVtLnRzJztcbmV4cG9ydCAqIGFzIEZyb250bWF0dGVyIGZyb20gJy4vRnJvbnRtYXR0ZXIudHMnO1xuZXhwb3J0ICogYXMgTGluayBmcm9tICcuL0xpbmsudHMnO1xuZXhwb3J0ICogYXMgTG9nZ2VyIGZyb20gJy4vTG9nZ2VyLnRzJztcbmV4cG9ydCAqIGFzIExvb3AgZnJvbSAnLi9Mb29wLnRzJztcbmV4cG9ydCAqIGFzIE1hcmtkb3duIGZyb20gJy4vTWFya2Rvd24udHMnO1xuZXhwb3J0ICogYXMgTWFya2Rvd25Db2RlQmxvY2tQcm9jZXNzb3IgZnJvbSAnLi9NYXJrZG93bkNvZGVCbG9ja1Byb2Nlc3Nvci50cyc7XG5leHBvcnQgKiBhcyBNYXJrZG93blZpZXcgZnJvbSAnLi9NYXJrZG93blZpZXcudHMnO1xuZXhwb3J0ICogYXMgTWV0YWRhdGFDYWNoZSBmcm9tICcuL01ldGFkYXRhQ2FjaGUudHMnO1xuZXhwb3J0ICogYXMgTW9kYWxzIGZyb20gJy4vTW9kYWxzL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIE1vbmtleUFyb3VuZCBmcm9tICcuL01vbmtleUFyb3VuZC50cyc7XG5leHBvcnQgKiBhcyBPYnNpZGlhblNldHRpbmdzIGZyb20gJy4vT2JzaWRpYW5TZXR0aW5ncy50cyc7XG5leHBvcnQgKiBhcyBQZGYgZnJvbSAnLi9QZGYudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luIGZyb20gJy4vUGx1Z2luL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIFF1ZXVlIGZyb20gJy4vUXVldWUudHMnO1xuZXhwb3J0ICogYXMgUmVhY3QgZnJvbSAnLi9SZWFjdC9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBSZWZlcmVuY2UgZnJvbSAnLi9SZWZlcmVuY2UudHMnO1xuZXhwb3J0ICogYXMgUmVuYW1lRGVsZXRlSGFuZGxlciBmcm9tICcuL1JlbmFtZURlbGV0ZUhhbmRsZXIudHMnO1xuZXhwb3J0ICogYXMgUmVzb3VyY2VVcmwgZnJvbSAnLi9SZXNvdXJjZVVybC50cyc7XG5leHBvcnQgKiBhcyBTZXR0aW5nRXggZnJvbSAnLi9TZXR0aW5nRXgudHMnO1xuZXhwb3J0ICogYXMgVmF1bHQgZnJvbSAnLi9WYXVsdC50cyc7XG5leHBvcnQgKiBhcyBWYXVsdEV4IGZyb20gJy4vVmF1bHRFeC50cyc7XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7O0FBRUEsWUFBWSxTQUFTO0FBQ3JCLFlBQVksb0JBQW9CO0FBQ2hDLFlBQVksY0FBYztBQUMxQixZQUFZLGFBQWE7QUFDekIsWUFBWSxnQkFBZ0I7QUFDNUIsWUFBWSxjQUFjO0FBQzFCLFlBQVksa0JBQWtCO0FBQzlCLFlBQVksZ0JBQWdCO0FBQzVCLFlBQVksaUJBQWlCO0FBQzdCLFlBQVksZ0JBQWdCO0FBQzVCLFlBQVksaUJBQWlCO0FBQzdCLFlBQVksVUFBVTtBQUN0QixZQUFZLFlBQVk7QUFDeEIsWUFBWSxVQUFVO0FBQ3RCLFlBQVksY0FBYztBQUMxQixZQUFZLGdDQUFnQztBQUM1QyxZQUFZLGtCQUFrQjtBQUM5QixZQUFZLG1CQUFtQjtBQUMvQixZQUFZLFlBQVk7QUFDeEIsWUFBWSxrQkFBa0I7QUFDOUIsWUFBWSxzQkFBc0I7QUFDbEMsWUFBWSxTQUFTO0FBQ3JCLFlBQVksWUFBWTtBQUN4QixZQUFZLFdBQVc7QUFDdkIsWUFBWSxXQUFXO0FBQ3ZCLFlBQVksZUFBZTtBQUMzQixZQUFZLHlCQUF5QjtBQUNyQyxZQUFZLGlCQUFpQjtBQUM3QixZQUFZLGVBQWU7QUFDM0IsWUFBWSxXQUFXO0FBQ3ZCLFlBQVksYUFBYTsiLAogICJuYW1lcyI6IFtdCn0K
74
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKiBUSElTIElTIEEgR0VORVJBVEVEL0JVTkRMRUQgRklMRSBCWSBCVUlMRCBTQ1JJUFQgKi9cblxuZXhwb3J0ICogYXMgQXBwIGZyb20gJy4vQXBwLnRzJztcbmV4cG9ydCAqIGFzIEF0dGFjaG1lbnRQYXRoIGZyb20gJy4vQXR0YWNobWVudFBhdGgudHMnO1xuZXhwb3J0ICogYXMgQmFja2xpbmsgZnJvbSAnLi9CYWNrbGluay50cyc7XG5leHBvcnQgKiBhcyBDYWxsb3V0IGZyb20gJy4vQ2FsbG91dC50cyc7XG5leHBvcnQgKiBhcyBDb21wb25lbnRzIGZyb20gJy4vQ29tcG9uZW50cy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBEYXRhdmlldyBmcm9tICcuL0RhdGF2aWV3LnRzJztcbmV4cG9ydCAqIGFzIERhdGF2aWV3TGluayBmcm9tICcuL0RhdGF2aWV3TGluay50cyc7XG5leHBvcnQgKiBhcyBGaWxlQ2hhbmdlIGZyb20gJy4vRmlsZUNoYW5nZS50cyc7XG5leHBvcnQgKiBhcyBGaWxlTWFuYWdlciBmcm9tICcuL0ZpbGVNYW5hZ2VyLnRzJztcbmV4cG9ydCAqIGFzIEZpbGVTeXN0ZW0gZnJvbSAnLi9GaWxlU3lzdGVtLnRzJztcbmV4cG9ydCAqIGFzIEZyb250bWF0dGVyIGZyb20gJy4vRnJvbnRtYXR0ZXIudHMnO1xuZXhwb3J0ICogYXMgTGluayBmcm9tICcuL0xpbmsudHMnO1xuZXhwb3J0ICogYXMgTG9nZ2VyIGZyb20gJy4vTG9nZ2VyLnRzJztcbmV4cG9ydCAqIGFzIExvb3AgZnJvbSAnLi9Mb29wLnRzJztcbmV4cG9ydCAqIGFzIE1hcmtkb3duIGZyb20gJy4vTWFya2Rvd24udHMnO1xuZXhwb3J0ICogYXMgTWFya2Rvd25Db2RlQmxvY2tQcm9jZXNzb3IgZnJvbSAnLi9NYXJrZG93bkNvZGVCbG9ja1Byb2Nlc3Nvci50cyc7XG5leHBvcnQgKiBhcyBNYXJrZG93blZpZXcgZnJvbSAnLi9NYXJrZG93blZpZXcudHMnO1xuZXhwb3J0ICogYXMgTWV0YWRhdGFDYWNoZSBmcm9tICcuL01ldGFkYXRhQ2FjaGUudHMnO1xuZXhwb3J0ICogYXMgTW9kYWxzIGZyb20gJy4vTW9kYWxzL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIE1vbmtleUFyb3VuZCBmcm9tICcuL01vbmtleUFyb3VuZC50cyc7XG5leHBvcnQgKiBhcyBPYnNpZGlhblNldHRpbmdzIGZyb20gJy4vT2JzaWRpYW5TZXR0aW5ncy50cyc7XG5leHBvcnQgKiBhcyBQZGYgZnJvbSAnLi9QZGYudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luIGZyb20gJy4vUGx1Z2luL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIFF1ZXVlIGZyb20gJy4vUXVldWUudHMnO1xuZXhwb3J0ICogYXMgUmVhY3QgZnJvbSAnLi9SZWFjdC9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBSZWZlcmVuY2UgZnJvbSAnLi9SZWZlcmVuY2UudHMnO1xuZXhwb3J0ICogYXMgUmVuYW1lRGVsZXRlSGFuZGxlciBmcm9tICcuL1JlbmFtZURlbGV0ZUhhbmRsZXIudHMnO1xuZXhwb3J0ICogYXMgUmVzb3VyY2VVcmwgZnJvbSAnLi9SZXNvdXJjZVVybC50cyc7XG5leHBvcnQgKiBhcyBTZXR0aW5nRXggZnJvbSAnLi9TZXR0aW5nRXgudHMnO1xuZXhwb3J0ICogYXMgVmFsaWRhdGlvbk1lc3NhZ2UgZnJvbSAnLi9WYWxpZGF0aW9uTWVzc2FnZS50cyc7XG5leHBvcnQgKiBhcyBWYXVsdCBmcm9tICcuL1ZhdWx0LnRzJztcbmV4cG9ydCAqIGFzIFZhdWx0RXggZnJvbSAnLi9WYXVsdEV4LnRzJztcbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFFQSxZQUFZLFNBQVM7QUFDckIsWUFBWSxvQkFBb0I7QUFDaEMsWUFBWSxjQUFjO0FBQzFCLFlBQVksYUFBYTtBQUN6QixZQUFZLGdCQUFnQjtBQUM1QixZQUFZLGNBQWM7QUFDMUIsWUFBWSxrQkFBa0I7QUFDOUIsWUFBWSxnQkFBZ0I7QUFDNUIsWUFBWSxpQkFBaUI7QUFDN0IsWUFBWSxnQkFBZ0I7QUFDNUIsWUFBWSxpQkFBaUI7QUFDN0IsWUFBWSxVQUFVO0FBQ3RCLFlBQVksWUFBWTtBQUN4QixZQUFZLFVBQVU7QUFDdEIsWUFBWSxjQUFjO0FBQzFCLFlBQVksZ0NBQWdDO0FBQzVDLFlBQVksa0JBQWtCO0FBQzlCLFlBQVksbUJBQW1CO0FBQy9CLFlBQVksWUFBWTtBQUN4QixZQUFZLGtCQUFrQjtBQUM5QixZQUFZLHNCQUFzQjtBQUNsQyxZQUFZLFNBQVM7QUFDckIsWUFBWSxZQUFZO0FBQ3hCLFlBQVksV0FBVztBQUN2QixZQUFZLFdBQVc7QUFDdkIsWUFBWSxlQUFlO0FBQzNCLFlBQVkseUJBQXlCO0FBQ3JDLFlBQVksaUJBQWlCO0FBQzdCLFlBQVksZUFBZTtBQUMzQixZQUFZLHVCQUF1QjtBQUNuQyxZQUFZLFdBQVc7QUFDdkIsWUFBWSxhQUFhOyIsCiAgIm5hbWVzIjogW10KfQo=