obsidian-dev-utils 22.1.1-beta.5 → 22.1.1-beta.7
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.
- package/CHANGELOG.md +12 -0
- package/dist/dprint.json +1 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.cjs +1 -1
- package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.cjs +38 -0
- package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.d.cts +19 -0
- package/dist/lib/cjs/obsidian/Components/TypedTextComponent.cjs +1 -1
- package/dist/lib/cjs/obsidian/Components/TypedTextComponent.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Components/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/Components/index.d.cts +1 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +100 -78
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +24 -18
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +16 -12
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +8 -8
- package/dist/lib/cjs/obsidian/ValidationMessage.cjs +2 -2
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/MultipleTextComponent.d.mts +2 -1
- package/dist/lib/esm/obsidian/Components/MultipleTextComponent.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/PlaceholderComponent.d.mts +19 -0
- package/dist/lib/esm/obsidian/Components/PlaceholderComponent.mjs +14 -0
- package/dist/lib/esm/obsidian/Components/TypedTextComponent.d.mts +2 -1
- package/dist/lib/esm/obsidian/Components/TypedTextComponent.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/Components/index.mjs +3 -1
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +24 -18
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +98 -77
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +8 -8
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +16 -12
- package/dist/lib/esm/obsidian/ValidationMessage.mjs +2 -2
- package/obsidian/Components/PlaceholderComponent/package.json +6 -0
- package/package.json +1 -1
@@ -5,6 +5,7 @@ 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 { Notice } from "obsidian";
|
8
9
|
import { noop } from "../../Function.mjs";
|
9
10
|
import { DateTransformer } from "../../Transformers/DateTransformer.mjs";
|
10
11
|
import { DurationTransformer } from "../../Transformers/DurationTransformer.mjs";
|
@@ -25,11 +26,11 @@ class ProxyHandlerBase {
|
|
25
26
|
if (typeof prop !== "string") {
|
26
27
|
return record[prop];
|
27
28
|
}
|
28
|
-
const
|
29
|
-
if (!
|
29
|
+
const property = this.properties.get(prop);
|
30
|
+
if (!property) {
|
30
31
|
return record[prop];
|
31
32
|
}
|
32
|
-
return this.getPropertyValue(
|
33
|
+
return this.getPropertyValue(property);
|
33
34
|
}
|
34
35
|
}
|
35
36
|
class EditableSettingsProxyHandler extends ProxyHandlerBase {
|
@@ -40,76 +41,34 @@ class EditableSettingsProxyHandler extends ProxyHandlerBase {
|
|
40
41
|
record[prop] = value;
|
41
42
|
return true;
|
42
43
|
}
|
43
|
-
const
|
44
|
-
if (!
|
44
|
+
const property = this.properties.get(prop);
|
45
|
+
if (!property) {
|
45
46
|
record[prop] = value;
|
46
47
|
return true;
|
47
48
|
}
|
48
|
-
|
49
|
-
this.validationPromise = this.validationPromise.then(() =>
|
49
|
+
property.set(value);
|
50
|
+
this.validationPromise = this.validationPromise.then(() => property.setAndValidate(value));
|
50
51
|
return true;
|
51
52
|
}
|
52
|
-
getPropertyValue(
|
53
|
-
return
|
54
|
-
}
|
55
|
-
}
|
56
|
-
class PluginSettingsProperty {
|
57
|
-
constructor(defaultValue, validator) {
|
58
|
-
this.defaultValue = defaultValue;
|
59
|
-
this.validator = validator;
|
60
|
-
}
|
61
|
-
get validationMessage() {
|
62
|
-
return this._validationMessage;
|
63
|
-
}
|
64
|
-
_validationMessage = "";
|
65
|
-
savedValue;
|
66
|
-
value;
|
67
|
-
clear() {
|
68
|
-
this.value = void 0;
|
69
|
-
this._validationMessage = "";
|
70
|
-
}
|
71
|
-
get() {
|
72
|
-
return this.value ?? this.defaultValue;
|
73
|
-
}
|
74
|
-
getSafe() {
|
75
|
-
return this._validationMessage ? this.defaultValue : this.get();
|
76
|
-
}
|
77
|
-
getSaved() {
|
78
|
-
return this.savedValue;
|
79
|
-
}
|
80
|
-
save() {
|
81
|
-
this.savedValue = this.value;
|
82
|
-
}
|
83
|
-
set(value) {
|
84
|
-
if (isValidationMessageHolder(value)) {
|
85
|
-
this._validationMessage = value.validationMessage;
|
86
|
-
} else {
|
87
|
-
this.value = value;
|
88
|
-
}
|
89
|
-
}
|
90
|
-
async setAndValidate(value) {
|
91
|
-
this.set(value);
|
92
|
-
if (this.value === void 0) {
|
93
|
-
return;
|
94
|
-
}
|
95
|
-
this._validationMessage = await this.validator(this.value) ?? "";
|
53
|
+
getPropertyValue(property) {
|
54
|
+
return property.get();
|
96
55
|
}
|
97
56
|
}
|
98
57
|
class PropertiesMap extends Map {
|
99
|
-
getTyped(
|
100
|
-
const property = super.get(
|
58
|
+
getTyped(propertyName) {
|
59
|
+
const property = super.get(propertyName);
|
101
60
|
if (!property) {
|
102
|
-
throw new Error(`Property ${String(
|
61
|
+
throw new Error(`Property ${String(propertyName)} not found`);
|
103
62
|
}
|
104
63
|
return property;
|
105
64
|
}
|
106
|
-
setTyped(
|
107
|
-
return super.set(
|
65
|
+
setTyped(propertyName, value) {
|
66
|
+
return super.set(propertyName, value);
|
108
67
|
}
|
109
68
|
}
|
110
69
|
class SafeSettingsProxyHandler extends ProxyHandlerBase {
|
111
|
-
getPropertyValue(
|
112
|
-
return
|
70
|
+
getPropertyValue(property) {
|
71
|
+
return property.getSafe();
|
113
72
|
}
|
114
73
|
}
|
115
74
|
class PluginSettingsManagerBase {
|
@@ -118,8 +77,11 @@ class PluginSettingsManagerBase {
|
|
118
77
|
this.defaultSettings = this.createDefaultSettings();
|
119
78
|
this.addValidators();
|
120
79
|
this.properties = new PropertiesMap();
|
121
|
-
for (const
|
122
|
-
this.properties.set(
|
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
|
+
);
|
123
85
|
}
|
124
86
|
this.validators.clear();
|
125
87
|
this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
|
@@ -135,8 +97,8 @@ class PluginSettingsManagerBase {
|
|
135
97
|
await editableSettings.validationPromise;
|
136
98
|
await this.saveToFile();
|
137
99
|
}
|
138
|
-
getProperty(
|
139
|
-
return this.properties.getTyped(
|
100
|
+
getProperty(propertyName) {
|
101
|
+
return this.properties.getTyped(propertyName);
|
140
102
|
}
|
141
103
|
async loadFromFile() {
|
142
104
|
for (const property of this.properties.values()) {
|
@@ -156,17 +118,21 @@ class PluginSettingsManagerBase {
|
|
156
118
|
const beforePrepareJson = JSON.stringify(record);
|
157
119
|
await this.prepareRecord(record);
|
158
120
|
const afterPrepareJson = JSON.stringify(record);
|
159
|
-
for (const [
|
160
|
-
const
|
161
|
-
if (!
|
162
|
-
console.warn(`Unknown property: ${
|
121
|
+
for (const [propertyName, value] of Object.entries(record)) {
|
122
|
+
const property = this.properties.get(propertyName);
|
123
|
+
if (!property) {
|
124
|
+
console.warn(`Unknown property: ${propertyName}`);
|
163
125
|
continue;
|
164
126
|
}
|
165
|
-
if (typeof value !== typeof
|
166
|
-
console.warn(
|
127
|
+
if (typeof value !== typeof property.defaultValue) {
|
128
|
+
console.warn("Invalid value type", {
|
129
|
+
propertyName,
|
130
|
+
propertyType: typeof property.defaultValue,
|
131
|
+
value
|
132
|
+
});
|
167
133
|
continue;
|
168
134
|
}
|
169
|
-
await
|
135
|
+
await property.setAndValidate(value);
|
170
136
|
}
|
171
137
|
if (afterPrepareJson !== beforePrepareJson) {
|
172
138
|
await this.saveToFile();
|
@@ -186,8 +152,8 @@ class PluginSettingsManagerBase {
|
|
186
152
|
await this.plugin.saveData(record);
|
187
153
|
await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
|
188
154
|
}
|
189
|
-
addValidator(
|
190
|
-
this.validators.set(
|
155
|
+
addValidator(propertyName, validator) {
|
156
|
+
this.validators.set(propertyName, validator);
|
191
157
|
}
|
192
158
|
addValidators() {
|
193
159
|
noop();
|
@@ -200,8 +166,8 @@ class PluginSettingsManagerBase {
|
|
200
166
|
}
|
201
167
|
getSavedSettings() {
|
202
168
|
const savedSettings = {};
|
203
|
-
for (const [
|
204
|
-
savedSettings[
|
169
|
+
for (const [propertyName, property] of this.properties.entries()) {
|
170
|
+
savedSettings[propertyName] = property.getSaved();
|
205
171
|
}
|
206
172
|
const proto = Object.getPrototypeOf(this.defaultSettings);
|
207
173
|
Object.setPrototypeOf(savedSettings, proto);
|
@@ -209,13 +175,68 @@ class PluginSettingsManagerBase {
|
|
209
175
|
}
|
210
176
|
getSettingsRecord() {
|
211
177
|
const settings = {};
|
212
|
-
for (const [
|
213
|
-
settings[
|
178
|
+
for (const [propertyName, property] of this.properties.entries()) {
|
179
|
+
settings[propertyName] = property.get();
|
214
180
|
}
|
215
181
|
return settings;
|
216
182
|
}
|
217
183
|
}
|
184
|
+
class PluginSettingsProperty {
|
185
|
+
constructor(propertyName, defaultValue, validator) {
|
186
|
+
this.propertyName = propertyName;
|
187
|
+
this.defaultValue = defaultValue;
|
188
|
+
this.validator = validator;
|
189
|
+
}
|
190
|
+
get validationMessage() {
|
191
|
+
return this._validationMessage;
|
192
|
+
}
|
193
|
+
_validationMessage = "";
|
194
|
+
savedValue;
|
195
|
+
value;
|
196
|
+
clear() {
|
197
|
+
this.value = void 0;
|
198
|
+
this._validationMessage = "";
|
199
|
+
}
|
200
|
+
get() {
|
201
|
+
return this.value ?? this.defaultValue;
|
202
|
+
}
|
203
|
+
getSafe() {
|
204
|
+
return this._validationMessage ? this.defaultValue : this.get();
|
205
|
+
}
|
206
|
+
getSaved() {
|
207
|
+
return this.savedValue;
|
208
|
+
}
|
209
|
+
save() {
|
210
|
+
this.savedValue = this.value;
|
211
|
+
}
|
212
|
+
set(value) {
|
213
|
+
if (isValidationMessageHolder(value)) {
|
214
|
+
this._validationMessage = value.validationMessage;
|
215
|
+
} else {
|
216
|
+
this.value = value;
|
217
|
+
}
|
218
|
+
}
|
219
|
+
async setAndValidate(value) {
|
220
|
+
this.set(value);
|
221
|
+
if (this.value === void 0) {
|
222
|
+
return;
|
223
|
+
}
|
224
|
+
this._validationMessage = await this.validator(this.value) ?? "";
|
225
|
+
if (!this._validationMessage) {
|
226
|
+
return;
|
227
|
+
}
|
228
|
+
const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;
|
229
|
+
new Notice(warningMessage);
|
230
|
+
console.warn(warningMessage, {
|
231
|
+
defaultValue: this.defaultValue,
|
232
|
+
propertyName: this.propertyName,
|
233
|
+
validationMessage: this._validationMessage,
|
234
|
+
value
|
235
|
+
});
|
236
|
+
}
|
237
|
+
}
|
218
238
|
export {
|
219
|
-
PluginSettingsManagerBase
|
239
|
+
PluginSettingsManagerBase,
|
240
|
+
PluginSettingsProperty
|
220
241
|
};
|
221
|
-
//# 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 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 propertyObj = this.properties.get(prop);\n    if (!propertyObj) {\n      return record[prop];\n    }\n\n    return this.getPropertyValue(propertyObj);\n  }\n\n  protected abstract getPropertyValue(propertyObj: 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 propertyObj = this.properties.get(prop);\n    if (!propertyObj) {\n      record[prop] = value;\n      return true;\n    }\n\n    propertyObj.set(value);\n    this.validationPromise = this.validationPromise.then(() => propertyObj.setAndValidate(value));\n\n    return true;\n  }\n\n  protected override getPropertyValue(propertyObj: PluginSettingsProperty<unknown>): unknown {\n    return propertyObj.get();\n  }\n}\n\nclass 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(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}\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\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(propertyObj: PluginSettingsProperty<unknown>): unknown {\n    return propertyObj.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 key of Object.keys(this.defaultSettings) as StringKeys<PluginSettings>[]) {\n      this.properties.set(key, new PluginSettingsProperty(this.defaultSettings[key], this.validators.get(key) ?? noop));\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<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    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 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.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<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 getSavedSettings(): Partial<PluginSettings> {\n    const savedSettings: Partial<PluginSettings> = {};\n    for (const [key, property] of this.properties.entries()) {\n      savedSettings[key 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 [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": []
}

|
242
|
+
//# 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 [propertyName, value] of Object.entries(record)) {\n      const property = this.properties.get(propertyName);\n      if (!property) {\n        console.warn(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof property.defaultValue) {\n        console.warn('Invalid value type', {\n          propertyName,\n          propertyType: typeof property.defaultValue,\n          value\n        });\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": ";;;;;;;AAKA,SAAS,cAAc;AAUvB,SAAS,YAAY;AACrB,SAAS,uBAAuB;AAChC,SAAS,2BAA2B;AACpC,SAAS,wBAAwB;AACjC,SAAS,sCAAsC;AAC/C,SAAS,iCAAiC;AAE1C,MAAM,qBAAqB,IAAI,iBAAiB;AAAA,EAC9C,IAAI,+BAA+B;AAAA,EACnC,IAAI,gBAAgB;AAAA,EACpB,IAAI,oBAAoB;AAC1B,CAAC;AAID,MAAe,iBAAwF;AAAA,EAC9F,YAA+B,YAA2C;AAA3C;AAAA,EAA4C;AAAA,EAE3E,IAAI,QAAwB,MAAgC;AACjE,UAAM,SAAS;AACf,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO,OAAO,IAAI;AAAA,IACpB;AAEA,UAAM,WAAW,KAAK,WAAW,IAAI,IAAI;AACzC,QAAI,CAAC,UAAU;AACb,aAAO,OAAO,IAAI;AAAA,IACpB;AAEA,WAAO,KAAK,iBAAiB,QAAQ;AAAA,EACvC;AAGF;AAEA,MAAM,qCAAoE,iBAAiC;AAAA,EACjG,oBAAoB,QAAQ,QAAQ;AAAA,EACrC,IAAI,QAAwB,MAAuB,OAAyB;AACjF,UAAM,SAAS;AAEf,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO,IAAI,IAAI;AACf,aAAO;AAAA,IACT;AAEA,UAAM,WAAW,KAAK,WAAW,IAAI,IAAI;AACzC,QAAI,CAAC,UAAU;AACb,aAAO,IAAI,IAAI;AACf,aAAO;AAAA,IACT;AAEA,aAAS,IAAI,KAAK;AAClB,SAAK,oBAAoB,KAAK,kBAAkB,KAAK,MAAM,SAAS,eAAe,KAAK,CAAC;AAEzF,WAAO;AAAA,EACT;AAAA,EAEmB,iBAAiB,UAAoD;AACtF,WAAO,SAAS,IAAI;AAAA,EACtB;AACF;AAGA,MAAM,sBAAqD,IAAyC;AAAA,EAC3F,SAA0D,cAAkF;AACjJ,UAAM,WAAW,MAAM,IAAI,YAAY;AACvC,QAAI,CAAC,UAAU;AACb,YAAM,IAAI,MAAM,YAAY,OAAO,YAAY,CAAC,YAAY;AAAA,IAC9D;AAEA,WAAO;AAAA,EACT;AAAA,EAEO,SACL,cACA,OACM;AACN,WAAO,MAAM,IAAI,cAAc,KAAK;AAAA,EACtC;AACF;AAEA,MAAM,iCAAgE,iBAAiC;AAAA,EAClF,iBAAiB,UAAoD;AACtF,WAAO,SAAS,QAAQ;AAAA,EAC1B;AACF;AAOO,MAAe,0BAAyD;AAAA,EAQtE,YAAoB,QAAoC;AAApC;AACzB,SAAK,kBAAkB,KAAK,sBAAsB;AAElD,SAAK,cAAc;AAEnB,SAAK,aAAa,IAAI,cAA8B;AAEpD,eAAW,gBAAgB,OAAO,KAAK,KAAK,eAAe,GAAmC;AAC5F,WAAK,WAAW;AAAA,QACd;AAAA,QACA,IAAI,uBAAuB,cAAc,KAAK,gBAAgB,YAAY,GAAG,KAAK,WAAW,IAAI,YAAY,KAAK,IAAI;AAAA,MACxH;AAAA,IACF;AAEA,SAAK,WAAW,MAAM;AAEtB,SAAK,eAAe,IAAI,MAAM,KAAK,iBAAiB,IAAI,yBAAyC,KAAK,UAAU,CAAC;AAAA,EACnH;AAAA,EAxBgB;AAAA,EAER;AAAA,EACA;AAAA;AAAA,EAEA,aAA0C,oBAAI,IAA4B;AAAA,EAqBlF,MAAa,YAAY,QAAuE;AAC9F,UAAM,mBAAmB,IAAI,MAAM,KAAK,iBAAiB,IAAI,6BAA6C,KAAK,UAAU,CAAC;AAG1H,UAAM,OAAO,gBAAgB;AAC7B,UAAM,iBAAiB;AACvB,UAAM,KAAK,WAAW;AAAA,EACxB;AAAA,EAEO,YAA6D,cAAkF;AACpJ,WAAO,KAAK,WAAW,SAAS,YAAY;AAAA,EAC9C;AAAA,EAEA,MAAa,eAA8B;AACzC,eAAW,YAAY,KAAK,WAAW,OAAO,GAAG;AAC/C,eAAS,MAAM;AAAA,IACjB;AAEA,UAAM,OAAO,MAAM,KAAK,OAAO,SAAS;AAExC,QAAI,SAAS,UAAa,SAAS,MAAM;AACvC;AAAA,IACF;AAEA,QAAI,OAAO,SAAS,YAAY,MAAM,QAAQ,IAAI,GAAG;AACnD,YAAM,OAAO,MAAM,QAAQ,IAAI,IAAI,UAAU,OAAO;AACpD,cAAQ,MAAM,4CAA4C,IAAI,EAAE;AAChE;AAAA,IACF;AAEA,QAAI,SAAS;AACb,aAAS,KAAK,eAAe,EAAE,2BAA2B,MAAM;AAChE,UAAM,oBAAoB,KAAK,UAAU,MAAM;AAC/C,UAAM,KAAK,cAAc,MAAM;AAC/B,UAAM,mBAAmB,KAAK,UAAU,MAAM;AAE9C,eAAW,CAAC,cAAc,KAAK,KAAK,OAAO,QAAQ,MAAM,GAAG;AAC1D,YAAM,WAAW,KAAK,WAAW,IAAI,YAAY;AACjD,UAAI,CAAC,UAAU;AACb,gBAAQ,KAAK,qBAAqB,YAAY,EAAE;AAChD;AAAA,MACF;AAEA,UAAI,OAAO,UAAU,OAAO,SAAS,cAAc;AACjD,gBAAQ,KAAK,sBAAsB;AAAA,UACjC;AAAA,UACA,cAAc,OAAO,SAAS;AAAA,UAC9B;AAAA,QACF,CAAC;AACD;AAAA,MACF;AAEA,YAAM,SAAS,eAAe,KAAK;AAAA,IACrC;AAEA,QAAI,qBAAqB,mBAAmB;AAC1C,YAAM,KAAK,WAAW;AAAA,IACxB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAa,aAA4B;AACvC,UAAM,cAAc,KAAK,iBAAiB;AAE1C,eAAW,YAAY,KAAK,WAAW,OAAO,GAAG;AAC/C,eAAS,KAAK;AAAA,IAChB;AAEA,UAAM,SAAS,KAAK,eAAe,EAAE,2BAA2B,KAAK,kBAAkB,CAAC;AACxF,UAAM,KAAK,OAAO,SAAS,MAAM;AAEjC,UAAM,KAAK,OAAO,eAAe,KAAK,iBAAiB,GAAG,WAAW;AAAA,EACvE;AAAA,EAEU,aACR,cACA,WACM;AACN,SAAK,WAAW,IAAI,cAAc,SAAS;AAAA,EAC7C;AAAA,EAEU,gBAAsB;AAC9B,SAAK;AAAA,EACP;AAAA,EAIU,iBAA8B;AACtC,WAAO;AAAA,EACT;AAAA,EAEU,cAAc,SAAoD;AAC1E,SAAK;AAAA,EACP;AAAA,EAEQ,mBAA4C;AAClD,UAAM,gBAAyC,CAAC;AAChD,eAAW,CAAC,cAAc,QAAQ,KAAK,KAAK,WAAW,QAAQ,GAAG;AAChE,oBAAc,YAA0C,IAAI,SAAS,SAAS;AAAA,IAChF;AACA,UAAM,QAAQ,OAAO,eAAe,KAAK,eAAe;AACxD,WAAO,eAAe,eAAe,KAAK;AAC1C,WAAO;AAAA,EACT;AAAA,EAEQ,oBAAiE;AACvE,UAAM,WAAwD,CAAC;AAC/D,eAAW,CAAC,cAAc,QAAQ,KAAK,KAAK,WAAW,QAAQ,GAAG;AAChE,eAAS,YAA0C,IAAI,SAAS,IAAI;AAAA,IACtE;AAEA,WAAO;AAAA,EACT;AACF;AAOO,MAAM,uBAA0B;AAAA,EAU9B,YAA6B,cAAsC,cAAkC,WAAyB;AAAjG;AAAsC;AAAkC;AAAA,EAA0B;AAAA,EATtI,IAAW,oBAA4B;AACrC,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ,qBAAqB;AAAA,EAErB;AAAA,EAEA;AAAA,EAGD,QAAc;AACnB,SAAK,QAAQ;AACb,SAAK,qBAAqB;AAAA,EAC5B;AAAA,EAEO,MAAS;AACd,WAAO,KAAK,SAAS,KAAK;AAAA,EAC5B;AAAA,EAEO,UAAa;AAClB,WAAO,KAAK,qBAAqB,KAAK,eAAe,KAAK,IAAI;AAAA,EAChE;AAAA,EAEO,WAA0B;AAC/B,WAAO,KAAK;AAAA,EACd;AAAA,EAEO,OAAa;AAClB,SAAK,aAAa,KAAK;AAAA,EACzB;AAAA,EAEO,IAAI,OAAsD;AAC/D,QAAI,0BAA0B,KAAK,GAAG;AACpC,WAAK,qBAAqB,MAAM;AAAA,IAClC,OAAO;AACL,WAAK,QAAQ;AAAA,IACf;AAAA,EACF;AAAA,EAEA,MAAa,eAAe,OAA+D;AACzF,SAAK,IAAI,KAAK;AACd,QAAI,KAAK,UAAU,QAAW;AAC5B;AAAA,IACF;AAEA,SAAK,qBAAsB,MAAM,KAAK,UAAU,KAAK,KAAK,KAA4B;AAEtF,QAAI,CAAC,KAAK,oBAAoB;AAC5B;AAAA,IACF;AAEA,UAAM,iBAAiB,iCAAiC,KAAK,YAAY;AACzE,QAAI,OAAO,cAAc;AACzB,YAAQ,KAAK,gBAAgB;AAAA,MAC3B,cAAc,KAAK;AAAA,MACnB,cAAc,KAAK;AAAA,MACnB,mBAAmB,KAAK;AAAA,MACxB;AAAA,IACF,CAAC;AAAA,EACH;AACF;",
  "names": []
}

|
@@ -25,19 +25,19 @@ export interface BindOptions<T> {
|
|
25
25
|
/**
|
26
26
|
* Extended options for binding a value component to a plugin setting.
|
27
27
|
*/
|
28
|
-
export interface BindOptionsExtended<PluginSettings extends object, UIValue,
|
28
|
+
export interface BindOptionsExtended<PluginSettings extends object, UIValue, PropertyName extends StringKeys<PluginSettings>> extends BindOptions<PluginSettings[PropertyName]> {
|
29
29
|
/**
|
30
30
|
* Converts the UI component's value back to the plugin settings value.
|
31
31
|
* @param uiValue - The value of the UI component.
|
32
32
|
* @returns The value to set on the plugin settings.
|
33
33
|
*/
|
34
|
-
componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[
|
34
|
+
componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;
|
35
35
|
/**
|
36
36
|
* Converts the plugin settings value to the value used by the UI component.
|
37
37
|
* @param pluginSettingsValue - The value of the property in the plugin settings.
|
38
38
|
* @returns The value to set on the UI component.
|
39
39
|
*/
|
40
|
-
pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[
|
40
|
+
pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[PropertyName]) => UIValue;
|
41
41
|
}
|
42
42
|
type ExtractPluginSettings<Plugin extends PluginBase<any>> = Plugin extends PluginBase<infer PluginSettings> ? PluginSettings : never;
|
43
43
|
/**
|
@@ -55,23 +55,23 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
|
|
55
55
|
* @typeParam UIValue - The type of the value of the UI component.
|
56
56
|
* @typeParam TValueComponent - The type of the value component.
|
57
57
|
* @param valueComponent - The value component to bind.
|
58
|
-
* @param
|
58
|
+
* @param propertyName - The property of the plugin settings to bind to.
|
59
59
|
* @param options - The options for binding the value component.
|
60
60
|
* @returns The value component.
|
61
61
|
*/
|
62
|
-
bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,
|
62
|
+
bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions<UIValue>): TValueComponent;
|
63
63
|
/**
|
64
64
|
* Binds a value component to a plugin setting.
|
65
65
|
*
|
66
66
|
* @typeParam UIValue - The type of the value of the UI component.
|
67
67
|
* @typeParam TValueComponent - The type of the value component.
|
68
|
-
* @typeParam
|
68
|
+
* @typeParam PropertyName - The property name of the plugin settings to bind to.
|
69
69
|
* @param valueComponent - The value component to bind.
|
70
|
-
* @param
|
70
|
+
* @param propertyName - The property name of the plugin settings to bind to.
|
71
71
|
* @param options - The options for binding the value component.
|
72
72
|
* @returns The value component.
|
73
73
|
*/
|
74
|
-
bind<UIValue, TValueComponent,
|
74
|
+
bind<UIValue, TValueComponent, PropertyName extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: PropertyName, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, PropertyName>): TValueComponent;
|
75
75
|
hide(): void;
|
76
76
|
}
|
77
77
|
export {};
|
@@ -9,6 +9,7 @@ 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";
|
13
14
|
import { isValidationMessageHolder } from "../ValidationMessage.mjs";
|
14
15
|
import { getPluginId } from "./PluginId.mjs";
|
@@ -23,13 +24,13 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
23
24
|
*
|
24
25
|
* @typeParam UIValue - The type of the value of the UI component.
|
25
26
|
* @typeParam TValueComponent - The type of the value component.
|
26
|
-
* @typeParam
|
27
|
+
* @typeParam PropertyName - The property name of the plugin settings to bind to.
|
27
28
|
* @param valueComponent - The value component to bind.
|
28
|
-
* @param
|
29
|
+
* @param propertyName - The property name of the plugin settings to bind to.
|
29
30
|
* @param options - The options for binding the value component.
|
30
31
|
* @returns The value component.
|
31
32
|
*/
|
32
|
-
bind(valueComponent,
|
33
|
+
bind(valueComponent, propertyName, options) {
|
33
34
|
const DEFAULT_OPTIONS = {
|
34
35
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
35
36
|
componentToPluginSettingsValueConverter: (value) => value,
|
@@ -39,14 +40,17 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
39
40
|
};
|
40
41
|
const optionsExt = { ...DEFAULT_OPTIONS, ...options };
|
41
42
|
const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;
|
42
|
-
const
|
43
|
-
valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(
|
44
|
-
const oldValue =
|
43
|
+
const property = this.plugin.settingsManager.getProperty(propertyName);
|
44
|
+
valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(property.get())).onChange(async (uiValue) => {
|
45
|
+
const oldValue = property.get();
|
45
46
|
const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);
|
46
|
-
await
|
47
|
+
await property.setAndValidate(convertedValue);
|
47
48
|
const newValue = isValidationMessageHolder(convertedValue) ? void 0 : convertedValue;
|
48
49
|
await optionsExt.onChanged(newValue, oldValue);
|
49
50
|
});
|
51
|
+
if (isPlaceholderComponent(valueComponent)) {
|
52
|
+
valueComponent.setPlaceholder(optionsExt.pluginSettingsToComponentValueConverter(property.defaultValue));
|
53
|
+
}
|
50
54
|
validatorElement?.addEventListener("focus", validate);
|
51
55
|
validatorElement?.addEventListener("blur", validate);
|
52
56
|
validate();
|
@@ -55,13 +59,13 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
55
59
|
if (!validatorElement) {
|
56
60
|
return;
|
57
61
|
}
|
58
|
-
if (!
|
62
|
+
if (!property.validationMessage) {
|
59
63
|
validatorElement.setCustomValidity("");
|
60
64
|
validatorElement.checkValidity();
|
61
|
-
|
65
|
+
property.set(validatorElement);
|
62
66
|
}
|
63
|
-
validatorElement.setCustomValidity(
|
64
|
-
validatorElement.title =
|
67
|
+
validatorElement.setCustomValidity(property.validationMessage);
|
68
|
+
validatorElement.title = property.validationMessage;
|
65
69
|
if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {
|
66
70
|
validatorElement.reportValidity();
|
67
71
|
}
|
@@ -75,4 +79,4 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
75
79
|
export {
|
76
80
|
PluginSettingsTabBase
|
77
81
|
};
|
78
|
-
//# 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';\n\nimport { invokeAsyncSafely } from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.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  Property extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[Property]> {\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<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 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<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 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<ExtractPluginSettings<TPlugin>[Property]>\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 oldValue = propertyObj.get() as PropertyType;\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        await propertyObj.setAndValidate(convertedValue);\n        const newValue = isValidationMessageHolder(convertedValue) ? undefined : convertedValue;\n        await optionsExt.onChanged(newValue, oldValue);\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.set(validatorElement);\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"],
  "mappings": ";;;;;;;AAWA,SAAS,wBAAwB;AAOjC,SAAS,yBAAyB;AAClC,SAAS,gBAAgB;AACzB,SAAS,YAAY;AACrB,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,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,WAAW,YAAY,IAAI;AACjC,YAAM,iBAAiB,WAAW,wCAAwC,OAAO;AACjF,YAAM,YAAY,eAAe,cAAc;AAC/C,YAAM,WAAW,0BAA0B,cAAc,IAAI,SAAY;AACzE,YAAM,WAAW,UAAU,UAAU,QAAQ;AAAA,IAC/C,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,IAAI,gBAAgB;AAAA,MAClC;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;",
  "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": []
}

|
@@ -6,9 +6,9 @@ if you want to view the source, please visit the github repository of this plugi
|
|
6
6
|
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
|
7
7
|
|
8
8
|
function isValidationMessageHolder(value) {
|
9
|
-
return
|
9
|
+
return value.validationMessage !== void 0;
|
10
10
|
}
|
11
11
|
export {
|
12
12
|
isValidationMessageHolder
|
13
13
|
};
|
14
|
-
//# sourceMappingURL=data:application/json;base64,
|
14
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1ZhbGlkYXRpb25NZXNzYWdlLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEhvbGRzIGEgdmFsaWRhdGlvbiBtZXNzYWdlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgLyoqXG4gICAqIFRoZSB2YWxpZGF0aW9uIG1lc3NhZ2UuXG4gICAqL1xuICB2YWxpZGF0aW9uTWVzc2FnZTogc3RyaW5nO1xufVxuXG4vKipcbiAqIFR5cGUgZ3VhcmQgdG8gY2hlY2sgaWYgYSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIuXG4gKlxuICogQHBhcmFtIHZhbHVlIC0gVGhlIHZhbHVlIHRvIGNoZWNrLlxuICogQHJldHVybnMgYHRydWVgIGlmIHRoZSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIsIGBmYWxzZWAgb3RoZXJ3aXNlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNWYWxpZGF0aW9uTWVzc2FnZUhvbGRlcih2YWx1ZTogdW5rbm93bik6IHZhbHVlIGlzIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgcmV0dXJuICh2YWx1ZSBhcyBQYXJ0aWFsPFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyPikudmFsaWRhdGlvbk1lc3NhZ2UgIT09IHVuZGVmaW5lZDtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFnQk8sU0FBUywwQkFBMEIsT0FBa0Q7QUFDMUYsU0FBUSxNQUEyQyxzQkFBc0I7QUFDM0U7IiwKICAibmFtZXMiOiBbXQp9Cg==
|
package/package.json
CHANGED