obsidian-dev-utils 52.2.0 → 53.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +10 -0
- package/dist/lib/cjs/library.cjs +1 -1
- package/dist/lib/cjs/obsidian/components/all-windows-event-handler.cjs +190 -0
- package/dist/lib/cjs/obsidian/components/all-windows-event-handler.d.cts +44 -0
- package/dist/lib/cjs/obsidian/components/async-events-component.cjs +6 -1
- package/dist/lib/cjs/obsidian/components/async-events-component.d.cts +5 -0
- 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/plugin-base.cjs +3 -49
- package/dist/lib/cjs/obsidian/plugin/plugin-base.d.cts +0 -24
- package/dist/lib/cjs/obsidian/plugin/plugin-settings-tab-base.cjs +3 -6
- package/dist/lib/cjs/obsidian/plugin/plugin-settings-tab-base.d.cts +0 -1
- package/dist/lib/cjs/script-utils/linters/eslint-config.cjs +13 -4
- package/dist/lib/cjs/script-utils/linters/eslint-rules/index.cjs +12 -6
- package/dist/lib/cjs/script-utils/linters/eslint-rules/index.d.cts +3 -1
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.cjs +217 -0
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.d.cts +3 -0
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-used-underscore-variables.cjs +180 -0
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-used-underscore-variables.d.cts +13 -0
- package/dist/lib/cjs/script-utils/linters/eslint-rules/obsidian-dev-utils-plugin.cjs +5 -3
- package/dist/lib/cjs/script-utils/linters/eslint-rules/rule-tester-helper.cjs +135 -0
- package/dist/lib/cjs/script-utils/linters/eslint-rules/rule-tester-helper.d.cts +17 -0
- package/dist/lib/esm/library.mjs +1 -1
- package/dist/lib/esm/obsidian/components/all-windows-event-handler.d.mts +44 -0
- package/dist/lib/esm/obsidian/components/all-windows-event-handler.mjs +84 -0
- package/dist/lib/esm/obsidian/components/async-events-component.d.mts +5 -0
- package/dist/lib/esm/obsidian/components/async-events-component.mjs +6 -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/plugin-base.d.mts +0 -24
- package/dist/lib/esm/obsidian/plugin/plugin-base.mjs +3 -49
- package/dist/lib/esm/obsidian/plugin/plugin-settings-tab-base.d.mts +0 -1
- package/dist/lib/esm/obsidian/plugin/plugin-settings-tab-base.mjs +3 -6
- package/dist/lib/esm/script-utils/linters/eslint-config.mjs +13 -4
- package/dist/lib/esm/script-utils/linters/eslint-rules/index.d.mts +3 -1
- package/dist/lib/esm/script-utils/linters/eslint-rules/index.mjs +8 -4
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.d.mts +3 -0
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.mjs +117 -0
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-used-underscore-variables.d.mts +13 -0
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-used-underscore-variables.mjs +73 -0
- package/dist/lib/esm/script-utils/linters/eslint-rules/obsidian-dev-utils-plugin.mjs +5 -3
- package/dist/lib/esm/script-utils/linters/eslint-rules/rule-tester-helper.d.mts +17 -0
- package/dist/lib/esm/script-utils/linters/eslint-rules/rule-tester-helper.mjs +29 -0
- package/obsidian/Components/all-windows-event-handler/package.json +6 -0
- package/package.json +5 -2
- package/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return/package.json +6 -0
- package/script-utils/linters/eslint-rules/no-used-underscore-variables/package.json +6 -0
- package/script-utils/linters/eslint-rules/{no-used-underscore-params → rule-tester-helper}/package.json +3 -3
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-used-underscore-params.cjs +0 -174
- package/dist/lib/cjs/script-utils/linters/eslint-rules/no-used-underscore-params.d.cts +0 -12
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-used-underscore-params.d.mts +0 -12
- package/dist/lib/esm/script-utils/linters/eslint-rules/no-used-underscore-params.mjs +0 -68
|
@@ -24,7 +24,6 @@ import {
|
|
|
24
24
|
PluginSettingTab,
|
|
25
25
|
setTooltip
|
|
26
26
|
} from "obsidian";
|
|
27
|
-
import { AsyncEvents } from "../../async-events.mjs";
|
|
28
27
|
import {
|
|
29
28
|
convertAsyncToSync,
|
|
30
29
|
invokeAsyncSafely
|
|
@@ -57,7 +56,6 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
|
57
56
|
this.saveSettingsDebounceTimeoutInMilliseconds
|
|
58
57
|
);
|
|
59
58
|
this.asyncEventsComponent = new AsyncEventsComponent();
|
|
60
|
-
this.asyncEvents = new AsyncEvents();
|
|
61
59
|
}
|
|
62
60
|
/**
|
|
63
61
|
* Whether the plugin settings tab is open.
|
|
@@ -77,7 +75,6 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
|
77
75
|
return DEFAULT;
|
|
78
76
|
}
|
|
79
77
|
_isOpen = false;
|
|
80
|
-
asyncEvents;
|
|
81
78
|
asyncEventsComponent;
|
|
82
79
|
saveSettingsDebounced;
|
|
83
80
|
get pluginSettings() {
|
|
@@ -282,7 +279,7 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
|
282
279
|
await this.updateValidations(validationMessages);
|
|
283
280
|
}
|
|
284
281
|
on(name, callback, thisArg) {
|
|
285
|
-
return this.asyncEvents.on(name, callback, thisArg);
|
|
282
|
+
return this.asyncEventsComponent.asyncEvents.on(name, callback, thisArg);
|
|
286
283
|
}
|
|
287
284
|
async onSaveSettings(newSettings, _oldSettings, context) {
|
|
288
285
|
if (context === SAVE_TO_FILE_CONTEXT) {
|
|
@@ -293,7 +290,7 @@ class PluginSettingsTabBase extends PluginSettingTab {
|
|
|
293
290
|
}
|
|
294
291
|
async updateValidations(validationMessages) {
|
|
295
292
|
for (const [propertyName, validationMessage] of Object.entries(validationMessages)) {
|
|
296
|
-
await this.asyncEvents.triggerAsync("validationMessageChanged", propertyName, validationMessage);
|
|
293
|
+
await this.asyncEventsComponent.asyncEvents.triggerAsync("validationMessageChanged", propertyName, validationMessage);
|
|
297
294
|
}
|
|
298
295
|
}
|
|
299
296
|
}
|
|
@@ -301,4 +298,4 @@ export {
|
|
|
301
298
|
PluginSettingsTabBase,
|
|
302
299
|
SAVE_TO_FILE_CONTEXT
|
|
303
300
|
};
|
|
304
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin-settings-tab-base.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { Debouncer } from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../async-events.ts';\nimport type { StringKeys } from '../../type.ts';\nimport type { ValueComponentWithChangeTracking } from '../components/setting-components/value-component-with-change-tracking.ts';\nimport type { ValidationMessageHolder } from '../validation.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `PluginSettingsManagerBase` to use it in the tsdocs.\nimport type { PluginSettingsManagerBase } from './plugin-settings-manager-base.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractPluginSettingsPropertyNames,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './plugin-types-base.ts';\n\nimport { AsyncEvents } from '../../async-events.ts';\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../async.ts';\nimport { CssClass } from '../../css-class.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../function.ts';\nimport { deepEqual } from '../../object-utils.ts';\nimport { AsyncEventsComponent } from '../components/async-events-component.ts';\nimport { ensureWrapped } from '../components/setting-components/setting-component-wrapper.ts';\nimport { getTextBasedComponentValue } from '../components/setting-components/text-based-component.ts';\nimport { getValidatorComponent } from '../components/setting-components/validator-component.ts';\nimport { isValidationMessageHolder } from '../validation.ts';\nimport { addPluginCssClasses } from './plugin-context.ts';\n\n/**\n * A context passed to the {@link PluginSettingsManagerBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  readonly onChanged?: (newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>) => Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  readonly shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  readonly componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  readonly pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsTabBase<PluginTypes extends PluginTypesBase> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * A debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEvents: AsyncEvents;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private readonly saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ExtractPluginSettings<PluginTypes> {\n    return this.plugin.settingsManager.settingsWrapper.settings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public override plugin: ExtractPlugin<PluginTypes>) {\n    super(plugin.app, plugin);\n    addPluginCssClasses(this.containerEl, CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n    this.asyncEvents = new AsyncEvents();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<ExtractPluginSettings<PluginTypes>, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<PluginTypes>, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<PluginTypes>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.pluginSettings[propertyName] as ReadonlyDeep<PropertyType>;\n    const defaultValue = (this.plugin.settingsManager.defaultSettings as PluginSettings)[propertyName] as PropertyType;\n    const defaultComponentValue = optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>);\n    textBasedComponent?.setPlaceholderValue(defaultComponentValue);\n\n    let validationMessage: string;\n    let tooltipEl: HTMLElement | null = null;\n    let tooltipContentEl: HTMLElement | null = null;\n    if (validatorEl) {\n      const wrapper = ensureWrapped(validatorEl);\n      tooltipEl = wrapper.createDiv();\n      addPluginCssClasses(tooltipEl, CssClass.Tooltip, CssClass.TooltipValidator);\n      tooltipContentEl = tooltipEl.createSpan();\n      const tooltipArrowEl = tooltipEl.createDiv();\n      addPluginCssClasses(tooltipArrowEl, CssClass.TooltipArrow);\n      tooltipEl.hide();\n      wrapper.appendChild(tooltipEl);\n    }\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, anotherValidationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      validationMessage = anotherValidationMessage;\n      updateValidatorElDebounced();\n    }));\n\n    let shouldEmptyOnBlur = false;\n    let shouldRevertToDefaultValueOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n    const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;\n    const updateValidatorElDebounced = debounce(() => {\n      requestAnimationFrame(() => {\n        updateValidatorEl();\n      });\n    }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.pluginSettings[propertyName];\n      let newValue: PropertyType | undefined = undefined;\n      let shouldSetProperty = true;\n      shouldRevertToDefaultValueOnBlur = !!textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty;\n      if (shouldRevertToDefaultValueOnBlur) {\n        newValue = defaultValue;\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n          shouldSetProperty = false;\n        } else {\n          newValue = convertedValue;\n        }\n      }\n\n      if (shouldSetProperty) {\n        validationMessage = await this.plugin.settingsManager.setProperty(propertyName, newValue);\n        if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n          shouldEmptyOnBlur = true;\n        }\n      }\n\n      updateValidatorElDebounced();\n      if (shouldSetProperty) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue as ReadonlyDeep<PropertyType>);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorEl?.addEventListener('focus', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('blur', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('click', () => {\n      requestAnimationFrame(() => {\n        updateValidatorElDebounced();\n      });\n    });\n\n    validationMessage = this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName] ?? '';\n    updateValidatorElDebounced();\n\n    return valueComponent;\n\n    function updateValidatorEl(): void {\n      if (!validatorEl?.isActiveElement()) {\n        if (shouldEmptyOnBlur) {\n          shouldEmptyOnBlur = false;\n\n          if (!textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            textBasedComponent?.empty();\n          }\n        } else if (shouldRevertToDefaultValueOnBlur) {\n          shouldRevertToDefaultValueOnBlur = false;\n\n          if (textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            valueComponent.setValue(defaultComponentValue);\n          }\n        }\n      }\n\n      if (!validatorEl) {\n        return;\n      }\n\n      if (validationMessage === '') {\n        validatorEl.setCustomValidity('');\n        validatorEl.checkValidity();\n        validationMessage = validatorEl.validationMessage;\n      }\n\n      validatorEl.setCustomValidity(validationMessage);\n      if (optionsExt.shouldShowValidationMessage) {\n        if (tooltipContentEl) {\n          tooltipContentEl.textContent = validationMessage;\n        }\n        tooltipEl?.toggle(!!validationMessage);\n      } else if (validationMessage) {\n        setTooltip(validatorEl, validationMessage);\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.hideAsync());\n  }\n\n  /**\n   * Async actions to perform when the settings tab is being hidden.\n   *\n   * @returns A {@link Promise} that resolves when the settings tab is hidden.\n   */\n  public async hideAsync(): Promise<void> {\n    await this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT);\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedSettings - The loaded settings.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    this.display();\n    await noopAsync();\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns A {@link Promise} that resolves when the settings are revalidated.\n   */\n  protected async revalidate(): Promise<void> {\n    const validationMessages = await this.plugin.settingsManager.revalidate();\n    await this.updateValidations(validationMessages);\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      await this.updateValidations(newSettings.validationMessages as Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>);\n      return;\n    }\n\n    this.display();\n  }\n\n  private async updateValidations(validationMessages: Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>): Promise<void> {\n    for (const [propertyName, validationMessage] of Object.entries(validationMessages)) {\n      await this.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n    }\n  }\n}\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}

|
|
301
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin-settings-tab-base.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { Debouncer } from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../async-events.ts';\nimport type { StringKeys } from '../../type.ts';\nimport type { ValueComponentWithChangeTracking } from '../components/setting-components/value-component-with-change-tracking.ts';\nimport type { ValidationMessageHolder } from '../validation.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `PluginSettingsManagerBase` to use it in the tsdocs.\nimport type { PluginSettingsManagerBase } from './plugin-settings-manager-base.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractPluginSettingsPropertyNames,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './plugin-types-base.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../async.ts';\nimport { CssClass } from '../../css-class.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../function.ts';\nimport { deepEqual } from '../../object-utils.ts';\nimport { AsyncEventsComponent } from '../components/async-events-component.ts';\nimport { ensureWrapped } from '../components/setting-components/setting-component-wrapper.ts';\nimport { getTextBasedComponentValue } from '../components/setting-components/text-based-component.ts';\nimport { getValidatorComponent } from '../components/setting-components/validator-component.ts';\nimport { isValidationMessageHolder } from '../validation.ts';\nimport { addPluginCssClasses } from './plugin-context.ts';\n\n/**\n * A context passed to the {@link PluginSettingsManagerBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  readonly onChanged?: (newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>) => Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  readonly shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  readonly componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  readonly pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsTabBase<PluginTypes extends PluginTypesBase> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * A debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private readonly saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ExtractPluginSettings<PluginTypes> {\n    return this.plugin.settingsManager.settingsWrapper.settings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public override plugin: ExtractPlugin<PluginTypes>) {\n    super(plugin.app, plugin);\n    addPluginCssClasses(this.containerEl, CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<ExtractPluginSettings<PluginTypes>, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<PluginTypes>, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<PluginTypes>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.pluginSettings[propertyName] as ReadonlyDeep<PropertyType>;\n    const defaultValue = (this.plugin.settingsManager.defaultSettings as PluginSettings)[propertyName] as PropertyType;\n    const defaultComponentValue = optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>);\n    textBasedComponent?.setPlaceholderValue(defaultComponentValue);\n\n    let validationMessage: string;\n    let tooltipEl: HTMLElement | null = null;\n    let tooltipContentEl: HTMLElement | null = null;\n    if (validatorEl) {\n      const wrapper = ensureWrapped(validatorEl);\n      tooltipEl = wrapper.createDiv();\n      addPluginCssClasses(tooltipEl, CssClass.Tooltip, CssClass.TooltipValidator);\n      tooltipContentEl = tooltipEl.createSpan();\n      const tooltipArrowEl = tooltipEl.createDiv();\n      addPluginCssClasses(tooltipArrowEl, CssClass.TooltipArrow);\n      tooltipEl.hide();\n      wrapper.appendChild(tooltipEl);\n    }\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, anotherValidationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      validationMessage = anotherValidationMessage;\n      updateValidatorElDebounced();\n    }));\n\n    let shouldEmptyOnBlur = false;\n    let shouldRevertToDefaultValueOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n    const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;\n    const updateValidatorElDebounced = debounce(() => {\n      requestAnimationFrame(() => {\n        updateValidatorEl();\n      });\n    }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.pluginSettings[propertyName];\n      let newValue: PropertyType | undefined = undefined;\n      let shouldSetProperty = true;\n      shouldRevertToDefaultValueOnBlur = !!textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty;\n      if (shouldRevertToDefaultValueOnBlur) {\n        newValue = defaultValue;\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n          shouldSetProperty = false;\n        } else {\n          newValue = convertedValue;\n        }\n      }\n\n      if (shouldSetProperty) {\n        validationMessage = await this.plugin.settingsManager.setProperty(propertyName, newValue);\n        if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n          shouldEmptyOnBlur = true;\n        }\n      }\n\n      updateValidatorElDebounced();\n      if (shouldSetProperty) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue as ReadonlyDeep<PropertyType>);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorEl?.addEventListener('focus', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('blur', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('click', () => {\n      requestAnimationFrame(() => {\n        updateValidatorElDebounced();\n      });\n    });\n\n    validationMessage = this.plugin.settingsManager.settingsWrapper.validationMessages[propertyName] ?? '';\n    updateValidatorElDebounced();\n\n    return valueComponent;\n\n    function updateValidatorEl(): void {\n      if (!validatorEl?.isActiveElement()) {\n        if (shouldEmptyOnBlur) {\n          shouldEmptyOnBlur = false;\n\n          if (!textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            textBasedComponent?.empty();\n          }\n        } else if (shouldRevertToDefaultValueOnBlur) {\n          shouldRevertToDefaultValueOnBlur = false;\n\n          if (textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            valueComponent.setValue(defaultComponentValue);\n          }\n        }\n      }\n\n      if (!validatorEl) {\n        return;\n      }\n\n      if (validationMessage === '') {\n        validatorEl.setCustomValidity('');\n        validatorEl.checkValidity();\n        validationMessage = validatorEl.validationMessage;\n      }\n\n      validatorEl.setCustomValidity(validationMessage);\n      if (optionsExt.shouldShowValidationMessage) {\n        if (tooltipContentEl) {\n          tooltipContentEl.textContent = validationMessage;\n        }\n        tooltipEl?.toggle(!!validationMessage);\n      } else if (validationMessage) {\n        setTooltip(validatorEl, validationMessage);\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.plugin.settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.hideAsync());\n  }\n\n  /**\n   * Async actions to perform when the settings tab is being hidden.\n   *\n   * @returns A {@link Promise} that resolves when the settings tab is hidden.\n   */\n  public async hideAsync(): Promise<void> {\n    await this.plugin.settingsManager.saveToFile(SAVE_TO_FILE_CONTEXT);\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedSettings - The loaded settings.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    this.display();\n    await noopAsync();\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns A {@link Promise} that resolves when the settings are revalidated.\n   */\n  protected async revalidate(): Promise<void> {\n    const validationMessages = await this.plugin.settingsManager.revalidate();\n    await this.updateValidations(validationMessages);\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEventsComponent.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      await this.updateValidations(newSettings.validationMessages as Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>);\n      return;\n    }\n\n    this.display();\n  }\n\n  private async updateValidations(validationMessages: Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>): Promise<void> {\n    for (const [propertyName, validationMessage] of Object.entries(validationMessages)) {\n      await this.asyncEventsComponent.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n    }\n  }\n}\n/* v8 ignore stop */\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAgBA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAgBP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,gBAAgB;AACzB;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,iBAAiB;AAC1B,SAAS,4BAA4B;AACrC,SAAS,qBAAqB;AAC9B,SAAS,kCAAkC;AAC3C,SAAS,6BAA6B;AACtC,SAAS,iCAAiC;AAC1C,SAAS,2BAA2B;AAK7B,MAAM,uBAAuB;AA4D7B,MAAe,8BAAmE,iBAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAiCjG,YAA4B,QAAoC;AACrE,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,wBAAoB,KAAK,aAAa,SAAS,iBAAiB;AAChE,SAAK,wBAAwB;AAAA,MAC3B,mBAAmB,MAAM,KAAK,OAAO,gBAAgB,WAAW,oBAAoB,CAAC;AAAA,MACrF,KAAK;AAAA,IACP;AACA,SAAK,uBAAuB,IAAI,qBAAqB;AAAA,EACvD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAnCA,IAAW,SAAkB;AAC3B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAc,4CAAoD;AAChE,UAAM,UAAU;AAChB,WAAO;AAAA,EACT;AAAA,EAEQ,UAAU;AAAA,EACD;AAAA,EACA;AAAA,EAEjB,IAAY,iBAAqD;AAC/D,WAAO,KAAK,OAAO,gBAAgB,gBAAgB;AAAA,EACrD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkEO,KAKL,gBACA,cACA,SACiB;AAGjB,UAAM,kBAAwF;AAAA,MAC5F,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,WAAW;AAAA,MACX,yCAAyC,CAAC,UAA+C;AAAA,MACzF,wCAAwC;AAAA,MACxC,uCAAuC;AAAA,MACvC,6BAA6B;AAAA,IAC/B;AAEA,UAAM,aAAmF,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAE1H,UAAM,cAAc,sBAAsB,cAAc,GAAG;AAE3D,UAAM,qBAAqB,2BAA2B,cAAc;AAEpE,UAAM,gBAAgB,KAAK,eAAe,YAAY;AACtD,UAAM,eAAgB,KAAK,OAAO,gBAAgB,gBAAmC,YAAY;AACjG,UAAM,wBAAwB,WAAW,wCAAwC,YAA0C;AAC3H,wBAAoB,oBAAoB,qBAAqB;AAE7D,QAAI;AACJ,QAAI,YAAgC;AACpC,QAAI,mBAAuC;AAC3C,QAAI,aAAa;AACf,YAAM,UAAU,cAAc,WAAW;AACzC,kBAAY,QAAQ,UAAU;AAC9B,0BAAoB,WAAW,SAAS,SAAS,SAAS,gBAAgB;AAC1E,yBAAmB,UAAU,WAAW;AACxC,YAAM,iBAAiB,UAAU,UAAU;AAC3C,0BAAoB,gBAAgB,SAAS,YAAY;AACzD,gBAAU,KAAK;AACf,cAAQ,YAAY,SAAS;AAAA,IAC/B;AAEA,SAAK,qBAAqB,mBAAmB,KAAK,GAAG,4BAA4B,CAAC,qBAAqB,6BAA6B;AAClI,UAAI,iBAAiB,qBAAqB;AACxC;AAAA,MACF;AAEA,0BAAoB;AACpB,iCAA2B;AAAA,IAC7B,CAAC,CAAC;AAEF,QAAI,oBAAoB;AACxB,QAAI,mCAAmC;AAEvC,QAAI,sBAAsB,WAAW,yCAAyC,UAAU,eAAe,YAAY,GAAG;AACpH,yBAAmB,MAAM;AAAA,IAC3B,OAAO;AACL,qBAAe,SAAS,WAAW,wCAAwC,aAAa,CAAC;AAAA,IAC3F;AAEA,QAAI,qBAAqB;AACzB,UAAM,8CAA8C;AACpD,UAAM,6BAA6B,SAAS,MAAM;AAChD,4BAAsB,MAAM;AAC1B,0BAAkB;AAAA,MACpB,CAAC;AAAA,IACH,GAAG,2CAA2C;AAE9C,mBAAe,SAAS,OAAO,YAAY;AACzC,UAAI,oBAAoB;AACtB,6BAAqB;AACrB;AAAA,MACF;AAEA,0BAAoB;AAEpB,YAAM,WAAW,KAAK,eAAe,YAAY;AACjD,UAAI,WAAqC;AACzC,UAAI,oBAAoB;AACxB,yCAAmC,CAAC,CAAC,oBAAoB,QAAQ,KAAK,WAAW;AACjF,UAAI,kCAAkC;AACpC,mBAAW;AAAA,MACb,OAAO;AACL,cAAM,iBAAiB,WAAW,wCAAwC,OAAO;AACjF,YAAI,0BAA0B,cAAc,GAAG;AAC7C,8BAAoB,eAAe;AACnC,8BAAoB;AAAA,QACtB,OAAO;AACL,qBAAW;AAAA,QACb;AAAA,MACF;AAEA,UAAI,mBAAmB;AACrB,4BAAoB,MAAM,KAAK,OAAO,gBAAgB,YAAY,cAAc,QAAQ;AACxF,YAAI,sBAAsB,WAAW,yCAAyC,CAAC,mBAAmB,QAAQ,KAAK,UAAU,UAAU,YAAY,GAAG;AAChJ,8BAAoB;AAAA,QACtB;AAAA,MACF;AAEA,iCAA2B;AAC3B,UAAI,mBAAmB;AACrB,cAAM,WAAW,UAAU,UAAwC,QAAsC;AAAA,MAC3G;AACA,WAAK,sBAAsB;AAAA,IAC7B,CAAC;AAED,iBAAa,iBAAiB,SAAS,MAAM;AAC3C,iCAA2B;AAAA,IAC7B,CAAC;AACD,iBAAa,iBAAiB,QAAQ,MAAM;AAC1C,iCAA2B;AAAA,IAC7B,CAAC;AACD,iBAAa,iBAAiB,SAAS,MAAM;AAC3C,4BAAsB,MAAM;AAC1B,mCAA2B;AAAA,MAC7B,CAAC;AAAA,IACH,CAAC;AAED,wBAAoB,KAAK,OAAO,gBAAgB,gBAAgB,mBAAmB,YAAY,KAAK;AACpG,+BAA2B;AAE3B,WAAO;AAEP,aAAS,oBAA0B;AACjC,UAAI,CAAC,aAAa,gBAAgB,GAAG;AACnC,YAAI,mBAAmB;AACrB,8BAAoB;AAEpB,cAAI,CAAC,oBAAoB,QAAQ,GAAG;AAClC,iCAAqB;AACrB,gCAAoB,MAAM;AAAA,UAC5B;AAAA,QACF,WAAW,kCAAkC;AAC3C,6CAAmC;AAEnC,cAAI,oBAAoB,QAAQ,GAAG;AACjC,iCAAqB;AACrB,2BAAe,SAAS,qBAAqB;AAAA,UAC/C;AAAA,QACF;AAAA,MACF;AAEA,UAAI,CAAC,aAAa;AAChB;AAAA,MACF;AAEA,UAAI,sBAAsB,IAAI;AAC5B,oBAAY,kBAAkB,EAAE;AAChC,oBAAY,cAAc;AAC1B,4BAAoB,YAAY;AAAA,MAClC;AAEA,kBAAY,kBAAkB,iBAAiB;AAC/C,UAAI,WAAW,6BAA6B;AAC1C,YAAI,kBAAkB;AACpB,2BAAiB,cAAc;AAAA,QACjC;AACA,mBAAW,OAAO,CAAC,CAAC,iBAAiB;AAAA,MACvC,WAAW,mBAAmB;AAC5B,mBAAW,aAAa,iBAAiB;AAAA,MAC3C;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKgB,UAAgB;AAC9B,SAAK,YAAY,MAAM;AACvB,SAAK,UAAU;AACf,SAAK,qBAAqB,KAAK;AAC/B,SAAK,qBAAqB,mBAAmB,KAAK,OAAO,gBAAgB,GAAG,gBAAgB,KAAK,eAAe,KAAK,IAAI,CAAC,CAAC;AAC3H,SAAK,qBAAqB,mBAAmB,KAAK,OAAO,gBAAgB,GAAG,gBAAgB,KAAK,eAAe,KAAK,IAAI,CAAC,CAAC;AAAA,EAC7H;AAAA;AAAA;AAAA;AAAA,EAKgB,OAAa;AAC3B,UAAM,KAAK;AACX,SAAK,sBAAsB,OAAO;AAClC,SAAK,UAAU;AACf,SAAK,qBAAqB,OAAO;AACjC,SAAK,qBAAqB,KAAK;AAC/B,sBAAkB,MAAM,KAAK,UAAU,CAAC;AAAA,EAC1C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAa,YAA2B;AACtC,UAAM,KAAK,OAAO,gBAAgB,WAAW,oBAAoB;AAAA,EACnE;AAAA;AAAA;AAAA;AAAA,EAKO,OAAa;AAClB,SAAK,IAAI,QAAQ,QAAQ,IAAI;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAgB,eAAe,iBAAoE,gBAAwC;AACzI,SAAK,QAAQ;AACb,UAAM,UAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,aAA4B;AAC1C,UAAM,qBAAqB,MAAM,KAAK,OAAO,gBAAgB,WAAW;AACxE,UAAM,KAAK,kBAAkB,kBAAkB;AAAA,EACjD;AAAA,EAUQ,GACN,MACA,UACA,SACe;AACf,WAAO,KAAK,qBAAqB,YAAY,GAAG,MAAM,UAAU,OAAO;AAAA,EACzE;AAAA,EAEA,MAAc,eACZ,aACA,cACA,SACe;AACf,QAAI,YAAY,sBAAsB;AACpC,YAAM,KAAK,kBAAkB,YAAY,kBAAqF;AAC9H;AAAA,IACF;AAEA,SAAK,QAAQ;AAAA,EACf;AAAA,EAEA,MAAc,kBAAkB,oBAAoG;AAClI,eAAW,CAAC,cAAc,iBAAiB,KAAK,OAAO,QAAQ,kBAAkB,GAAG;AAClF,YAAM,KAAK,qBAAqB,YAAY,aAAa,4BAA4B,cAAc,iBAAiB;AAAA,IACtH;AAAA,EACF;AACF;",
  "names": []
}

|
|
@@ -29,6 +29,7 @@ import obsidianmd from "eslint-plugin-obsidianmd";
|
|
|
29
29
|
import { configs as perfectionistConfigs } from "eslint-plugin-perfectionist";
|
|
30
30
|
import { defineConfig } from "eslint/config";
|
|
31
31
|
import globals from "globals";
|
|
32
|
+
import { existsSync } from "node:fs";
|
|
32
33
|
import tseslint from "typescript-eslint";
|
|
33
34
|
import { ObsidianPluginRepoPaths } from "../../obsidian/plugin/obsidian-plugin-repo-paths.mjs";
|
|
34
35
|
import { join } from "../../path.mjs";
|
|
@@ -84,7 +85,7 @@ function defineEslintConfigs(params = {}) {
|
|
|
84
85
|
}
|
|
85
86
|
const customConfigs = params.customConfigs?.(context) ?? [];
|
|
86
87
|
return defineConfig(
|
|
87
|
-
|
|
88
|
+
...getGitIgnoreConfigs(),
|
|
88
89
|
...getEslintConfigs(context),
|
|
89
90
|
...getTseslintConfigs(context),
|
|
90
91
|
...getStylisticConfigs(context),
|
|
@@ -292,6 +293,13 @@ function getEslintImportResolverTypescriptConfigs() {
|
|
|
292
293
|
}
|
|
293
294
|
]);
|
|
294
295
|
}
|
|
296
|
+
function getGitIgnoreConfigs() {
|
|
297
|
+
const gitignorePath = join(getRootFolder() ?? "", ".gitignore");
|
|
298
|
+
if (!existsSync(gitignorePath)) {
|
|
299
|
+
return [];
|
|
300
|
+
}
|
|
301
|
+
return [includeIgnoreFile(gitignorePath)];
|
|
302
|
+
}
|
|
295
303
|
function getImportXConfigs(context) {
|
|
296
304
|
return defineConfig([
|
|
297
305
|
{
|
|
@@ -367,7 +375,7 @@ function getObsidianDevUtilsPluginConfigs(context) {
|
|
|
367
375
|
"obsidian-dev-utils": obsidianDevUtilsPlugin
|
|
368
376
|
},
|
|
369
377
|
rules: {
|
|
370
|
-
"obsidian-dev-utils/no-used-underscore-
|
|
378
|
+
"obsidian-dev-utils/no-used-underscore-variables": "error"
|
|
371
379
|
}
|
|
372
380
|
}
|
|
373
381
|
]);
|
|
@@ -518,7 +526,8 @@ function getTseslintConfigs(context) {
|
|
|
518
526
|
}
|
|
519
527
|
],
|
|
520
528
|
"@typescript-eslint/prefer-readonly": "error",
|
|
521
|
-
"obsidian-dev-utils/no-
|
|
529
|
+
"obsidian-dev-utils/no-async-callback-to-unsafe-return": "error",
|
|
530
|
+
"obsidian-dev-utils/no-used-underscore-variables": "error"
|
|
522
531
|
}
|
|
523
532
|
},
|
|
524
533
|
{
|
|
@@ -534,4 +543,4 @@ export {
|
|
|
534
543
|
EslintConfigContext,
|
|
535
544
|
defineEslintConfigs
|
|
536
545
|
};
|
|
537
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/script-utils/linters/eslint-config.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * ESLint configuration for TypeScript projects with various plugins.\n *\n * This module exports ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n * such as `@typescript-eslint/parser`, `@typescript-eslint/eslint-plugin`,\n * `eslint-plugin-modules-newlines`, `@stylistic/eslint-plugin`.\n * It sets up parsers, plugins, and rules for maintaining code quality and consistency.\n */\n\n/* v8 ignore start -- Declarative ESLint rule/plugin configuration; correctness is verified by running ESLint, not unit tests. */\n\nimport type {\n  ESLint,\n  Linter\n} from 'eslint';\n\n/* eslint-disable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\nimport commentsConfigs from '@eslint-community/eslint-plugin-eslint-comments/configs';\nimport { includeIgnoreFile } from '@eslint/compat';\nimport eslint from '@eslint/js';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport stylistic from '@stylistic/eslint-plugin';\nimport { createTypeScriptImportResolver } from 'eslint-import-resolver-typescript';\nimport { flatConfigs as eslintPluginImportXFlatConfigs } from 'eslint-plugin-import-x';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport obsidianmd from 'eslint-plugin-obsidianmd';\nimport { configs as perfectionistConfigs } from 'eslint-plugin-perfectionist';\nimport { defineConfig } from 'eslint/config';\nimport globals from 'globals';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `_default` is too confusing.\nimport tseslint from 'typescript-eslint';\n\nimport { ObsidianPluginRepoPaths } from '../../obsidian/plugin/obsidian-plugin-repo-paths.ts';\nimport { join } from '../../path.ts';\nimport { getRootFolder } from '../root.ts';\nimport { obsidianDevUtilsPlugin } from './eslint-rules/obsidian-dev-utils-plugin.ts';\n\n/**\n * The parameters for defining ESLint configurations.\n */\nexport interface DefineEslintConfigsParams {\n  /**\n   * A function that builds custom ESLint configurations.\n   *\n   * @param context - The ESLint configuration context.\n   * @returns The custom ESLint configurations.\n   */\n  customConfigs?(context: EslintConfigContext): Linter.Config[];\n\n  /**\n   * A function that edits the ESLint configuration context.\n   *\n   * @param context - The ESLint configuration context.\n   */\n  editContext?(context: EslintConfigContext): void;\n}\n\n/**\n * The context for defining ESLint configurations.\n */\nexport class EslintConfigContext {\n  /**\n   * The root configuration files.\n   */\n  public readonly rootConfigFiles: string[] = [];\n\n  /**\n   * The script files.\n   */\n  public readonly scriptFiles: string[] = [];\n\n  /**\n   * The source files.\n   */\n  public readonly sourceFiles: string[] = [];\n\n  /**\n   * The test files.\n   */\n  public readonly testFiles: string[] = [];\n\n  /**\n   * The all files.\n   *\n   * @returns The all files.\n   */\n  public allFiles(): string[] {\n    return [...this.testFiles, ...this.scriptFiles, ...this.rootConfigFiles, ...this.sourceFiles];\n  }\n}\n\n/**\n * Build ESLint configurations.\n *\n * This function builds ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n *\n * @param params - The parameters for defining ESLint configurations.\n * @returns The ESLint configurations.\n */\nexport function defineEslintConfigs(params: DefineEslintConfigsParams = {}): Linter.Config[] {\n  const context = new EslintConfigContext();\n  context.rootConfigFiles.push(\n    ObsidianPluginRepoPaths.CommitlintConfigTs,\n    ObsidianPluginRepoPaths.EslintConfigMts,\n    ObsidianPluginRepoPaths.VitestConfigTs\n  );\n  context.scriptFiles.push(\n    join(ObsidianPluginRepoPaths.Scripts, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs)\n  );\n  context.sourceFiles.push(\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTsx)\n  );\n  context.testFiles.push(\n    join(ObsidianPluginRepoPaths.Tests, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Mocks, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTestTs)\n  );\n\n  if (params.editContext) {\n    params.editContext(context);\n  }\n\n  const customConfigs = params.customConfigs?.(context) ?? [];\n\n  return defineConfig(\n    includeIgnoreFile(join(getRootFolder() ?? '', '.gitignore')),\n    ...getEslintConfigs(context),\n    ...getTseslintConfigs(context),\n    ...getStylisticConfigs(context),\n    ...getObsidianLintConfigs(context),\n    ...getImportXConfigs(context),\n    ...getPerfectionistConfigs(context),\n    ...getEslintImportResolverTypescriptConfigs(),\n    ...getEslintCommentsConfigs(context),\n    ...getObsidianDevUtilsPluginConfigs(context),\n    ...customConfigs\n  );\n}\n\nfunction getEslintCommentsConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `recommended` is too confusing.\n      extends: [commentsConfigs.recommended],\n      files: context.allFiles(),\n      rules: {\n        '@eslint-community/eslint-comments/require-description': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [eslint.configs.recommended],\n      files: context.allFiles(),\n      rules: {\n        'accessor-pairs': 'error',\n        'array-callback-return': 'error',\n        'camelcase': 'error',\n        'capitalized-comments': ['error', 'always', { block: { ignorePattern: 'v8' } }],\n        'complexity': 'error',\n        'consistent-this': 'error',\n        'curly': 'error',\n        'default-case': 'error',\n        'default-case-last': 'error',\n        'default-param-last': 'error',\n        'eqeqeq': 'error',\n        'func-name-matching': 'error',\n        'func-names': 'error',\n        'func-style': [\n          'error',\n          'declaration',\n          {\n            allowArrowFunctions: false\n          }\n        ],\n        'grouped-accessor-pairs': [\n          'error',\n          'getBeforeSet'\n        ],\n        'guard-for-in': 'error',\n        'no-alert': 'error',\n        'no-array-constructor': 'error',\n        'no-bitwise': 'error',\n        'no-caller': 'error',\n        'no-console': [\n          'error',\n          {\n            allow: [\n              'warn',\n              'error'\n            ]\n          }\n        ],\n        'no-constructor-return': 'error',\n        'no-div-regex': 'error',\n        'no-else-return': [\n          'error',\n          {\n            allowElseIf: false\n          }\n        ],\n        'no-empty-function': 'error',\n        'no-extend-native': 'error',\n        'no-extra-bind': 'error',\n        'no-extra-label': 'error',\n        'no-implicit-coercion': [\n          'error',\n          {\n            allow: [\n              '!!'\n            ]\n          }\n        ],\n        'no-implied-eval': 'error',\n        'no-inner-declarations': 'error',\n        'no-iterator': 'error',\n        'no-label-var': 'error',\n        'no-labels': 'error',\n        'no-lone-blocks': 'error',\n        'no-lonely-if': 'error',\n        'no-loop-func': 'error',\n        'no-magic-numbers': [\n          'error',\n          {\n            detectObjects: true,\n            enforceConst: true,\n            ignore: [\n              -1,\n              0,\n              1\n            ]\n          }\n        ],\n        'no-multi-assign': 'error',\n        'no-multi-str': 'error',\n        'no-negated-condition': 'error',\n        'no-nested-ternary': 'error',\n        'no-new-func': 'error',\n        'no-new-wrappers': 'error',\n        'no-object-constructor': 'error',\n        'no-octal-escape': 'error',\n        'no-promise-executor-return': 'error',\n        'no-proto': 'error',\n        'no-restricted-syntax': [\n          'error',\n          {\n            message: 'Do not use definite assignment assertions (!). Initialize the field or make it optional.',\n            selector: 'PropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use definite assignment assertions (!) on abstract fields.',\n            selector: 'TSAbstractPropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function parameters. Define a named interface instead.',\n            selector: ':function > Identifier TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function return types. Define a named interface instead.',\n            selector: ':function > TSTypeAnnotation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in interface/method signatures. Define a named interface instead.',\n            selector: 'TSMethodSignature TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types as type arguments. Define a named interface instead.',\n            selector: 'TSTypeParameterInstantiation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in type annotations. Define a named interface instead.',\n            selector: 'TSTypeAnnotation TSTypeLiteral'\n          }\n        ],\n        'no-return-assign': 'error',\n        'no-script-url': 'error',\n        'no-self-compare': 'error',\n        'no-sequences': 'error',\n        'no-shadow': 'error',\n        'no-template-curly-in-string': 'error',\n        'no-throw-literal': 'error',\n        'no-unmodified-loop-condition': 'error',\n        'no-unneeded-ternary': 'error',\n        'no-unreachable-loop': 'error',\n        'no-unused-expressions': 'error',\n        'no-useless-assignment': 'error',\n        'no-useless-call': 'error',\n        'no-useless-computed-key': 'error',\n        'no-useless-concat': 'error',\n        'no-useless-constructor': 'error',\n        'no-useless-rename': 'error',\n        'no-useless-return': 'error',\n        'no-var': 'error',\n        'no-void': 'error',\n        'object-shorthand': 'error',\n        'operator-assignment': 'error',\n        'prefer-arrow-callback': 'error',\n        'prefer-const': 'error',\n        'prefer-exponentiation-operator': 'error',\n        'prefer-named-capture-group': 'error',\n        'prefer-numeric-literals': 'error',\n        'prefer-object-has-own': 'error',\n        'prefer-object-spread': 'error',\n        'prefer-promise-reject-errors': 'error',\n        'prefer-regex-literals': 'error',\n        'prefer-rest-params': 'error',\n        'prefer-spread': 'error',\n        'prefer-template': 'error',\n        'radix': 'error',\n        'require-atomic-updates': 'error',\n        'require-await': 'error',\n        'symbol-description': 'error',\n        'unicode-bom': 'error',\n        'vars-on-top': 'error',\n        'yoda': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintImportResolverTypescriptConfigs(): Linter.Config[] {\n  return defineConfig([\n    {\n      settings: {\n        'import-x/resolver-next': [\n          createTypeScriptImportResolver({\n            alwaysTryTypes: true\n          })\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getImportXConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        eslintPluginImportXFlatConfigs.recommended as Linter.Config,\n        eslintPluginImportXFlatConfigs.typescript as Linter.Config,\n        eslintPluginImportXFlatConfigs.errors as Linter.Config,\n        eslintPluginImportXFlatConfigs.warnings as Linter.Config\n      ],\n      files: context.allFiles(),\n      rules: {\n        'import-x/consistent-type-specifier-style': 'error',\n        'import-x/extensions': ['error', 'ignorePackages'],\n        'import-x/first': 'error',\n        'import-x/imports-first': 'error',\n        'import-x/newline-after-import': 'error',\n        'import-x/no-absolute-path': 'error',\n        'import-x/no-amd': 'error',\n        'import-x/no-anonymous-default-export': 'error',\n        'import-x/no-commonjs': 'error',\n        'import-x/no-cycle': 'error',\n        'import-x/no-default-export': 'error',\n        'import-x/no-deprecated': 'error',\n        'import-x/no-duplicates': 'error',\n        'import-x/no-dynamic-require': 'error',\n        'import-x/no-empty-named-blocks': 'error',\n        'import-x/no-extraneous-dependencies': 'error',\n        'import-x/no-import-module-exports': 'error',\n        'import-x/no-mutable-exports': 'error',\n        'import-x/no-named-default': 'error',\n        'import-x/no-namespace': 'error',\n        'import-x/no-nodejs-modules': 'error',\n        'import-x/no-relative-packages': 'error',\n        'import-x/no-restricted-paths': 'error',\n        'import-x/no-self-import': 'error',\n        'import-x/no-unassigned-import': [\n          'error',\n          {\n            allow: [\n              '**/*.css',\n              '**/*.sass',\n              '**/*.scss'\n            ]\n          }\n        ],\n        'import-x/no-unused-modules': 'off',\n        'import-x/no-useless-path-segments': 'error',\n        'import-x/no-webpack-loader-syntax': 'error'\n      }\n    },\n    {\n      files: context.scriptFiles,\n      rules: {\n        'import-x/no-nodejs-modules': 'off'\n      }\n    },\n    {\n      files: [\n        ...context.rootConfigFiles,\n        join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.MainTs)\n      ],\n      rules: {\n        'import-x/no-default-export': 'off'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianDevUtilsPluginConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      files: context.allFiles(),\n      plugins: {\n        'obsidian-dev-utils': obsidianDevUtilsPlugin\n      },\n      rules: {\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianLintConfigs(context: EslintConfigContext): Linter.Config[] {\n  const obsidianRecommendedConfigs = Array.from(obsidianmd.configs?.['recommended'] as Iterable<Linter.Config>);\n\n  const scopedObsidianRecommendedConfigs = obsidianRecommendedConfigs.map((config) => {\n    if (config.files?.includes('package.json')) {\n      return config;\n    }\n\n    return {\n      ...config,\n      files: context.sourceFiles\n    };\n  });\n\n  return defineConfig([\n    ...scopedObsidianRecommendedConfigs,\n    {\n      plugins: {\n        obsidianmd: obsidianmd as ESLint.Plugin\n      }\n    },\n    {\n      languageOptions: {\n        globals: {\n          ...globals.browser,\n          ...globals.node,\n          activeDocument: 'readonly',\n          activeWindow: 'readonly',\n          ajax: 'readonly',\n          ajaxPromise: 'readonly',\n          createDiv: 'readonly',\n          createEl: 'readonly',\n          createFragment: 'readonly',\n          createSpan: 'readonly',\n          createSvg: 'readonly',\n          DomElementInfo: 'readonly',\n          fish: 'readonly',\n          fishAll: 'readonly',\n          isBoolean: 'readonly',\n          nextFrame: 'readonly',\n          NodeJS: 'readonly',\n          ready: 'readonly',\n          sleep: 'readonly'\n        }\n      }\n    }\n  ]);\n}\n\nfunction getPerfectionistConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([{\n    extends: [perfectionistConfigs['recommended-alphabetical']],\n    files: context.allFiles()\n  }]);\n}\n\nfunction getStylisticConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        stylistic.configs.recommended,\n        stylistic.configs.customize({\n          arrowParens: true,\n          braceStyle: '1tbs',\n          commaDangle: 'never',\n          semi: true\n        })\n      ],\n      files: context.allFiles(),\n      rules: {\n        '@stylistic/generator-star-spacing': 'off',\n        '@stylistic/indent': 'off',\n        '@stylistic/indent-binary-ops': 'off',\n        '@stylistic/jsx-one-expression-per-line': 'off',\n        '@stylistic/no-extra-semi': 'error',\n        '@stylistic/object-curly-newline': [\n          'error',\n          {\n            ExportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            },\n            ImportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            }\n          }\n        ],\n        '@stylistic/operator-linebreak': [\n          'error',\n          'before',\n          {\n            overrides: {\n              '=': 'after'\n            }\n          }\n        ],\n        '@stylistic/quotes': [\n          'error',\n          'single',\n          {\n            allowTemplateLiterals: 'never'\n          }\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getTseslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.strictTypeChecked,\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.stylisticTypeChecked\n      ],\n      files: context.allFiles(),\n      languageOptions: {\n        parserOptions: {\n          ecmaFeatures: {\n            jsx: true\n          },\n          projectService: true,\n          tsconfigRootDir: getRootFolder() ?? ''\n        }\n      },\n      rules: {\n        '@typescript-eslint/explicit-function-return-type': 'error',\n        '@typescript-eslint/explicit-member-accessibility': 'error',\n        '@typescript-eslint/no-invalid-void-type': ['error', {\n          allowAsThisParameter: true\n        }],\n        '@typescript-eslint/no-this-alias': ['error', {\n          allowedNames: [\n            'that'\n          ]\n        }],\n        '@typescript-eslint/no-unused-vars': [\n          'error',\n          {\n            args: 'all',\n            argsIgnorePattern: '^_',\n            caughtErrors: 'all',\n            caughtErrorsIgnorePattern: '^_',\n            destructuredArrayIgnorePattern: '^_',\n            ignoreRestSiblings: true,\n            varsIgnorePattern: '^_'\n          }\n        ],\n        '@typescript-eslint/prefer-readonly': 'error',\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    },\n    {\n      settings: {\n        react: {\n          version: 'detect'\n        }\n      }\n    }\n  ]);\n}\n\n/* eslint-enable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\n\n/* v8 ignore stop */\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAmBA,OAAO,qBAAqB;AAC5B,SAAS,yBAAyB;AAClC,OAAO,YAAY;AAEnB,OAAO,eAAe;AACtB,SAAS,sCAAsC;AAC/C,SAAS,eAAe,sCAAsC;AAE9D,OAAO,gBAAgB;AACvB,SAAS,WAAW,4BAA4B;AAChD,SAAS,oBAAoB;AAC7B,OAAO,aAAa;AAEpB,OAAO,cAAc;AAErB,SAAS,+BAA+B;AACxC,SAAS,YAAY;AACrB,SAAS,qBAAqB;AAC9B,SAAS,8BAA8B;AAyBhC,MAAM,oBAAoB;AAAA;AAAA;AAAA;AAAA,EAIf,kBAA4B,CAAC;AAAA;AAAA;AAAA;AAAA,EAK7B,cAAwB,CAAC;AAAA;AAAA;AAAA;AAAA,EAKzB,cAAwB,CAAC;AAAA;AAAA;AAAA;AAAA,EAKzB,YAAsB,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOhC,WAAqB;AAC1B,WAAO,CAAC,GAAG,KAAK,WAAW,GAAG,KAAK,aAAa,GAAG,KAAK,iBAAiB,GAAG,KAAK,WAAW;AAAA,EAC9F;AACF;AAUO,SAAS,oBAAoB,SAAoC,CAAC,GAAoB;AAC3F,QAAM,UAAU,IAAI,oBAAoB;AACxC,UAAQ,gBAAgB;AAAA,IACtB,wBAAwB;AAAA,IACxB,wBAAwB;AAAA,IACxB,wBAAwB;AAAA,EAC1B;AACA,UAAQ,YAAY;AAAA,IAClB,KAAK,wBAAwB,SAAS,wBAAwB,SAAS,wBAAwB,KAAK;AAAA,EACtG;AACA,UAAQ,YAAY;AAAA,IAClB,KAAK,wBAAwB,KAAK,wBAAwB,SAAS,wBAAwB,KAAK;AAAA,IAChG,KAAK,wBAAwB,KAAK,wBAAwB,SAAS,wBAAwB,MAAM;AAAA,EACnG;AACA,UAAQ,UAAU;AAAA,IAChB,KAAK,wBAAwB,OAAO,wBAAwB,SAAS,wBAAwB,KAAK;AAAA,IAClG,KAAK,wBAAwB,OAAO,wBAAwB,SAAS,wBAAwB,KAAK;AAAA,IAClG,KAAK,wBAAwB,KAAK,wBAAwB,SAAS,wBAAwB,SAAS;AAAA,EACtG;AAEA,MAAI,OAAO,aAAa;AACtB,WAAO,YAAY,OAAO;AAAA,EAC5B;AAEA,QAAM,gBAAgB,OAAO,gBAAgB,OAAO,KAAK,CAAC;AAE1D,SAAO;AAAA,IACL,kBAAkB,KAAK,cAAc,KAAK,IAAI,YAAY,CAAC;AAAA,IAC3D,GAAG,iBAAiB,OAAO;AAAA,IAC3B,GAAG,mBAAmB,OAAO;AAAA,IAC7B,GAAG,oBAAoB,OAAO;AAAA,IAC9B,GAAG,uBAAuB,OAAO;AAAA,IACjC,GAAG,kBAAkB,OAAO;AAAA,IAC5B,GAAG,wBAAwB,OAAO;AAAA,IAClC,GAAG,yCAAyC;AAAA,IAC5C,GAAG,yBAAyB,OAAO;AAAA,IACnC,GAAG,iCAAiC,OAAO;AAAA,IAC3C,GAAG;AAAA,EACL;AACF;AAEA,SAAS,yBAAyB,SAA+C;AAC/E,SAAO,aAAa;AAAA,IAClB;AAAA;AAAA,MAEE,SAAS,CAAC,gBAAgB,WAAW;AAAA,MACrC,OAAO,QAAQ,SAAS;AAAA,MACxB,OAAO;AAAA,QACL,yDAAyD;AAAA,MAC3D;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,iBAAiB,SAA+C;AACvE,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,SAAS,CAAC,OAAO,QAAQ,WAAW;AAAA,MACpC,OAAO,QAAQ,SAAS;AAAA,MACxB,OAAO;AAAA,QACL,kBAAkB;AAAA,QAClB,yBAAyB;AAAA,QACzB,aAAa;AAAA,QACb,wBAAwB,CAAC,SAAS,UAAU,EAAE,OAAO,EAAE,eAAe,KAAK,EAAE,CAAC;AAAA,QAC9E,cAAc;AAAA,QACd,mBAAmB;AAAA,QACnB,SAAS;AAAA,QACT,gBAAgB;AAAA,QAChB,qBAAqB;AAAA,QACrB,sBAAsB;AAAA,QACtB,UAAU;AAAA,QACV,sBAAsB;AAAA,QACtB,cAAc;AAAA,QACd,cAAc;AAAA,UACZ;AAAA,UACA;AAAA,UACA;AAAA,YACE,qBAAqB;AAAA,UACvB;AAAA,QACF;AAAA,QACA,0BAA0B;AAAA,UACxB;AAAA,UACA;AAAA,QACF;AAAA,QACA,gBAAgB;AAAA,QAChB,YAAY;AAAA,QACZ,wBAAwB;AAAA,QACxB,cAAc;AAAA,QACd,aAAa;AAAA,QACb,cAAc;AAAA,UACZ;AAAA,UACA;AAAA,YACE,OAAO;AAAA,cACL;AAAA,cACA;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,QACA,yBAAyB;AAAA,QACzB,gBAAgB;AAAA,QAChB,kBAAkB;AAAA,UAChB;AAAA,UACA;AAAA,YACE,aAAa;AAAA,UACf;AAAA,QACF;AAAA,QACA,qBAAqB;AAAA,QACrB,oBAAoB;AAAA,QACpB,iBAAiB;AAAA,QACjB,kBAAkB;AAAA,QAClB,wBAAwB;AAAA,UACtB;AAAA,UACA;AAAA,YACE,OAAO;AAAA,cACL;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,QACA,mBAAmB;AAAA,QACnB,yBAAyB;AAAA,QACzB,eAAe;AAAA,QACf,gBAAgB;AAAA,QAChB,aAAa;AAAA,QACb,kBAAkB;AAAA,QAClB,gBAAgB;AAAA,QAChB,gBAAgB;AAAA,QAChB,oBAAoB;AAAA,UAClB;AAAA,UACA;AAAA,YACE,eAAe;AAAA,YACf,cAAc;AAAA,YACd,QAAQ;AAAA,cACN;AAAA,cACA;AAAA,cACA;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,QACA,mBAAmB;AAAA,QACnB,gBAAgB;AAAA,QAChB,wBAAwB;AAAA,QACxB,qBAAqB;AAAA,QACrB,eAAe;AAAA,QACf,mBAAmB;AAAA,QACnB,yBAAyB;AAAA,QACzB,mBAAmB;AAAA,QACnB,8BAA8B;AAAA,QAC9B,YAAY;AAAA,QACZ,wBAAwB;AAAA,UACtB;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,UACA;AAAA,YACE,SAAS;AAAA,YACT,UAAU;AAAA,UACZ;AAAA,QACF;AAAA,QACA,oBAAoB;AAAA,QACpB,iBAAiB;AAAA,QACjB,mBAAmB;AAAA,QACnB,gBAAgB;AAAA,QAChB,aAAa;AAAA,QACb,+BAA+B;AAAA,QAC/B,oBAAoB;AAAA,QACpB,gCAAgC;AAAA,QAChC,uBAAuB;AAAA,QACvB,uBAAuB;AAAA,QACvB,yBAAyB;AAAA,QACzB,yBAAyB;AAAA,QACzB,mBAAmB;AAAA,QACnB,2BAA2B;AAAA,QAC3B,qBAAqB;AAAA,QACrB,0BAA0B;AAAA,QAC1B,qBAAqB;AAAA,QACrB,qBAAqB;AAAA,QACrB,UAAU;AAAA,QACV,WAAW;AAAA,QACX,oBAAoB;AAAA,QACpB,uBAAuB;AAAA,QACvB,yBAAyB;AAAA,QACzB,gBAAgB;AAAA,QAChB,kCAAkC;AAAA,QAClC,8BAA8B;AAAA,QAC9B,2BAA2B;AAAA,QAC3B,yBAAyB;AAAA,QACzB,wBAAwB;AAAA,QACxB,gCAAgC;AAAA,QAChC,yBAAyB;AAAA,QACzB,sBAAsB;AAAA,QACtB,iBAAiB;AAAA,QACjB,mBAAmB;AAAA,QACnB,SAAS;AAAA,QACT,0BAA0B;AAAA,QAC1B,iBAAiB;AAAA,QACjB,sBAAsB;AAAA,QACtB,eAAe;AAAA,QACf,eAAe;AAAA,QACf,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,2CAA4D;AACnE,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,UAAU;AAAA,QACR,0BAA0B;AAAA,UACxB,+BAA+B;AAAA,YAC7B,gBAAgB;AAAA,UAClB,CAAC;AAAA,QACH;AAAA,MACF;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,kBAAkB,SAA+C;AACxE,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,SAAS;AAAA,QACP,+BAA+B;AAAA,QAC/B,+BAA+B;AAAA,QAC/B,+BAA+B;AAAA,QAC/B,+BAA+B;AAAA,MACjC;AAAA,MACA,OAAO,QAAQ,SAAS;AAAA,MACxB,OAAO;AAAA,QACL,4CAA4C;AAAA,QAC5C,uBAAuB,CAAC,SAAS,gBAAgB;AAAA,QACjD,kBAAkB;AAAA,QAClB,0BAA0B;AAAA,QAC1B,iCAAiC;AAAA,QACjC,6BAA6B;AAAA,QAC7B,mBAAmB;AAAA,QACnB,wCAAwC;AAAA,QACxC,wBAAwB;AAAA,QACxB,qBAAqB;AAAA,QACrB,8BAA8B;AAAA,QAC9B,0BAA0B;AAAA,QAC1B,0BAA0B;AAAA,QAC1B,+BAA+B;AAAA,QAC/B,kCAAkC;AAAA,QAClC,uCAAuC;AAAA,QACvC,qCAAqC;AAAA,QACrC,+BAA+B;AAAA,QAC/B,6BAA6B;AAAA,QAC7B,yBAAyB;AAAA,QACzB,8BAA8B;AAAA,QAC9B,iCAAiC;AAAA,QACjC,gCAAgC;AAAA,QAChC,2BAA2B;AAAA,QAC3B,iCAAiC;AAAA,UAC/B;AAAA,UACA;AAAA,YACE,OAAO;AAAA,cACL;AAAA,cACA;AAAA,cACA;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,QACA,8BAA8B;AAAA,QAC9B,qCAAqC;AAAA,QACrC,qCAAqC;AAAA,MACvC;AAAA,IACF;AAAA,IACA;AAAA,MACE,OAAO,QAAQ;AAAA,MACf,OAAO;AAAA,QACL,8BAA8B;AAAA,MAChC;AAAA,IACF;AAAA,IACA;AAAA,MACE,OAAO;AAAA,QACL,GAAG,QAAQ;AAAA,QACX,KAAK,wBAAwB,KAAK,wBAAwB,MAAM;AAAA,MAClE;AAAA,MACA,OAAO;AAAA,QACL,8BAA8B;AAAA,MAChC;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,iCAAiC,SAA+C;AACvF,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,OAAO,QAAQ,SAAS;AAAA,MACxB,SAAS;AAAA,QACP,sBAAsB;AAAA,MACxB;AAAA,MACA,OAAO;AAAA,QACL,gDAAgD;AAAA,MAClD;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,uBAAuB,SAA+C;AAC7E,QAAM,6BAA6B,MAAM,KAAK,WAAW,UAAU,aAAa,CAA4B;AAE5G,QAAM,mCAAmC,2BAA2B,IAAI,CAAC,WAAW;AAClF,QAAI,OAAO,OAAO,SAAS,cAAc,GAAG;AAC1C,aAAO;AAAA,IACT;AAEA,WAAO;AAAA,MACL,GAAG;AAAA,MACH,OAAO,QAAQ;AAAA,IACjB;AAAA,EACF,CAAC;AAED,SAAO,aAAa;AAAA,IAClB,GAAG;AAAA,IACH;AAAA,MACE,SAAS;AAAA,QACP;AAAA,MACF;AAAA,IACF;AAAA,IACA;AAAA,MACE,iBAAiB;AAAA,QACf,SAAS;AAAA,UACP,GAAG,QAAQ;AAAA,UACX,GAAG,QAAQ;AAAA,UACX,gBAAgB;AAAA,UAChB,cAAc;AAAA,UACd,MAAM;AAAA,UACN,aAAa;AAAA,UACb,WAAW;AAAA,UACX,UAAU;AAAA,UACV,gBAAgB;AAAA,UAChB,YAAY;AAAA,UACZ,WAAW;AAAA,UACX,gBAAgB;AAAA,UAChB,MAAM;AAAA,UACN,SAAS;AAAA,UACT,WAAW;AAAA,UACX,WAAW;AAAA,UACX,QAAQ;AAAA,UACR,OAAO;AAAA,UACP,OAAO;AAAA,QACT;AAAA,MACF;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,wBAAwB,SAA+C;AAC9E,SAAO,aAAa,CAAC;AAAA,IACnB,SAAS,CAAC,qBAAqB,0BAA0B,CAAC;AAAA,IAC1D,OAAO,QAAQ,SAAS;AAAA,EAC1B,CAAC,CAAC;AACJ;AAEA,SAAS,oBAAoB,SAA+C;AAC1E,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,SAAS;AAAA,QACP,UAAU,QAAQ;AAAA,QAClB,UAAU,QAAQ,UAAU;AAAA,UAC1B,aAAa;AAAA,UACb,YAAY;AAAA,UACZ,aAAa;AAAA,UACb,MAAM;AAAA,QACR,CAAC;AAAA,MACH;AAAA,MACA,OAAO,QAAQ,SAAS;AAAA,MACxB,OAAO;AAAA,QACL,qCAAqC;AAAA,QACrC,qBAAqB;AAAA,QACrB,gCAAgC;AAAA,QAChC,0CAA0C;AAAA,QAC1C,4BAA4B;AAAA,QAC5B,mCAAmC;AAAA,UACjC;AAAA,UACA;AAAA,YACE,mBAAmB;AAAA,cACjB,eAAe;AAAA,cACf,WAAW;AAAA,YACb;AAAA,YACA,mBAAmB;AAAA,cACjB,eAAe;AAAA,cACf,WAAW;AAAA,YACb;AAAA,UACF;AAAA,QACF;AAAA,QACA,iCAAiC;AAAA,UAC/B;AAAA,UACA;AAAA,UACA;AAAA,YACE,WAAW;AAAA,cACT,KAAK;AAAA,YACP;AAAA,UACF;AAAA,QACF;AAAA,QACA,qBAAqB;AAAA,UACnB;AAAA,UACA;AAAA,UACA;AAAA,YACE,uBAAuB;AAAA,UACzB;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAEA,SAAS,mBAAmB,SAA+C;AACzE,SAAO,aAAa;AAAA,IAClB;AAAA,MACE,SAAS;AAAA;AAAA,QAEP,GAAG,SAAS,QAAQ;AAAA;AAAA,QAEpB,GAAG,SAAS,QAAQ;AAAA,MACtB;AAAA,MACA,OAAO,QAAQ,SAAS;AAAA,MACxB,iBAAiB;AAAA,QACf,eAAe;AAAA,UACb,cAAc;AAAA,YACZ,KAAK;AAAA,UACP;AAAA,UACA,gBAAgB;AAAA,UAChB,iBAAiB,cAAc,KAAK;AAAA,QACtC;AAAA,MACF;AAAA,MACA,OAAO;AAAA,QACL,oDAAoD;AAAA,QACpD,oDAAoD;AAAA,QACpD,2CAA2C,CAAC,SAAS;AAAA,UACnD,sBAAsB;AAAA,QACxB,CAAC;AAAA,QACD,oCAAoC,CAAC,SAAS;AAAA,UAC5C,cAAc;AAAA,YACZ;AAAA,UACF;AAAA,QACF,CAAC;AAAA,QACD,qCAAqC;AAAA,UACnC;AAAA,UACA;AAAA,YACE,MAAM;AAAA,YACN,mBAAmB;AAAA,YACnB,cAAc;AAAA,YACd,2BAA2B;AAAA,YAC3B,gCAAgC;AAAA,YAChC,oBAAoB;AAAA,YACpB,mBAAmB;AAAA,UACrB;AAAA,QACF;AAAA,QACA,sCAAsC;AAAA,QACtC,gDAAgD;AAAA,MAClD;AAAA,IACF;AAAA,IACA;AAAA,MACE,UAAU;AAAA,QACR,OAAO;AAAA,UACL,SAAS;AAAA,QACX;AAAA,MACF;AAAA,IACF;AAAA,EACF,CAAC;AACH;",
  "names": []
}

|
|
546
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/script-utils/linters/eslint-config.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * ESLint configuration for TypeScript projects with various plugins.\n *\n * This module exports ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n * such as `@typescript-eslint/parser`, `@typescript-eslint/eslint-plugin`,\n * `eslint-plugin-modules-newlines`, `@stylistic/eslint-plugin`.\n * It sets up parsers, plugins, and rules for maintaining code quality and consistency.\n */\n\n/* v8 ignore start -- Declarative ESLint rule/plugin configuration; correctness is verified by running ESLint, not unit tests. */\n\nimport type {\n  ESLint,\n  Linter\n} from 'eslint';\n\n/* eslint-disable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\nimport commentsConfigs from '@eslint-community/eslint-plugin-eslint-comments/configs';\nimport { includeIgnoreFile } from '@eslint/compat';\nimport eslint from '@eslint/js';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport stylistic from '@stylistic/eslint-plugin';\nimport { createTypeScriptImportResolver } from 'eslint-import-resolver-typescript';\nimport { flatConfigs as eslintPluginImportXFlatConfigs } from 'eslint-plugin-import-x';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport obsidianmd from 'eslint-plugin-obsidianmd';\nimport { configs as perfectionistConfigs } from 'eslint-plugin-perfectionist';\nimport { defineConfig } from 'eslint/config';\nimport globals from 'globals';\nimport { existsSync } from 'node:fs';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `_default` is too confusing.\nimport tseslint from 'typescript-eslint';\n\nimport { ObsidianPluginRepoPaths } from '../../obsidian/plugin/obsidian-plugin-repo-paths.ts';\nimport { join } from '../../path.ts';\nimport { getRootFolder } from '../root.ts';\nimport { obsidianDevUtilsPlugin } from './eslint-rules/obsidian-dev-utils-plugin.ts';\n\n/**\n * The parameters for defining ESLint configurations.\n */\nexport interface DefineEslintConfigsParams {\n  /**\n   * A function that builds custom ESLint configurations.\n   *\n   * @param context - The ESLint configuration context.\n   * @returns The custom ESLint configurations.\n   */\n  customConfigs?(context: EslintConfigContext): Linter.Config[];\n\n  /**\n   * A function that edits the ESLint configuration context.\n   *\n   * @param context - The ESLint configuration context.\n   */\n  editContext?(context: EslintConfigContext): void;\n}\n\n/**\n * The context for defining ESLint configurations.\n */\nexport class EslintConfigContext {\n  /**\n   * The root configuration files.\n   */\n  public readonly rootConfigFiles: string[] = [];\n\n  /**\n   * The script files.\n   */\n  public readonly scriptFiles: string[] = [];\n\n  /**\n   * The source files.\n   */\n  public readonly sourceFiles: string[] = [];\n\n  /**\n   * The test files.\n   */\n  public readonly testFiles: string[] = [];\n\n  /**\n   * The all files.\n   *\n   * @returns The all files.\n   */\n  public allFiles(): string[] {\n    return [...this.testFiles, ...this.scriptFiles, ...this.rootConfigFiles, ...this.sourceFiles];\n  }\n}\n\n/**\n * Build ESLint configurations.\n *\n * This function builds ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n *\n * @param params - The parameters for defining ESLint configurations.\n * @returns The ESLint configurations.\n */\nexport function defineEslintConfigs(params: DefineEslintConfigsParams = {}): Linter.Config[] {\n  const context = new EslintConfigContext();\n  context.rootConfigFiles.push(\n    ObsidianPluginRepoPaths.CommitlintConfigTs,\n    ObsidianPluginRepoPaths.EslintConfigMts,\n    ObsidianPluginRepoPaths.VitestConfigTs\n  );\n  context.scriptFiles.push(\n    join(ObsidianPluginRepoPaths.Scripts, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs)\n  );\n  context.sourceFiles.push(\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTsx)\n  );\n  context.testFiles.push(\n    join(ObsidianPluginRepoPaths.Tests, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Mocks, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTestTs)\n  );\n\n  if (params.editContext) {\n    params.editContext(context);\n  }\n\n  const customConfigs = params.customConfigs?.(context) ?? [];\n\n  return defineConfig(\n    ...getGitIgnoreConfigs(),\n    ...getEslintConfigs(context),\n    ...getTseslintConfigs(context),\n    ...getStylisticConfigs(context),\n    ...getObsidianLintConfigs(context),\n    ...getImportXConfigs(context),\n    ...getPerfectionistConfigs(context),\n    ...getEslintImportResolverTypescriptConfigs(),\n    ...getEslintCommentsConfigs(context),\n    ...getObsidianDevUtilsPluginConfigs(context),\n    ...customConfigs\n  );\n}\n\nfunction getEslintCommentsConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `recommended` is too confusing.\n      extends: [commentsConfigs.recommended],\n      files: context.allFiles(),\n      rules: {\n        '@eslint-community/eslint-comments/require-description': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [eslint.configs.recommended],\n      files: context.allFiles(),\n      rules: {\n        'accessor-pairs': 'error',\n        'array-callback-return': 'error',\n        'camelcase': 'error',\n        'capitalized-comments': ['error', 'always', { block: { ignorePattern: 'v8' } }],\n        'complexity': 'error',\n        'consistent-this': 'error',\n        'curly': 'error',\n        'default-case': 'error',\n        'default-case-last': 'error',\n        'default-param-last': 'error',\n        'eqeqeq': 'error',\n        'func-name-matching': 'error',\n        'func-names': 'error',\n        'func-style': [\n          'error',\n          'declaration',\n          {\n            allowArrowFunctions: false\n          }\n        ],\n        'grouped-accessor-pairs': [\n          'error',\n          'getBeforeSet'\n        ],\n        'guard-for-in': 'error',\n        'no-alert': 'error',\n        'no-array-constructor': 'error',\n        'no-bitwise': 'error',\n        'no-caller': 'error',\n        'no-console': [\n          'error',\n          {\n            allow: [\n              'warn',\n              'error'\n            ]\n          }\n        ],\n        'no-constructor-return': 'error',\n        'no-div-regex': 'error',\n        'no-else-return': [\n          'error',\n          {\n            allowElseIf: false\n          }\n        ],\n        'no-empty-function': 'error',\n        'no-extend-native': 'error',\n        'no-extra-bind': 'error',\n        'no-extra-label': 'error',\n        'no-implicit-coercion': [\n          'error',\n          {\n            allow: [\n              '!!'\n            ]\n          }\n        ],\n        'no-implied-eval': 'error',\n        'no-inner-declarations': 'error',\n        'no-iterator': 'error',\n        'no-label-var': 'error',\n        'no-labels': 'error',\n        'no-lone-blocks': 'error',\n        'no-lonely-if': 'error',\n        'no-loop-func': 'error',\n        'no-magic-numbers': [\n          'error',\n          {\n            detectObjects: true,\n            enforceConst: true,\n            ignore: [\n              -1,\n              0,\n              1\n            ]\n          }\n        ],\n        'no-multi-assign': 'error',\n        'no-multi-str': 'error',\n        'no-negated-condition': 'error',\n        'no-nested-ternary': 'error',\n        'no-new-func': 'error',\n        'no-new-wrappers': 'error',\n        'no-object-constructor': 'error',\n        'no-octal-escape': 'error',\n        'no-promise-executor-return': 'error',\n        'no-proto': 'error',\n        'no-restricted-syntax': [\n          'error',\n          {\n            message: 'Do not use definite assignment assertions (!). Initialize the field or make it optional.',\n            selector: 'PropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use definite assignment assertions (!) on abstract fields.',\n            selector: 'TSAbstractPropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function parameters. Define a named interface instead.',\n            selector: ':function > Identifier TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function return types. Define a named interface instead.',\n            selector: ':function > TSTypeAnnotation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in interface/method signatures. Define a named interface instead.',\n            selector: 'TSMethodSignature TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types as type arguments. Define a named interface instead.',\n            selector: 'TSTypeParameterInstantiation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in type annotations. Define a named interface instead.',\n            selector: 'TSTypeAnnotation TSTypeLiteral'\n          }\n        ],\n        'no-return-assign': 'error',\n        'no-script-url': 'error',\n        'no-self-compare': 'error',\n        'no-sequences': 'error',\n        'no-shadow': 'error',\n        'no-template-curly-in-string': 'error',\n        'no-throw-literal': 'error',\n        'no-unmodified-loop-condition': 'error',\n        'no-unneeded-ternary': 'error',\n        'no-unreachable-loop': 'error',\n        'no-unused-expressions': 'error',\n        'no-useless-assignment': 'error',\n        'no-useless-call': 'error',\n        'no-useless-computed-key': 'error',\n        'no-useless-concat': 'error',\n        'no-useless-constructor': 'error',\n        'no-useless-rename': 'error',\n        'no-useless-return': 'error',\n        'no-var': 'error',\n        'no-void': 'error',\n        'object-shorthand': 'error',\n        'operator-assignment': 'error',\n        'prefer-arrow-callback': 'error',\n        'prefer-const': 'error',\n        'prefer-exponentiation-operator': 'error',\n        'prefer-named-capture-group': 'error',\n        'prefer-numeric-literals': 'error',\n        'prefer-object-has-own': 'error',\n        'prefer-object-spread': 'error',\n        'prefer-promise-reject-errors': 'error',\n        'prefer-regex-literals': 'error',\n        'prefer-rest-params': 'error',\n        'prefer-spread': 'error',\n        'prefer-template': 'error',\n        'radix': 'error',\n        'require-atomic-updates': 'error',\n        'require-await': 'error',\n        'symbol-description': 'error',\n        'unicode-bom': 'error',\n        'vars-on-top': 'error',\n        'yoda': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintImportResolverTypescriptConfigs(): Linter.Config[] {\n  return defineConfig([\n    {\n      settings: {\n        'import-x/resolver-next': [\n          createTypeScriptImportResolver({\n            alwaysTryTypes: true\n          })\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getGitIgnoreConfigs(): Linter.Config[] {\n  const gitignorePath = join(getRootFolder() ?? '', '.gitignore');\n  if (!existsSync(gitignorePath)) {\n    return [];\n  }\n  return [includeIgnoreFile(gitignorePath)];\n}\n\nfunction getImportXConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        eslintPluginImportXFlatConfigs.recommended as Linter.Config,\n        eslintPluginImportXFlatConfigs.typescript as Linter.Config,\n        eslintPluginImportXFlatConfigs.errors as Linter.Config,\n        eslintPluginImportXFlatConfigs.warnings as Linter.Config\n      ],\n      files: context.allFiles(),\n      rules: {\n        'import-x/consistent-type-specifier-style': 'error',\n        'import-x/extensions': ['error', 'ignorePackages'],\n        'import-x/first': 'error',\n        'import-x/imports-first': 'error',\n        'import-x/newline-after-import': 'error',\n        'import-x/no-absolute-path': 'error',\n        'import-x/no-amd': 'error',\n        'import-x/no-anonymous-default-export': 'error',\n        'import-x/no-commonjs': 'error',\n        'import-x/no-cycle': 'error',\n        'import-x/no-default-export': 'error',\n        'import-x/no-deprecated': 'error',\n        'import-x/no-duplicates': 'error',\n        'import-x/no-dynamic-require': 'error',\n        'import-x/no-empty-named-blocks': 'error',\n        'import-x/no-extraneous-dependencies': 'error',\n        'import-x/no-import-module-exports': 'error',\n        'import-x/no-mutable-exports': 'error',\n        'import-x/no-named-default': 'error',\n        'import-x/no-namespace': 'error',\n        'import-x/no-nodejs-modules': 'error',\n        'import-x/no-relative-packages': 'error',\n        'import-x/no-restricted-paths': 'error',\n        'import-x/no-self-import': 'error',\n        'import-x/no-unassigned-import': [\n          'error',\n          {\n            allow: [\n              '**/*.css',\n              '**/*.sass',\n              '**/*.scss'\n            ]\n          }\n        ],\n        'import-x/no-unused-modules': 'off',\n        'import-x/no-useless-path-segments': 'error',\n        'import-x/no-webpack-loader-syntax': 'error'\n      }\n    },\n    {\n      files: context.scriptFiles,\n      rules: {\n        'import-x/no-nodejs-modules': 'off'\n      }\n    },\n    {\n      files: [\n        ...context.rootConfigFiles,\n        join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.MainTs)\n      ],\n      rules: {\n        'import-x/no-default-export': 'off'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianDevUtilsPluginConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      files: context.allFiles(),\n      plugins: {\n        'obsidian-dev-utils': obsidianDevUtilsPlugin\n      },\n      rules: {\n        'obsidian-dev-utils/no-used-underscore-variables': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianLintConfigs(context: EslintConfigContext): Linter.Config[] {\n  const obsidianRecommendedConfigs = Array.from(obsidianmd.configs?.['recommended'] as Iterable<Linter.Config>);\n\n  const scopedObsidianRecommendedConfigs = obsidianRecommendedConfigs.map((config) => {\n    if (config.files?.includes('package.json')) {\n      return config;\n    }\n\n    return {\n      ...config,\n      files: context.sourceFiles\n    };\n  });\n\n  return defineConfig([\n    ...scopedObsidianRecommendedConfigs,\n    {\n      plugins: {\n        obsidianmd: obsidianmd as ESLint.Plugin\n      }\n    },\n    {\n      languageOptions: {\n        globals: {\n          ...globals.browser,\n          ...globals.node,\n          activeDocument: 'readonly',\n          activeWindow: 'readonly',\n          ajax: 'readonly',\n          ajaxPromise: 'readonly',\n          createDiv: 'readonly',\n          createEl: 'readonly',\n          createFragment: 'readonly',\n          createSpan: 'readonly',\n          createSvg: 'readonly',\n          DomElementInfo: 'readonly',\n          fish: 'readonly',\n          fishAll: 'readonly',\n          isBoolean: 'readonly',\n          nextFrame: 'readonly',\n          NodeJS: 'readonly',\n          ready: 'readonly',\n          sleep: 'readonly'\n        }\n      }\n    }\n  ]);\n}\n\nfunction getPerfectionistConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([{\n    extends: [perfectionistConfigs['recommended-alphabetical']],\n    files: context.allFiles()\n  }]);\n}\n\nfunction getStylisticConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        stylistic.configs.recommended,\n        stylistic.configs.customize({\n          arrowParens: true,\n          braceStyle: '1tbs',\n          commaDangle: 'never',\n          semi: true\n        })\n      ],\n      files: context.allFiles(),\n      rules: {\n        '@stylistic/generator-star-spacing': 'off',\n        '@stylistic/indent': 'off',\n        '@stylistic/indent-binary-ops': 'off',\n        '@stylistic/jsx-one-expression-per-line': 'off',\n        '@stylistic/no-extra-semi': 'error',\n        '@stylistic/object-curly-newline': [\n          'error',\n          {\n            ExportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            },\n            ImportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            }\n          }\n        ],\n        '@stylistic/operator-linebreak': [\n          'error',\n          'before',\n          {\n            overrides: {\n              '=': 'after'\n            }\n          }\n        ],\n        '@stylistic/quotes': [\n          'error',\n          'single',\n          {\n            allowTemplateLiterals: 'never'\n          }\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getTseslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.strictTypeChecked,\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.stylisticTypeChecked\n      ],\n      files: context.allFiles(),\n      languageOptions: {\n        parserOptions: {\n          ecmaFeatures: {\n            jsx: true\n          },\n          projectService: true,\n          tsconfigRootDir: getRootFolder() ?? ''\n        }\n      },\n      rules: {\n        '@typescript-eslint/explicit-function-return-type': 'error',\n        '@typescript-eslint/explicit-member-accessibility': 'error',\n        '@typescript-eslint/no-invalid-void-type': ['error', {\n          allowAsThisParameter: true\n        }],\n        '@typescript-eslint/no-this-alias': ['error', {\n          allowedNames: [\n            'that'\n          ]\n        }],\n        '@typescript-eslint/no-unused-vars': [\n          'error',\n          {\n            args: 'all',\n            argsIgnorePattern: '^_',\n            caughtErrors: 'all',\n            caughtErrorsIgnorePattern: '^_',\n            destructuredArrayIgnorePattern: '^_',\n            ignoreRestSiblings: true,\n            varsIgnorePattern: '^_'\n          }\n        ],\n        '@typescript-eslint/prefer-readonly': 'error',\n        'obsidian-dev-utils/no-async-callback-to-unsafe-return': 'error',\n        'obsidian-dev-utils/no-used-underscore-variables': 'error'\n      }\n    },\n    {\n      settings: {\n        react: {\n          version: 'detect'\n        }\n      }\n    }\n  ]);\n}\n\n/* eslint-enable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -1,2 +1,4 @@
|
|
|
1
|
-
export * as
|
|
1
|
+
export * as no_async_callback_to_unsafe_return from './no-async-callback-to-unsafe-return.mjs';
|
|
2
|
+
export * as no_used_underscore_variables from './no-used-underscore-variables.mjs';
|
|
2
3
|
export * as obsidian_dev_utils_plugin from './obsidian-dev-utils-plugin.mjs';
|
|
4
|
+
export * as rule_tester_helper from './rule-tester-helper.mjs';
|
|
@@ -19,10 +19,14 @@ if you want to view the source, please visit the github repository of this plugi
|
|
|
19
19
|
globalThis.process = browserProcess;
|
|
20
20
|
})();
|
|
21
21
|
|
|
22
|
-
import * as
|
|
22
|
+
import * as no_async_callback_to_unsafe_return from "./no-async-callback-to-unsafe-return.mjs";
|
|
23
|
+
import * as no_used_underscore_variables from "./no-used-underscore-variables.mjs";
|
|
23
24
|
import * as obsidian_dev_utils_plugin from "./obsidian-dev-utils-plugin.mjs";
|
|
25
|
+
import * as rule_tester_helper from "./rule-tester-helper.mjs";
|
|
24
26
|
export {
|
|
25
|
-
|
|
26
|
-
|
|
27
|
+
no_async_callback_to_unsafe_return,
|
|
28
|
+
no_used_underscore_variables,
|
|
29
|
+
obsidian_dev_utils_plugin,
|
|
30
|
+
rule_tester_helper
|
|
27
31
|
};
|
|
28
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL3NjcmlwdC11dGlscy9saW50ZXJzL2VzbGludC1ydWxlcy9pbmRleC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyogVEhJUyBJUyBBIEdFTkVSQVRFRC9CVU5ETEVEIEZJTEUgQlkgQlVJTEQgU0NSSVBUICovXG5cbmV4cG9ydCAqIGFzIG5vX2FzeW5jX2NhbGxiYWNrX3RvX3Vuc2FmZV9yZXR1cm4gZnJvbSAnLi9uby1hc3luYy1jYWxsYmFjay10by11bnNhZmUtcmV0dXJuLnRzJztcbmV4cG9ydCAqIGFzIG5vX3VzZWRfdW5kZXJzY29yZV92YXJpYWJsZXMgZnJvbSAnLi9uby11c2VkLXVuZGVyc2NvcmUtdmFyaWFibGVzLnRzJztcbmV4cG9ydCAqIGFzIG9ic2lkaWFuX2Rldl91dGlsc19wbHVnaW4gZnJvbSAnLi9vYnNpZGlhbi1kZXYtdXRpbHMtcGx1Z2luLnRzJztcbmV4cG9ydCAqIGFzIHJ1bGVfdGVzdGVyX2hlbHBlciBmcm9tICcuL3J1bGUtdGVzdGVyLWhlbHBlci50cyc7XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQSxZQUFZLHdDQUF3QztBQUNwRCxZQUFZLGtDQUFrQztBQUM5QyxZQUFZLCtCQUErQjtBQUMzQyxZQUFZLHdCQUF3QjsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
/*
|
|
2
|
+
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
|
3
|
+
if you want to view the source, please visit the github repository of this plugin
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
(function initEsm() {
|
|
7
|
+
if (globalThis.process) {
|
|
8
|
+
return;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const browserProcess = {
|
|
12
|
+
browser: true,
|
|
13
|
+
cwd() {
|
|
14
|
+
return '/';
|
|
15
|
+
},
|
|
16
|
+
env: {},
|
|
17
|
+
platform: 'android'
|
|
18
|
+
};
|
|
19
|
+
globalThis.process = browserProcess;
|
|
20
|
+
})();
|
|
21
|
+
|
|
22
|
+
import {
|
|
23
|
+
isIdentifier,
|
|
24
|
+
isTypeAliasDeclaration,
|
|
25
|
+
isTypeReferenceNode,
|
|
26
|
+
isUnionTypeNode,
|
|
27
|
+
SymbolFlags,
|
|
28
|
+
TypeFlags
|
|
29
|
+
} from "typescript";
|
|
30
|
+
import { assertNonNullable } from "../../../type-guards.mjs";
|
|
31
|
+
const MESSAGE_ID = "noAsyncCallbackToUnsafeReturn";
|
|
32
|
+
function hasUnsafeReturnCallSignature(checker, type) {
|
|
33
|
+
return type.getCallSignatures().some((sig) => isUnsafeReturnSignature(checker, sig));
|
|
34
|
+
}
|
|
35
|
+
function isAsyncFunctionNode(node) {
|
|
36
|
+
return (node.type === "ArrowFunctionExpression" || node.type === "FunctionExpression") && node.async;
|
|
37
|
+
}
|
|
38
|
+
const UNSAFE_RETURN_FLAGS = TypeFlags.Any | TypeFlags.Unknown;
|
|
39
|
+
const PROMISE_TYPE_NAMES = /* @__PURE__ */ new Set(["Promise", "PromiseLike"]);
|
|
40
|
+
function containsPromiseReference(checker, node) {
|
|
41
|
+
if (isUnionTypeNode(node)) {
|
|
42
|
+
return node.types.some((member) => containsPromiseReference(checker, member));
|
|
43
|
+
}
|
|
44
|
+
if (isTypeReferenceNode(node)) {
|
|
45
|
+
const name = isIdentifier(node.typeName) ? node.typeName.text : "";
|
|
46
|
+
if (PROMISE_TYPE_NAMES.has(name)) {
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
let symbol = checker.getSymbolAtLocation(node.typeName);
|
|
50
|
+
if (symbol && symbol.flags & SymbolFlags.Alias) {
|
|
51
|
+
symbol = checker.getAliasedSymbol(symbol);
|
|
52
|
+
}
|
|
53
|
+
const decl = symbol?.declarations?.[0];
|
|
54
|
+
if (decl && isTypeAliasDeclaration(decl)) {
|
|
55
|
+
return containsPromiseReference(checker, decl.type);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
return false;
|
|
59
|
+
}
|
|
60
|
+
function isUnsafeReturnSignature(checker, sig) {
|
|
61
|
+
const returnType = checker.getReturnTypeOfSignature(sig);
|
|
62
|
+
if (!(returnType.flags & UNSAFE_RETURN_FLAGS)) {
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
const decl = sig.getDeclaration();
|
|
66
|
+
const returnTypeNode = decl.type;
|
|
67
|
+
assertNonNullable(returnTypeNode, "Signature declarations with any/unknown return always have a return type annotation");
|
|
68
|
+
return !containsPromiseReference(checker, returnTypeNode);
|
|
69
|
+
}
|
|
70
|
+
const noAsyncCallbackToUnsafeReturn = {
|
|
71
|
+
create(context) {
|
|
72
|
+
const services = context.sourceCode.parserServices;
|
|
73
|
+
const checker = services.program.getTypeChecker();
|
|
74
|
+
return {
|
|
75
|
+
CallExpression(node) {
|
|
76
|
+
const callNode = node;
|
|
77
|
+
for (let i = 0; i < callNode.arguments.length; i++) {
|
|
78
|
+
const arg = callNode.arguments[i];
|
|
79
|
+
if (!arg || !isAsyncFunctionNode(arg)) {
|
|
80
|
+
continue;
|
|
81
|
+
}
|
|
82
|
+
const tsCalleeNode = services.esTreeNodeToTSNodeMap.get(callNode.callee);
|
|
83
|
+
const calleeType = checker.getTypeAtLocation(tsCalleeNode);
|
|
84
|
+
for (const sig of calleeType.getCallSignatures()) {
|
|
85
|
+
const param = sig.getParameters()[i];
|
|
86
|
+
if (!param) {
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
89
|
+
const paramType = checker.getTypeOfSymbol(param);
|
|
90
|
+
if (hasUnsafeReturnCallSignature(checker, paramType)) {
|
|
91
|
+
context.report({
|
|
92
|
+
messageId: MESSAGE_ID,
|
|
93
|
+
node: arg
|
|
94
|
+
});
|
|
95
|
+
break;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
},
|
|
102
|
+
meta: {
|
|
103
|
+
docs: {
|
|
104
|
+
description: "Disallow passing async functions as callbacks to parameters with `any` or `unknown` return type"
|
|
105
|
+
},
|
|
106
|
+
messages: {
|
|
107
|
+
[MESSAGE_ID]: "Async function passed as callback to a parameter with `any`/`unknown` return type. This may cause unhandled promise rejections. Wrap the call: `(...args) => { yourAsyncFn(...args); }`."
|
|
108
|
+
},
|
|
109
|
+
schema: [],
|
|
110
|
+
type: "problem"
|
|
111
|
+
}
|
|
112
|
+
};
|
|
113
|
+
export {
|
|
114
|
+
MESSAGE_ID,
|
|
115
|
+
noAsyncCallbackToUnsafeReturn
|
|
116
|
+
};
|
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * ESLint rule: no-async-callback-to-unsafe-return\n *\n * Reports an error when an async function is passed as a callback argument\n * to a function parameter whose return type is `any` or `unknown`. Because\n * these types silently accept `Promise<T>`, the caller won't await the\n * result, leading to unhandled promise rejections at runtime.\n *\n * Example:\n * ```typescript\n * // onClick(fn: (evt: MouseEvent) => any)\n * button.onClick(async () => {\n *   await someFn(); // Unhandled rejection if someFn throws\n * });\n * ```\n *\n * `@typescript-eslint/no-misused-promises` only catches `=> void` callbacks.\n * This rule closes the gap for `=> any` and `=> unknown` callbacks.\n */\nimport type {\n  ParserServicesWithTypeInformation,\n  TSESTree\n} from '@typescript-eslint/utils';\nimport type { Rule } from 'eslint';\nimport type {\n  Signature,\n  Type,\n  TypeChecker,\n  TypeNode\n} from 'typescript';\n\nimport {\n  isIdentifier,\n  isTypeAliasDeclaration,\n  isTypeReferenceNode,\n  isUnionTypeNode,\n  SymbolFlags,\n  TypeFlags\n} from 'typescript';\n\nimport { assertNonNullable } from '../../../type-guards.ts';\n\nexport const MESSAGE_ID = 'noAsyncCallbackToUnsafeReturn';\n\n/**\n * Checks whether the given TypeScript type is a function type with an unsafe\n * return type for async callbacks.\n *\n * @param checker - TypeScript type checker.\n * @param type - The type to inspect.\n * @returns `true` if the type has at least one call signature with an unsafe return.\n */\nfunction hasUnsafeReturnCallSignature(checker: TypeChecker, type: Type): boolean {\n  return type.getCallSignatures().some((sig) => isUnsafeReturnSignature(checker, sig));\n}\n\n/**\n * Checks whether a node is an async function expression (arrow or regular).\n *\n * @param node - AST node to check.\n * @returns `true` if the node is an async arrow or function expression.\n */\nfunction isAsyncFunctionNode(node: TSESTree.Node): node is TSESTree.ArrowFunctionExpression | TSESTree.FunctionExpression {\n  // eslint-disable-next-line @typescript-eslint/no-unsafe-enum-comparison -- AST node type string literals match the TSESTree enum values.\n  return (node.type === 'ArrowFunctionExpression' || node.type === 'FunctionExpression') && node.async;\n}\n\n// eslint-disable-next-line no-bitwise -- Bitwise flag mask is idiomatic for TypeScript compiler API.\nconst UNSAFE_RETURN_FLAGS = TypeFlags.Any | TypeFlags.Unknown;\n\nconst PROMISE_TYPE_NAMES = new Set(['Promise', 'PromiseLike']);\n\n/**\n * Checks whether a type node contains a reference to `Promise` or `PromiseLike`,\n * either directly or through a type alias. This indicates the caller explicitly\n * accounts for promise returns.\n *\n * @param checker - TypeScript type checker.\n * @param node - The type node to inspect.\n * @returns `true` if the node references `Promise` or `PromiseLike`.\n */\nfunction containsPromiseReference(checker: TypeChecker, node: TypeNode): boolean {\n  if (isUnionTypeNode(node)) {\n    return node.types.some((member) => containsPromiseReference(checker, member));\n  }\n\n  if (isTypeReferenceNode(node)) {\n    /* v8 ignore start -- Qualified names (e.g., ns.Promise) and import aliases require multi-file setup not available in RuleTester. */\n    const name = isIdentifier(node.typeName) ? node.typeName.text : '';\n    /* v8 ignore stop */\n    if (PROMISE_TYPE_NAMES.has(name)) {\n      return true;\n    }\n\n    // Resolve type alias (following imports) and check its definition body\n    let symbol = checker.getSymbolAtLocation(node.typeName);\n    /* v8 ignore start -- Import aliases require multi-file setup not available in RuleTester. */\n    // eslint-disable-next-line no-bitwise -- Bitwise flag check is idiomatic for TypeScript compiler API.\n    if (symbol && symbol.flags & SymbolFlags.Alias) {\n      symbol = checker.getAliasedSymbol(symbol);\n    }\n    /* v8 ignore stop */\n    const decl = symbol?.declarations?.[0];\n    if (decl && isTypeAliasDeclaration(decl)) {\n      return containsPromiseReference(checker, decl.type);\n    }\n  }\n\n  return false;\n}\n\n/**\n * Checks whether a call signature has an unsafe return type (`any` or `unknown`)\n * without explicitly accounting for promises.\n *\n * Flags bare `any`, `unknown`, and unions like `any | string` that resolve to\n * `any`/`unknown` without including `Promise`/`PromiseLike`. Does NOT flag type\n * aliases like `Awaitable<T>` or unions like `any | Promise<any>` that explicitly\n * handle promise returns.\n *\n * @param checker - TypeScript type checker.\n * @param sig - The call signature to inspect.\n * @returns `true` if the return type is unsafe for async callbacks.\n */\nfunction isUnsafeReturnSignature(checker: TypeChecker, sig: Signature): boolean {\n  const returnType = checker.getReturnTypeOfSignature(sig);\n\n  // eslint-disable-next-line no-bitwise -- Bitwise flag check is idiomatic for TypeScript compiler API.\n  if (!(returnType.flags & UNSAFE_RETURN_FLAGS)) {\n    return false;\n  }\n\n  // Check the syntactic return type annotation. If it contains a reference to\n  // Promise/PromiseLike (directly or via a type alias), the caller explicitly\n  // Handles async returns and should not be flagged.\n  const decl = sig.getDeclaration();\n  const returnTypeNode = decl.type;\n  assertNonNullable(returnTypeNode, 'Signature declarations with any/unknown return always have a return type annotation');\n\n  return !containsPromiseReference(checker, returnTypeNode);\n}\n\nexport const noAsyncCallbackToUnsafeReturn: Rule.RuleModule = {\n  create(context) {\n    const services = context.sourceCode.parserServices as ParserServicesWithTypeInformation;\n    const checker = services.program.getTypeChecker();\n\n    return {\n      CallExpression(node: Rule.Node): void {\n        const callNode = node as TSESTree.CallExpression;\n\n        for (let i = 0; i < callNode.arguments.length; i++) {\n          const arg = callNode.arguments[i];\n          if (!arg || !isAsyncFunctionNode(arg)) {\n            continue;\n          }\n\n          const tsCalleeNode = services.esTreeNodeToTSNodeMap.get(callNode.callee);\n          const calleeType = checker.getTypeAtLocation(tsCalleeNode);\n\n          for (const sig of calleeType.getCallSignatures()) {\n            const param = sig.getParameters()[i];\n            if (!param) {\n              continue;\n            }\n\n            const paramType = checker.getTypeOfSymbol(param);\n\n            if (hasUnsafeReturnCallSignature(checker, paramType)) {\n              context.report({\n                messageId: MESSAGE_ID,\n                node: arg as Rule.Node\n              });\n              break;\n            }\n          }\n        }\n      }\n    };\n  },\n  meta: {\n    docs: {\n      description: 'Disallow passing async functions as callbacks to parameters with `any` or `unknown` return type'\n    },\n    messages: {\n      [MESSAGE_ID]:\n        'Async function passed as callback to a parameter with `any`/`unknown` return type. This may cause unhandled promise rejections. Wrap the call: `(...args) => { yourAsyncFn(...args); }`.'\n    },\n    schema: [],\n    type: 'problem'\n  }\n};\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAiCA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAEP,SAAS,yBAAyB;AAE3B,MAAM,aAAa;AAU1B,SAAS,6BAA6B,SAAsB,MAAqB;AAC/E,SAAO,KAAK,kBAAkB,EAAE,KAAK,CAAC,QAAQ,wBAAwB,SAAS,GAAG,CAAC;AACrF;AAQA,SAAS,oBAAoB,MAA6F;AAExH,UAAQ,KAAK,SAAS,6BAA6B,KAAK,SAAS,yBAAyB,KAAK;AACjG;AAGA,MAAM,sBAAsB,UAAU,MAAM,UAAU;AAEtD,MAAM,qBAAqB,oBAAI,IAAI,CAAC,WAAW,aAAa,CAAC;AAW7D,SAAS,yBAAyB,SAAsB,MAAyB;AAC/E,MAAI,gBAAgB,IAAI,GAAG;AACzB,WAAO,KAAK,MAAM,KAAK,CAAC,WAAW,yBAAyB,SAAS,MAAM,CAAC;AAAA,EAC9E;AAEA,MAAI,oBAAoB,IAAI,GAAG;AAE7B,UAAM,OAAO,aAAa,KAAK,QAAQ,IAAI,KAAK,SAAS,OAAO;AAEhE,QAAI,mBAAmB,IAAI,IAAI,GAAG;AAChC,aAAO;AAAA,IACT;AAGA,QAAI,SAAS,QAAQ,oBAAoB,KAAK,QAAQ;AAGtD,QAAI,UAAU,OAAO,QAAQ,YAAY,OAAO;AAC9C,eAAS,QAAQ,iBAAiB,MAAM;AAAA,IAC1C;AAEA,UAAM,OAAO,QAAQ,eAAe,CAAC;AACrC,QAAI,QAAQ,uBAAuB,IAAI,GAAG;AACxC,aAAO,yBAAyB,SAAS,KAAK,IAAI;AAAA,IACpD;AAAA,EACF;AAEA,SAAO;AACT;AAeA,SAAS,wBAAwB,SAAsB,KAAyB;AAC9E,QAAM,aAAa,QAAQ,yBAAyB,GAAG;AAGvD,MAAI,EAAE,WAAW,QAAQ,sBAAsB;AAC7C,WAAO;AAAA,EACT;AAKA,QAAM,OAAO,IAAI,eAAe;AAChC,QAAM,iBAAiB,KAAK;AAC5B,oBAAkB,gBAAgB,qFAAqF;AAEvH,SAAO,CAAC,yBAAyB,SAAS,cAAc;AAC1D;AAEO,MAAM,gCAAiD;AAAA,EAC5D,OAAO,SAAS;AACd,UAAM,WAAW,QAAQ,WAAW;AACpC,UAAM,UAAU,SAAS,QAAQ,eAAe;AAEhD,WAAO;AAAA,MACL,eAAe,MAAuB;AACpC,cAAM,WAAW;AAEjB,iBAAS,IAAI,GAAG,IAAI,SAAS,UAAU,QAAQ,KAAK;AAClD,gBAAM,MAAM,SAAS,UAAU,CAAC;AAChC,cAAI,CAAC,OAAO,CAAC,oBAAoB,GAAG,GAAG;AACrC;AAAA,UACF;AAEA,gBAAM,eAAe,SAAS,sBAAsB,IAAI,SAAS,MAAM;AACvE,gBAAM,aAAa,QAAQ,kBAAkB,YAAY;AAEzD,qBAAW,OAAO,WAAW,kBAAkB,GAAG;AAChD,kBAAM,QAAQ,IAAI,cAAc,EAAE,CAAC;AACnC,gBAAI,CAAC,OAAO;AACV;AAAA,YACF;AAEA,kBAAM,YAAY,QAAQ,gBAAgB,KAAK;AAE/C,gBAAI,6BAA6B,SAAS,SAAS,GAAG;AACpD,sBAAQ,OAAO;AAAA,gBACb,WAAW;AAAA,gBACX,MAAM;AAAA,cACR,CAAC;AACD;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM;AAAA,IACJ,MAAM;AAAA,MACJ,aAAa;AAAA,IACf;AAAA,IACA,UAAU;AAAA,MACR,CAAC,UAAU,GACT;AAAA,IACJ;AAAA,IACA,QAAQ,CAAC;AAAA,IACT,MAAM;AAAA,EACR;AACF;",
  "names": []
}

|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @packageDocumentation
|
|
3
|
+
*
|
|
4
|
+
* ESLint rule: no-used-underscore-variables
|
|
5
|
+
*
|
|
6
|
+
* Reports an error when a parameter or local variable with a `_` prefix is
|
|
7
|
+
* actually referenced in the function body. The `_` prefix convention signals
|
|
8
|
+
* "this identifier is intentionally unused" — if it IS used, the prefix is
|
|
9
|
+
* misleading and should be removed.
|
|
10
|
+
*/
|
|
11
|
+
import type { Rule } from 'eslint';
|
|
12
|
+
export declare const MESSAGE_ID = "noUsedUnderscoreVariables";
|
|
13
|
+
export declare const noUsedUnderscoreVariables: Rule.RuleModule;
|