obsidian-dev-utils 22.1.1-beta.4 → 22.1.1-beta.5
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 +13 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/Object.cjs +1 -1
- package/dist/lib/cjs/Object.d.cts +6 -0
- package/dist/lib/cjs/ScriptUtils/CliUtils.cjs +1 -2
- package/dist/lib/cjs/ScriptUtils/CliUtils.d.cts +3 -2
- package/dist/lib/cjs/ScriptUtils/cli.cjs +1 -1
- package/dist/lib/cjs/String.cjs +1 -1
- package/dist/lib/cjs/String.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Callout.cjs +1 -1
- package/dist/lib/cjs/obsidian/Callout.d.cts +2 -1
- package/dist/lib/cjs/obsidian/FileManager.cjs +1 -1
- package/dist/lib/cjs/obsidian/FileManager.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Link.cjs +1 -1
- package/dist/lib/cjs/obsidian/Link.d.cts +4 -3
- package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
- package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +11 -1
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +8 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +102 -24
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +16 -7
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +7 -12
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +7 -9
- package/dist/lib/cjs/obsidian/ValidationMessage.cjs +38 -0
- package/dist/lib/cjs/obsidian/ValidationMessage.d.cts +16 -0
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/Object.d.mts +6 -0
- package/dist/lib/esm/Object.mjs +1 -1
- package/dist/lib/esm/ScriptUtils/CliUtils.d.mts +3 -2
- package/dist/lib/esm/ScriptUtils/CliUtils.mjs +1 -2
- package/dist/lib/esm/ScriptUtils/cli.mjs +1 -1
- package/dist/lib/esm/String.d.mts +2 -1
- package/dist/lib/esm/String.mjs +1 -1
- package/dist/lib/esm/obsidian/Callout.d.mts +2 -1
- package/dist/lib/esm/obsidian/Callout.mjs +1 -1
- package/dist/lib/esm/obsidian/FileManager.d.mts +2 -1
- package/dist/lib/esm/obsidian/FileManager.mjs +1 -1
- package/dist/lib/esm/obsidian/Link.d.mts +4 -3
- package/dist/lib/esm/obsidian/Link.mjs +1 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +2 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
- package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +8 -0
- package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +15 -2
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +16 -7
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +102 -24
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +7 -9
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +7 -12
- package/dist/lib/esm/obsidian/ValidationMessage.d.mts +16 -0
- package/dist/lib/esm/obsidian/ValidationMessage.mjs +14 -0
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/obsidian/ValidationMessage/package.json +6 -0
- package/package.json +1 -1
@@ -28,39 +28,94 @@ __export(PluginSettingsManagerBase_exports, {
|
|
28
28
|
PluginSettingsManagerBase: () => PluginSettingsManagerBase
|
29
29
|
});
|
30
30
|
module.exports = __toCommonJS(PluginSettingsManagerBase_exports);
|
31
|
-
var import_obsidian = require('obsidian');
|
32
31
|
var import_Function = require('../../Function.cjs');
|
33
32
|
var import_DateTransformer = require('../../Transformers/DateTransformer.cjs');
|
34
33
|
var import_DurationTransformer = require('../../Transformers/DurationTransformer.cjs');
|
35
34
|
var import_GroupTransformer = require('../../Transformers/GroupTransformer.cjs');
|
36
35
|
var import_SkipPrivatePropertyTransformer = require('../../Transformers/SkipPrivatePropertyTransformer.cjs');
|
36
|
+
var import_ValidationMessage = require('../ValidationMessage.cjs');
|
37
37
|
const defaultTransformer = new import_GroupTransformer.GroupTransformer([
|
38
38
|
new import_SkipPrivatePropertyTransformer.SkipPrivatePropertyTransformer(),
|
39
39
|
new import_DateTransformer.DateTransformer(),
|
40
40
|
new import_DurationTransformer.DurationTransformer()
|
41
41
|
]);
|
42
|
+
class ProxyHandlerBase {
|
43
|
+
constructor(properties) {
|
44
|
+
this.properties = properties;
|
45
|
+
}
|
46
|
+
get(target, prop) {
|
47
|
+
const record = target;
|
48
|
+
if (typeof prop !== "string") {
|
49
|
+
return record[prop];
|
50
|
+
}
|
51
|
+
const propertyObj = this.properties.get(prop);
|
52
|
+
if (!propertyObj) {
|
53
|
+
return record[prop];
|
54
|
+
}
|
55
|
+
return this.getPropertyValue(propertyObj);
|
56
|
+
}
|
57
|
+
}
|
58
|
+
class EditableSettingsProxyHandler extends ProxyHandlerBase {
|
59
|
+
validationPromise = Promise.resolve();
|
60
|
+
set(target, prop, value) {
|
61
|
+
const record = target;
|
62
|
+
if (typeof prop !== "string") {
|
63
|
+
record[prop] = value;
|
64
|
+
return true;
|
65
|
+
}
|
66
|
+
const propertyObj = this.properties.get(prop);
|
67
|
+
if (!propertyObj) {
|
68
|
+
record[prop] = value;
|
69
|
+
return true;
|
70
|
+
}
|
71
|
+
propertyObj.set(value);
|
72
|
+
this.validationPromise = this.validationPromise.then(() => propertyObj.setAndValidate(value));
|
73
|
+
return true;
|
74
|
+
}
|
75
|
+
getPropertyValue(propertyObj) {
|
76
|
+
return propertyObj.get();
|
77
|
+
}
|
78
|
+
}
|
42
79
|
class PluginSettingsProperty {
|
43
80
|
constructor(defaultValue, validator) {
|
44
81
|
this.defaultValue = defaultValue;
|
45
82
|
this.validator = validator;
|
46
83
|
}
|
47
|
-
validationMessage
|
84
|
+
get validationMessage() {
|
85
|
+
return this._validationMessage;
|
86
|
+
}
|
87
|
+
_validationMessage = "";
|
88
|
+
savedValue;
|
48
89
|
value;
|
49
90
|
clear() {
|
50
91
|
this.value = void 0;
|
51
|
-
this.
|
92
|
+
this._validationMessage = "";
|
52
93
|
}
|
53
94
|
get() {
|
54
95
|
return this.value ?? this.defaultValue;
|
55
96
|
}
|
56
97
|
getSafe() {
|
57
|
-
return this.
|
98
|
+
return this._validationMessage ? this.defaultValue : this.get();
|
99
|
+
}
|
100
|
+
getSaved() {
|
101
|
+
return this.savedValue;
|
102
|
+
}
|
103
|
+
save() {
|
104
|
+
this.savedValue = this.value;
|
105
|
+
}
|
106
|
+
set(value) {
|
107
|
+
if ((0, import_ValidationMessage.isValidationMessageHolder)(value)) {
|
108
|
+
this._validationMessage = value.validationMessage;
|
109
|
+
} else {
|
110
|
+
this.value = value;
|
111
|
+
}
|
58
112
|
}
|
59
|
-
async
|
60
|
-
this.value
|
61
|
-
if (this.value
|
62
|
-
|
113
|
+
async setAndValidate(value) {
|
114
|
+
this.set(value);
|
115
|
+
if (this.value === void 0) {
|
116
|
+
return;
|
63
117
|
}
|
118
|
+
this._validationMessage = await this.validator(this.value) ?? "";
|
64
119
|
}
|
65
120
|
}
|
66
121
|
class PropertiesMap extends Map {
|
@@ -75,30 +130,34 @@ class PropertiesMap extends Map {
|
|
75
130
|
return super.set(key, value);
|
76
131
|
}
|
77
132
|
}
|
133
|
+
class SafeSettingsProxyHandler extends ProxyHandlerBase {
|
134
|
+
getPropertyValue(propertyObj) {
|
135
|
+
return propertyObj.getSafe();
|
136
|
+
}
|
137
|
+
}
|
78
138
|
class PluginSettingsManagerBase {
|
79
139
|
constructor(plugin) {
|
80
140
|
this.plugin = plugin;
|
81
|
-
|
141
|
+
this.defaultSettings = this.createDefaultSettings();
|
82
142
|
this.addValidators();
|
83
143
|
this.properties = new PropertiesMap();
|
84
|
-
for (const key of Object.keys(defaultSettings)) {
|
85
|
-
this.properties.set(key, new PluginSettingsProperty(defaultSettings[key], this.validators.get(key) ?? import_Function.noop));
|
144
|
+
for (const key of Object.keys(this.defaultSettings)) {
|
145
|
+
this.properties.set(key, new PluginSettingsProperty(this.defaultSettings[key], this.validators.get(key) ?? import_Function.noop));
|
86
146
|
}
|
87
147
|
this.validators.clear();
|
88
|
-
this.safeSettings = new Proxy(defaultSettings,
|
89
|
-
get: (_target, prop) => {
|
90
|
-
if (typeof prop !== "string") {
|
91
|
-
return void 0;
|
92
|
-
}
|
93
|
-
return this.properties.get(prop);
|
94
|
-
}
|
95
|
-
});
|
96
|
-
this.addValidators();
|
148
|
+
this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
|
97
149
|
}
|
98
150
|
safeSettings;
|
151
|
+
defaultSettings;
|
99
152
|
properties;
|
100
153
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
101
154
|
validators = /* @__PURE__ */ new Map();
|
155
|
+
async editAndSave(editor) {
|
156
|
+
const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
|
157
|
+
await editor(editableSettings);
|
158
|
+
await editableSettings.validationPromise;
|
159
|
+
await this.saveToFile();
|
160
|
+
}
|
102
161
|
getProperty(property) {
|
103
162
|
return this.properties.getTyped(property);
|
104
163
|
}
|
@@ -117,7 +176,9 @@ class PluginSettingsManagerBase {
|
|
117
176
|
}
|
118
177
|
let record = data;
|
119
178
|
record = this.getTransformer().transformObjectRecursively(record);
|
179
|
+
const beforePrepareJson = JSON.stringify(record);
|
120
180
|
await this.prepareRecord(record);
|
181
|
+
const afterPrepareJson = JSON.stringify(record);
|
121
182
|
for (const [key, value] of Object.entries(record)) {
|
122
183
|
const propertyObj = this.properties.get(key);
|
123
184
|
if (!propertyObj) {
|
@@ -128,7 +189,10 @@ class PluginSettingsManagerBase {
|
|
128
189
|
console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);
|
129
190
|
continue;
|
130
191
|
}
|
131
|
-
await propertyObj.
|
192
|
+
await propertyObj.setAndValidate(value);
|
193
|
+
}
|
194
|
+
if (afterPrepareJson !== beforePrepareJson) {
|
195
|
+
await this.saveToFile();
|
132
196
|
}
|
133
197
|
}
|
134
198
|
/**
|
@@ -137,8 +201,13 @@ class PluginSettingsManagerBase {
|
|
137
201
|
* @returns A promise that resolves when the settings are saved.
|
138
202
|
*/
|
139
203
|
async saveToFile() {
|
140
|
-
const
|
204
|
+
const oldSettings = this.getSavedSettings();
|
205
|
+
for (const property of this.properties.values()) {
|
206
|
+
property.save();
|
207
|
+
}
|
208
|
+
const record = this.getTransformer().transformObjectRecursively(this.getSettingsRecord());
|
141
209
|
await this.plugin.saveData(record);
|
210
|
+
await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
|
142
211
|
}
|
143
212
|
addValidator(property, validator) {
|
144
213
|
this.validators.set(property, validator);
|
@@ -152,7 +221,16 @@ class PluginSettingsManagerBase {
|
|
152
221
|
prepareRecord(_record) {
|
153
222
|
(0, import_Function.noop)();
|
154
223
|
}
|
155
|
-
|
224
|
+
getSavedSettings() {
|
225
|
+
const savedSettings = {};
|
226
|
+
for (const [key, property] of this.properties.entries()) {
|
227
|
+
savedSettings[key] = property.getSaved();
|
228
|
+
}
|
229
|
+
const proto = Object.getPrototypeOf(this.defaultSettings);
|
230
|
+
Object.setPrototypeOf(savedSettings, proto);
|
231
|
+
return savedSettings;
|
232
|
+
}
|
233
|
+
getSettingsRecord() {
|
156
234
|
const settings = {};
|
157
235
|
for (const [key, property] of this.properties.entries()) {
|
158
236
|
settings[key] = property.get();
|
@@ -164,4 +242,4 @@ class PluginSettingsManagerBase {
|
|
164
242
|
0 && (module.exports = {
|
165
243
|
PluginSettingsManagerBase
|
166
244
|
});
|
167
|
-
//# 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": []
}

|
245
|
+
//# 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": []
}

|
@@ -1,18 +1,24 @@
|
|
1
1
|
import type { Promisable, ReadonlyDeep } from 'type-fest';
|
2
|
-
import {
|
3
|
-
import type { StringKeys } from '../../Object.cjs';
|
2
|
+
import type { MaybeReturn, StringKeys } from '../../Object.cjs';
|
4
3
|
import type { Transformer } from '../../Transformers/Transformer.cjs';
|
5
|
-
type
|
4
|
+
import type { ValidationMessageHolder } from '../ValidationMessage.cjs';
|
5
|
+
import type { PluginBase } from './PluginBase.cjs';
|
6
|
+
type Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;
|
6
7
|
declare class PluginSettingsProperty<T> {
|
7
8
|
readonly defaultValue: T;
|
8
9
|
private readonly validator;
|
9
|
-
validationMessage: string;
|
10
|
+
get validationMessage(): string;
|
11
|
+
private _validationMessage;
|
12
|
+
private savedValue;
|
10
13
|
private value;
|
11
14
|
constructor(defaultValue: T, validator: Validator<T>);
|
12
15
|
clear(): void;
|
13
16
|
get(): T;
|
14
17
|
getSafe(): T;
|
15
|
-
|
18
|
+
getSaved(): T | undefined;
|
19
|
+
save(): void;
|
20
|
+
set(value: T | undefined | ValidationMessageHolder): void;
|
21
|
+
setAndValidate(value: T | undefined | ValidationMessageHolder): Promise<void>;
|
16
22
|
}
|
17
23
|
/**
|
18
24
|
* Base class for managing plugin settings.
|
@@ -22,9 +28,11 @@ declare class PluginSettingsProperty<T> {
|
|
22
28
|
export declare abstract class PluginSettingsManagerBase<PluginSettings extends object> {
|
23
29
|
private plugin;
|
24
30
|
readonly safeSettings: ReadonlyDeep<PluginSettings>;
|
31
|
+
private defaultSettings;
|
25
32
|
private properties;
|
26
33
|
private validators;
|
27
|
-
constructor(plugin:
|
34
|
+
constructor(plugin: PluginBase<PluginSettings>);
|
35
|
+
editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void>;
|
28
36
|
getProperty<Property extends StringKeys<PluginSettings>>(property: Property): PluginSettingsProperty<PluginSettings[Property]>;
|
29
37
|
loadFromFile(): Promise<void>;
|
30
38
|
/**
|
@@ -38,6 +46,7 @@ export declare abstract class PluginSettingsManagerBase<PluginSettings extends o
|
|
38
46
|
protected abstract createDefaultSettings(): PluginSettings;
|
39
47
|
protected getTransformer(): Transformer;
|
40
48
|
protected prepareRecord(_record: Record<string, unknown>): Promisable<void>;
|
41
|
-
private
|
49
|
+
private getSavedSettings;
|
50
|
+
private getSettingsRecord;
|
42
51
|
}
|
43
52
|
export {};
|
@@ -33,6 +33,7 @@ var import_Async = require('../../Async.cjs');
|
|
33
33
|
var import_CssClass = require('../../CssClass.cjs');
|
34
34
|
var import_Function = require('../../Function.cjs');
|
35
35
|
var import_ValidatorComponent = require('../Components/ValidatorComponent.cjs');
|
36
|
+
var import_ValidationMessage = require('../ValidationMessage.cjs');
|
36
37
|
var import_PluginId = require('./PluginId.cjs');
|
37
38
|
class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
38
39
|
constructor(plugin) {
|
@@ -63,14 +64,11 @@ class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
|
63
64
|
const validatorElement = (0, import_ValidatorComponent.getValidatorComponent)(valueComponent)?.validatorEl;
|
64
65
|
const propertyObj = this.plugin.settingsManager.getProperty(property);
|
65
66
|
valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(propertyObj.get())).onChange(async (uiValue) => {
|
67
|
+
const oldValue = propertyObj.get();
|
66
68
|
const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
} else {
|
71
|
-
await propertyObj.set(convertedValue);
|
72
|
-
}
|
73
|
-
await optionsExt.onChanged();
|
69
|
+
await propertyObj.setAndValidate(convertedValue);
|
70
|
+
const newValue = (0, import_ValidationMessage.isValidationMessageHolder)(convertedValue) ? void 0 : convertedValue;
|
71
|
+
await optionsExt.onChanged(newValue, oldValue);
|
74
72
|
});
|
75
73
|
validatorElement?.addEventListener("focus", validate);
|
76
74
|
validatorElement?.addEventListener("blur", validate);
|
@@ -83,7 +81,7 @@ class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
|
83
81
|
if (!propertyObj.validationMessage) {
|
84
82
|
validatorElement.setCustomValidity("");
|
85
83
|
validatorElement.checkValidity();
|
86
|
-
propertyObj.
|
84
|
+
propertyObj.set(validatorElement);
|
87
85
|
}
|
88
86
|
validatorElement.setCustomValidity(propertyObj.validationMessage);
|
89
87
|
validatorElement.title = propertyObj.validationMessage;
|
@@ -97,11 +95,8 @@ class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
|
97
95
|
(0, import_Async.invokeAsyncSafely)(() => this.plugin.settingsManager.saveToFile());
|
98
96
|
}
|
99
97
|
}
|
100
|
-
function isValidationMessageHolder(value) {
|
101
|
-
return !!value.validationMessage;
|
102
|
-
}
|
103
98
|
// Annotate the CommonJS export names for ESM import in node:
|
104
99
|
0 && (module.exports = {
|
105
100
|
PluginSettingsTabBase
|
106
101
|
});
|
107
|
-
//# 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": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAYA,sBAAiC;AAMjC,mBAAkC;AAClC,sBAAyB;AACzB,sBAAqB;AACrB,gCAAsC;AACtC,sBAA4B;AAkDrB,MAAe,8BAA+D,iCAAiB;AAAA,EAC7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,yBAAS,iBAAa,6BAAY,GAAG,yBAAS,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,uBAAmB,iDAAsB,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,wCAAkB,MAAM,KAAK,OAAO,gBAAgB,WAAW,CAAC;AAAA,EAClE;AACF;AAEA,SAAS,0BAA0B,OAAkD;AACnF,SAAO,CAAC,CAAE,MAA2C;AACvD;",
  "names": []
}

|
102
|
+
//# 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": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAWA,sBAAiC;AAOjC,mBAAkC;AAClC,sBAAyB;AACzB,sBAAqB;AACrB,gCAAsC;AACtC,+BAA0C;AAC1C,sBAA4B;AAkDrB,MAAe,8BAA+D,iCAAiB;AAAA,EAC7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,yBAAS,iBAAa,6BAAY,GAAG,yBAAS,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,uBAAmB,iDAAsB,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,eAAW,oDAA0B,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,wCAAkB,MAAM,KAAK,OAAO,gBAAgB,WAAW,CAAC;AAAA,EAClE;AACF;",
  "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
|
6
|
+
import type { ConditionalKeys, Promisable } from 'type-fest';
|
7
7
|
import { PluginSettingTab } from 'obsidian';
|
8
8
|
import type { StringKeys } from '../../Object.cjs';
|
9
9
|
import type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.cjs';
|
10
|
+
import type { ValidationMessageHolder } from '../ValidationMessage.cjs';
|
10
11
|
import type { PluginBase } from './PluginBase.cjs';
|
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,7 +25,7 @@ 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, Property extends StringKeys<PluginSettings>> extends BindOptions<PluginSettings[Property]> {
|
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.
|
@@ -38,10 +39,7 @@ export interface BindOptionsExtended<PluginSettings extends object, UIValue, Pro
|
|
38
39
|
*/
|
39
40
|
pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[Property]) => UIValue;
|
40
41
|
}
|
41
|
-
type ExtractPluginSettings<
|
42
|
-
interface ValidationMessageHolder {
|
43
|
-
validationMessage: string;
|
44
|
-
}
|
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.
|
@@ -61,7 +59,7 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
|
|
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>, property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions<UIValue>): TValueComponent;
|
65
63
|
/**
|
66
64
|
* Binds a value component to a plugin setting.
|
67
65
|
*
|
@@ -0,0 +1,38 @@
|
|
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 initCjs(){const globalThisRecord=globalThis;globalThisRecord["__name"]??=name;const originalRequire=require;if(originalRequire&&!originalRequire.__isPatched){require=Object.assign(id=>requirePatched(id),originalRequire,{__isPatched:true})}const newFuncs={__extractDefault:__name(()=>extractDefault,"__extractDefault"),process:__name(()=>{const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};return browserProcess},"process")};for(const key of Object.keys(newFuncs)){globalThisRecord[key]??=newFuncs[key]?.()}function name(obj){return obj}__name(name,"name");function extractDefault(module){return module&&module.__esModule&&module.default?module.default:module}__name(extractDefault,"extractDefault");function requirePatched(id){const module=originalRequire?.(id);if(module){return extractDefault(module)}if(id==="process"||id==="node:process"){console.error(`Module not found: ${id}. Fake process object is returned instead.`);return globalThis.process}console.error(`Module not found: ${id}. Empty object is returned instead.`);return{}}__name(requirePatched,"requirePatched")})();
|
7
|
+
|
8
|
+
"use strict";
|
9
|
+
var __defProp = Object.defineProperty;
|
10
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
11
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
12
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
13
|
+
var __export = (target, all) => {
|
14
|
+
for (var name in all)
|
15
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
16
|
+
};
|
17
|
+
var __copyProps = (to, from, except, desc) => {
|
18
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
19
|
+
for (let key of __getOwnPropNames(from))
|
20
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
21
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
22
|
+
}
|
23
|
+
return to;
|
24
|
+
};
|
25
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
26
|
+
var ValidationMessage_exports = {};
|
27
|
+
__export(ValidationMessage_exports, {
|
28
|
+
isValidationMessageHolder: () => isValidationMessageHolder
|
29
|
+
});
|
30
|
+
module.exports = __toCommonJS(ValidationMessage_exports);
|
31
|
+
function isValidationMessageHolder(value) {
|
32
|
+
return !!value.validationMessage;
|
33
|
+
}
|
34
|
+
// Annotate the CommonJS export names for ESM import in node:
|
35
|
+
0 && (module.exports = {
|
36
|
+
isValidationMessageHolder
|
37
|
+
});
|
38
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1ZhbGlkYXRpb25NZXNzYWdlLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEhvbGRzIGEgdmFsaWRhdGlvbiBtZXNzYWdlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgLyoqXG4gICAqIFRoZSB2YWxpZGF0aW9uIG1lc3NhZ2UuXG4gICAqL1xuICB2YWxpZGF0aW9uTWVzc2FnZTogc3RyaW5nO1xufVxuXG4vKipcbiAqIFR5cGUgZ3VhcmQgdG8gY2hlY2sgaWYgYSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIuXG4gKlxuICogQHBhcmFtIHZhbHVlIC0gVGhlIHZhbHVlIHRvIGNoZWNrLlxuICogQHJldHVybnMgYHRydWVgIGlmIHRoZSB2YWx1ZSBpcyBhIHZhbGlkYXRpb24gbWVzc2FnZSBob2xkZXIsIGBmYWxzZWAgb3RoZXJ3aXNlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNWYWxpZGF0aW9uTWVzc2FnZUhvbGRlcih2YWx1ZTogdW5rbm93bik6IHZhbHVlIGlzIFZhbGlkYXRpb25NZXNzYWdlSG9sZGVyIHtcbiAgcmV0dXJuICEhKHZhbHVlIGFzIFBhcnRpYWw8VmFsaWRhdGlvbk1lc3NhZ2VIb2xkZXI+KS52YWxpZGF0aW9uTWVzc2FnZTtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBZ0JPLFNBQVMsMEJBQTBCLE9BQWtEO0FBQzFGLFNBQU8sQ0FBQyxDQUFFLE1BQTJDO0FBQ3ZEOyIsCiAgIm5hbWVzIjogW10KfQo=
|
@@ -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;
|
@@ -64,6 +64,7 @@ __export(obsidian_exports, {
|
|
64
64
|
RenameDeleteHandler: () => RenameDeleteHandler,
|
65
65
|
ResourceUrl: () => ResourceUrl,
|
66
66
|
SettingEx: () => SettingEx,
|
67
|
+
ValidationMessage: () => ValidationMessage,
|
67
68
|
Vault: () => Vault,
|
68
69
|
VaultEx: () => VaultEx
|
69
70
|
});
|
@@ -97,6 +98,7 @@ var Reference = __toESM(__extractDefault(require('./Reference.cjs')), 1);
|
|
97
98
|
var RenameDeleteHandler = __toESM(__extractDefault(require('./RenameDeleteHandler.cjs')), 1);
|
98
99
|
var ResourceUrl = __toESM(__extractDefault(require('./ResourceUrl.cjs')), 1);
|
99
100
|
var SettingEx = __toESM(__extractDefault(require('./SettingEx.cjs')), 1);
|
101
|
+
var ValidationMessage = __toESM(__extractDefault(require('./ValidationMessage.cjs')), 1);
|
100
102
|
var Vault = __toESM(__extractDefault(require('./Vault.cjs')), 1);
|
101
103
|
var VaultEx = __toESM(__extractDefault(require('./VaultEx.cjs')), 1);
|
102
104
|
// Annotate the CommonJS export names for ESM import in node:
|
@@ -130,7 +132,8 @@ var VaultEx = __toESM(__extractDefault(require('./VaultEx.cjs')), 1);
|
|
130
132
|
RenameDeleteHandler,
|
131
133
|
ResourceUrl,
|
132
134
|
SettingEx,
|
135
|
+
ValidationMessage,
|
133
136
|
Vault,
|
134
137
|
VaultEx
|
135
138
|
});
|
136
|
-
//# sourceMappingURL=data:application/json;base64,
|
139
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKiBUSElTIElTIEEgR0VORVJBVEVEL0JVTkRMRUQgRklMRSBCWSBCVUlMRCBTQ1JJUFQgKi9cblxuZXhwb3J0ICogYXMgQXBwIGZyb20gJy4vQXBwLnRzJztcbmV4cG9ydCAqIGFzIEF0dGFjaG1lbnRQYXRoIGZyb20gJy4vQXR0YWNobWVudFBhdGgudHMnO1xuZXhwb3J0ICogYXMgQmFja2xpbmsgZnJvbSAnLi9CYWNrbGluay50cyc7XG5leHBvcnQgKiBhcyBDYWxsb3V0IGZyb20gJy4vQ2FsbG91dC50cyc7XG5leHBvcnQgKiBhcyBDb21wb25lbnRzIGZyb20gJy4vQ29tcG9uZW50cy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBEYXRhdmlldyBmcm9tICcuL0RhdGF2aWV3LnRzJztcbmV4cG9ydCAqIGFzIERhdGF2aWV3TGluayBmcm9tICcuL0RhdGF2aWV3TGluay50cyc7XG5leHBvcnQgKiBhcyBGaWxlQ2hhbmdlIGZyb20gJy4vRmlsZUNoYW5nZS50cyc7XG5leHBvcnQgKiBhcyBGaWxlTWFuYWdlciBmcm9tICcuL0ZpbGVNYW5hZ2VyLnRzJztcbmV4cG9ydCAqIGFzIEZpbGVTeXN0ZW0gZnJvbSAnLi9GaWxlU3lzdGVtLnRzJztcbmV4cG9ydCAqIGFzIEZyb250bWF0dGVyIGZyb20gJy4vRnJvbnRtYXR0ZXIudHMnO1xuZXhwb3J0ICogYXMgTGluayBmcm9tICcuL0xpbmsudHMnO1xuZXhwb3J0ICogYXMgTG9nZ2VyIGZyb20gJy4vTG9nZ2VyLnRzJztcbmV4cG9ydCAqIGFzIExvb3AgZnJvbSAnLi9Mb29wLnRzJztcbmV4cG9ydCAqIGFzIE1hcmtkb3duIGZyb20gJy4vTWFya2Rvd24udHMnO1xuZXhwb3J0ICogYXMgTWFya2Rvd25Db2RlQmxvY2tQcm9jZXNzb3IgZnJvbSAnLi9NYXJrZG93bkNvZGVCbG9ja1Byb2Nlc3Nvci50cyc7XG5leHBvcnQgKiBhcyBNYXJrZG93blZpZXcgZnJvbSAnLi9NYXJrZG93blZpZXcudHMnO1xuZXhwb3J0ICogYXMgTWV0YWRhdGFDYWNoZSBmcm9tICcuL01ldGFkYXRhQ2FjaGUudHMnO1xuZXhwb3J0ICogYXMgTW9kYWxzIGZyb20gJy4vTW9kYWxzL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIE1vbmtleUFyb3VuZCBmcm9tICcuL01vbmtleUFyb3VuZC50cyc7XG5leHBvcnQgKiBhcyBPYnNpZGlhblNldHRpbmdzIGZyb20gJy4vT2JzaWRpYW5TZXR0aW5ncy50cyc7XG5leHBvcnQgKiBhcyBQZGYgZnJvbSAnLi9QZGYudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luIGZyb20gJy4vUGx1Z2luL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIFF1ZXVlIGZyb20gJy4vUXVldWUudHMnO1xuZXhwb3J0ICogYXMgUmVhY3QgZnJvbSAnLi9SZWFjdC9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBSZWZlcmVuY2UgZnJvbSAnLi9SZWZlcmVuY2UudHMnO1xuZXhwb3J0ICogYXMgUmVuYW1lRGVsZXRlSGFuZGxlciBmcm9tICcuL1JlbmFtZURlbGV0ZUhhbmRsZXIudHMnO1xuZXhwb3J0ICogYXMgUmVzb3VyY2VVcmwgZnJvbSAnLi9SZXNvdXJjZVVybC50cyc7XG5leHBvcnQgKiBhcyBTZXR0aW5nRXggZnJvbSAnLi9TZXR0aW5nRXgudHMnO1xuZXhwb3J0ICogYXMgVmFsaWRhdGlvbk1lc3NhZ2UgZnJvbSAnLi9WYWxpZGF0aW9uTWVzc2FnZS50cyc7XG5leHBvcnQgKiBhcyBWYXVsdCBmcm9tICcuL1ZhdWx0LnRzJztcbmV4cG9ydCAqIGFzIFZhdWx0RXggZnJvbSAnLi9WYXVsdEV4LnRzJztcbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBRUEsVUFBcUI7QUFDckIscUJBQWdDO0FBQ2hDLGVBQTBCO0FBQzFCLGNBQXlCO0FBQ3pCLGlCQUE0QjtBQUM1QixlQUEwQjtBQUMxQixtQkFBOEI7QUFDOUIsaUJBQTRCO0FBQzVCLGtCQUE2QjtBQUM3QixpQkFBNEI7QUFDNUIsa0JBQTZCO0FBQzdCLFdBQXNCO0FBQ3RCLGFBQXdCO0FBQ3hCLFdBQXNCO0FBQ3RCLGVBQTBCO0FBQzFCLGlDQUE0QztBQUM1QyxtQkFBOEI7QUFDOUIsb0JBQStCO0FBQy9CLGFBQXdCO0FBQ3hCLG1CQUE4QjtBQUM5Qix1QkFBa0M7QUFDbEMsVUFBcUI7QUFDckIsYUFBd0I7QUFDeEIsWUFBdUI7QUFDdkIsWUFBdUI7QUFDdkIsZ0JBQTJCO0FBQzNCLDBCQUFxQztBQUNyQyxrQkFBNkI7QUFDN0IsZ0JBQTJCO0FBQzNCLHdCQUFtQztBQUNuQyxZQUF1QjtBQUN2QixjQUF5QjsiLAogICJuYW1lcyI6IFtdCn0K
|
@@ -27,5 +27,6 @@ export * as Reference from './Reference.cjs';
|
|
27
27
|
export * as RenameDeleteHandler from './RenameDeleteHandler.cjs';
|
28
28
|
export * as ResourceUrl from './ResourceUrl.cjs';
|
29
29
|
export * as SettingEx from './SettingEx.cjs';
|
30
|
+
export * as ValidationMessage from './ValidationMessage.cjs';
|
30
31
|
export * as Vault from './Vault.cjs';
|
31
32
|
export * as VaultEx from './VaultEx.cjs';
|
package/dist/lib/esm/Library.mjs
CHANGED
@@ -5,7 +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
|
-
const LIBRARY_VERSION = "22.1.1-beta.
|
8
|
+
const LIBRARY_VERSION = "22.1.1-beta.5";
|
9
9
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
10
10
|
const LIBRARY_STYLES = ".obsidian-dev-utils input[type=url] {\n height: var(--input-height);\n}\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=url],\n.obsidian-dev-utils input[type=week] {\n -webkit-app-region: no-drag;\n background: var(--background-modifier-form-field);\n border: var(--input-border-width) solid var(--background-modifier-border);\n color: var(--text-normal);\n font-family: inherit;\n padding: var(--size-4-1) var(--size-4-2);\n font-size: var(--font-ui-small);\n border-radius: var(--input-radius);\n outline: none;\n}\n@media (hover: hover) {\n .obsidian-dev-utils input[type=month]:hover,\n .obsidian-dev-utils input[type=time]:hover,\n .obsidian-dev-utils input[type=url]:hover,\n .obsidian-dev-utils input[type=week]:hover {\n border-color: var(--background-modifier-border-hover);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n }\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus {\n border-color: var(--background-modifier-border-focus);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus, .obsidian-dev-utils input[type=month]:focus-visible,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=time]:focus-visible,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=url]:focus-visible,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus,\n.obsidian-dev-utils input[type=week]:focus-visible {\n box-shadow: 0 0 0 2px var(--background-modifier-border-focus);\n}\n.obsidian-dev-utils input[type=month]::placeholder,\n.obsidian-dev-utils input[type=time]::placeholder,\n.obsidian-dev-utils input[type=url]::placeholder,\n.obsidian-dev-utils input[type=week]::placeholder {\n color: var(--text-faint);\n}\n.mod-rtl input[type=month],\n.mod-rtl input[type=time],\n.mod-rtl input[type=week],\n.is-rtl input[type=month],\n.is-rtl input[type=time],\n.is-rtl input[type=week],\n.rtl input[type=month],\n.rtl input[type=time],\n.rtl input[type=week] {\n direction: rtl;\n}\n.mod-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.rtl input[type=month]::-webkit-calendar-picker-indicator,\n.rtl input[type=time]::-webkit-calendar-picker-indicator,\n.rtl input[type=week]::-webkit-calendar-picker-indicator {\n right: var(--size-4-1);\n left: auto;\n}\n\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=week] {\n font-variant-numeric: tabular-nums;\n position: relative;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-text {\n color: var(--text-faint);\n padding-inline-end: 0;\n}\n.obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n position: absolute;\n left: var(--size-4-1);\n right: auto;\n opacity: 0.5;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:focus {\n background-color: var(--text-selection);\n color: var(--text-normal);\n cursor: text;\n}\n.mod-rtl .obsidian-dev-utils input[type=month], .is-rtl .obsidian-dev-utils input[type=month], .rtl .obsidian-dev-utils input[type=month],\n.mod-rtl .obsidian-dev-utils input[type=time],\n.is-rtl .obsidian-dev-utils input[type=time],\n.rtl .obsidian-dev-utils input[type=time],\n.mod-rtl .obsidian-dev-utils input[type=week],\n.is-rtl .obsidian-dev-utils input[type=week],\n.rtl .obsidian-dev-utils input[type=week] {\n direction: rtl;\n}\n.mod-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .is-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n left: auto;\n right: var(--size-4-1);\n}\n\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=month],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=time],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {\n padding-inline-start: var(--size-4-6);\n}\n\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator {\n margin-inline-start: 0;\n}\n\n.obsidian-dev-utils.modal-container .ok-button {\n margin-right: 10px;\n margin-top: 20px;\n}\n\n.obsidian-dev-utils .multiple-dropdown-component select,\n.obsidian-dev-utils .multiple-dropdown-component select:focus,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown {\n height: auto;\n}\n.obsidian-dev-utils .multiple-dropdown-component select option:checked,\n.obsidian-dev-utils .multiple-dropdown-component select:focus option:checked,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown option:checked {\n background-color: #1967d2;\n color: #fff;\n}\n\n.obsidian-dev-utils.prompt-modal .text-box {\n width: 100%;\n}\n\n.obsidian-dev-utils :invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n\n/*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AACE;EACE;;AAGF;AAAA;AAAA;AAAA;EAIE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC5DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;EACE;EACA;;;ACPJ;EACE;;;ACFJ;EACE%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */\n";
|
11
11
|
export {
|