UNPKG

monaco-editor-core

Version:

A browser based code editor

390 lines (389 loc) • 20.7 kB
/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { distinct } from '../../../base/common/arrays.js'; import { Emitter } from '../../../base/common/event.js'; import * as types from '../../../base/common/types.js'; import * as nls from '../../../nls.js'; import { getLanguageTagSettingPlainKey } from './configuration.js'; import { Extensions as JSONExtensions } from '../../jsonschemas/common/jsonContributionRegistry.js'; import { Registry } from '../../registry/common/platform.js'; export const Extensions = { Configuration: 'base.contributions.configuration' }; export const allSettings = { properties: {}, patternProperties: {} }; export const applicationSettings = { properties: {}, patternProperties: {} }; export const machineSettings = { properties: {}, patternProperties: {} }; export const machineOverridableSettings = { properties: {}, patternProperties: {} }; export const windowSettings = { properties: {}, patternProperties: {} }; export const resourceSettings = { properties: {}, patternProperties: {} }; export const resourceLanguageSettingsSchemaId = 'vscode://schemas/settings/resourceLanguage'; const contributionRegistry = Registry.as(JSONExtensions.JSONContribution); class ConfigurationRegistry { constructor() { this.registeredConfigurationDefaults = []; this.overrideIdentifiers = new Set(); this._onDidSchemaChange = new Emitter(); this._onDidUpdateConfiguration = new Emitter(); this.configurationDefaultsOverrides = new Map(); this.defaultLanguageConfigurationOverridesNode = { id: 'defaultOverrides', title: nls.localize('defaultLanguageConfigurationOverrides.title', "Default Language Configuration Overrides"), properties: {} }; this.configurationContributors = [this.defaultLanguageConfigurationOverridesNode]; this.resourceLanguageSettingsSchema = { properties: {}, patternProperties: {}, additionalProperties: true, allowTrailingCommas: true, allowComments: true }; this.configurationProperties = {}; this.policyConfigurations = new Map(); this.excludedConfigurationProperties = {}; contributionRegistry.registerSchema(resourceLanguageSettingsSchemaId, this.resourceLanguageSettingsSchema); this.registerOverridePropertyPatternKey(); } registerConfiguration(configuration, validate = true) { this.registerConfigurations([configuration], validate); } registerConfigurations(configurations, validate = true) { const properties = new Set(); this.doRegisterConfigurations(configurations, validate, properties); contributionRegistry.registerSchema(resourceLanguageSettingsSchemaId, this.resourceLanguageSettingsSchema); this._onDidSchemaChange.fire(); this._onDidUpdateConfiguration.fire({ properties }); } registerDefaultConfigurations(configurationDefaults) { const properties = new Set(); this.doRegisterDefaultConfigurations(configurationDefaults, properties); this._onDidSchemaChange.fire(); this._onDidUpdateConfiguration.fire({ properties, defaultsOverrides: true }); } doRegisterDefaultConfigurations(configurationDefaults, bucket) { this.registeredConfigurationDefaults.push(...configurationDefaults); const overrideIdentifiers = []; for (const { overrides, source } of configurationDefaults) { for (const key in overrides) { bucket.add(key); const configurationDefaultOverridesForKey = this.configurationDefaultsOverrides.get(key) ?? this.configurationDefaultsOverrides.set(key, { configurationDefaultOverrides: [] }).get(key); const value = overrides[key]; configurationDefaultOverridesForKey.configurationDefaultOverrides.push({ value, source }); // Configuration defaults for Override Identifiers if (OVERRIDE_PROPERTY_REGEX.test(key)) { const newDefaultOverride = this.mergeDefaultConfigurationsForOverrideIdentifier(key, value, source, configurationDefaultOverridesForKey.configurationDefaultOverrideValue); if (!newDefaultOverride) { continue; } configurationDefaultOverridesForKey.configurationDefaultOverrideValue = newDefaultOverride; this.updateDefaultOverrideProperty(key, newDefaultOverride, source); overrideIdentifiers.push(...overrideIdentifiersFromKey(key)); } // Configuration defaults for Configuration Properties else { const newDefaultOverride = this.mergeDefaultConfigurationsForConfigurationProperty(key, value, source, configurationDefaultOverridesForKey.configurationDefaultOverrideValue); if (!newDefaultOverride) { continue; } configurationDefaultOverridesForKey.configurationDefaultOverrideValue = newDefaultOverride; const property = this.configurationProperties[key]; if (property) { this.updatePropertyDefaultValue(key, property); this.updateSchema(key, property); } } } } this.doRegisterOverrideIdentifiers(overrideIdentifiers); } updateDefaultOverrideProperty(key, newDefaultOverride, source) { const property = { type: 'object', default: newDefaultOverride.value, description: nls.localize('defaultLanguageConfiguration.description', "Configure settings to be overridden for the {0} language.", getLanguageTagSettingPlainKey(key)), $ref: resourceLanguageSettingsSchemaId, defaultDefaultValue: newDefaultOverride.value, source, defaultValueSource: source }; this.configurationProperties[key] = property; this.defaultLanguageConfigurationOverridesNode.properties[key] = property; } mergeDefaultConfigurationsForOverrideIdentifier(overrideIdentifier, configurationValueObject, valueSource, existingDefaultOverride) { const defaultValue = existingDefaultOverride?.value || {}; const source = existingDefaultOverride?.source ?? new Map(); // This should not happen if (!(source instanceof Map)) { console.error('objectConfigurationSources is not a Map'); return undefined; } for (const propertyKey of Object.keys(configurationValueObject)) { const propertyDefaultValue = configurationValueObject[propertyKey]; const isObjectSetting = types.isObject(propertyDefaultValue) && (types.isUndefined(defaultValue[propertyKey]) || types.isObject(defaultValue[propertyKey])); // If the default value is an object, merge the objects and store the source of each keys if (isObjectSetting) { defaultValue[propertyKey] = { ...(defaultValue[propertyKey] ?? {}), ...propertyDefaultValue }; // Track the source of each value in the object if (valueSource) { for (const objectKey in propertyDefaultValue) { source.set(`${propertyKey}.${objectKey}`, valueSource); } } } // Primitive values are overridden else { defaultValue[propertyKey] = propertyDefaultValue; if (valueSource) { source.set(propertyKey, valueSource); } else { source.delete(propertyKey); } } } return { value: defaultValue, source }; } mergeDefaultConfigurationsForConfigurationProperty(propertyKey, value, valuesSource, existingDefaultOverride) { const property = this.configurationProperties[propertyKey]; const existingDefaultValue = existingDefaultOverride?.value ?? property?.defaultDefaultValue; let source = valuesSource; const isObjectSetting = types.isObject(value) && (property !== undefined && property.type === 'object' || property === undefined && (types.isUndefined(existingDefaultValue) || types.isObject(existingDefaultValue))); // If the default value is an object, merge the objects and store the source of each keys if (isObjectSetting) { source = existingDefaultOverride?.source ?? new Map(); // This should not happen if (!(source instanceof Map)) { console.error('defaultValueSource is not a Map'); return undefined; } for (const objectKey in value) { if (valuesSource) { source.set(`${propertyKey}.${objectKey}`, valuesSource); } } value = { ...(types.isObject(existingDefaultValue) ? existingDefaultValue : {}), ...value }; } return { value, source }; } registerOverrideIdentifiers(overrideIdentifiers) { this.doRegisterOverrideIdentifiers(overrideIdentifiers); this._onDidSchemaChange.fire(); } doRegisterOverrideIdentifiers(overrideIdentifiers) { for (const overrideIdentifier of overrideIdentifiers) { this.overrideIdentifiers.add(overrideIdentifier); } this.updateOverridePropertyPatternKey(); } doRegisterConfigurations(configurations, validate, bucket) { configurations.forEach(configuration => { this.validateAndRegisterProperties(configuration, validate, configuration.extensionInfo, configuration.restrictedProperties, undefined, bucket); this.configurationContributors.push(configuration); this.registerJSONConfiguration(configuration); }); } validateAndRegisterProperties(configuration, validate = true, extensionInfo, restrictedProperties, scope = 3 /* ConfigurationScope.WINDOW */, bucket) { scope = types.isUndefinedOrNull(configuration.scope) ? scope : configuration.scope; const properties = configuration.properties; if (properties) { for (const key in properties) { const property = properties[key]; if (validate && validateProperty(key, property)) { delete properties[key]; continue; } property.source = extensionInfo; // update default value property.defaultDefaultValue = properties[key].default; this.updatePropertyDefaultValue(key, property); // update scope if (OVERRIDE_PROPERTY_REGEX.test(key)) { property.scope = undefined; // No scope for overridable properties `[${identifier}]` } else { property.scope = types.isUndefinedOrNull(property.scope) ? scope : property.scope; property.restricted = types.isUndefinedOrNull(property.restricted) ? !!restrictedProperties?.includes(key) : property.restricted; } // Add to properties maps // Property is included by default if 'included' is unspecified if (properties[key].hasOwnProperty('included') && !properties[key].included) { this.excludedConfigurationProperties[key] = properties[key]; delete properties[key]; continue; } else { this.configurationProperties[key] = properties[key]; if (properties[key].policy?.name) { this.policyConfigurations.set(properties[key].policy.name, key); } } if (!properties[key].deprecationMessage && properties[key].markdownDeprecationMessage) { // If not set, default deprecationMessage to the markdown source properties[key].deprecationMessage = properties[key].markdownDeprecationMessage; } bucket.add(key); } } const subNodes = configuration.allOf; if (subNodes) { for (const node of subNodes) { this.validateAndRegisterProperties(node, validate, extensionInfo, restrictedProperties, scope, bucket); } } } getConfigurationProperties() { return this.configurationProperties; } getPolicyConfigurations() { return this.policyConfigurations; } registerJSONConfiguration(configuration) { const register = (configuration) => { const properties = configuration.properties; if (properties) { for (const key in properties) { this.updateSchema(key, properties[key]); } } const subNodes = configuration.allOf; subNodes?.forEach(register); }; register(configuration); } updateSchema(key, property) { allSettings.properties[key] = property; switch (property.scope) { case 1 /* ConfigurationScope.APPLICATION */: applicationSettings.properties[key] = property; break; case 2 /* ConfigurationScope.MACHINE */: machineSettings.properties[key] = property; break; case 6 /* ConfigurationScope.MACHINE_OVERRIDABLE */: machineOverridableSettings.properties[key] = property; break; case 3 /* ConfigurationScope.WINDOW */: windowSettings.properties[key] = property; break; case 4 /* ConfigurationScope.RESOURCE */: resourceSettings.properties[key] = property; break; case 5 /* ConfigurationScope.LANGUAGE_OVERRIDABLE */: resourceSettings.properties[key] = property; this.resourceLanguageSettingsSchema.properties[key] = property; break; } } updateOverridePropertyPatternKey() { for (const overrideIdentifier of this.overrideIdentifiers.values()) { const overrideIdentifierProperty = `[${overrideIdentifier}]`; const resourceLanguagePropertiesSchema = { type: 'object', description: nls.localize('overrideSettings.defaultDescription', "Configure editor settings to be overridden for a language."), errorMessage: nls.localize('overrideSettings.errorMessage', "This setting does not support per-language configuration."), $ref: resourceLanguageSettingsSchemaId, }; this.updatePropertyDefaultValue(overrideIdentifierProperty, resourceLanguagePropertiesSchema); allSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; applicationSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; machineSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; machineOverridableSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; windowSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; resourceSettings.properties[overrideIdentifierProperty] = resourceLanguagePropertiesSchema; } } registerOverridePropertyPatternKey() { const resourceLanguagePropertiesSchema = { type: 'object', description: nls.localize('overrideSettings.defaultDescription', "Configure editor settings to be overridden for a language."), errorMessage: nls.localize('overrideSettings.errorMessage', "This setting does not support per-language configuration."), $ref: resourceLanguageSettingsSchemaId, }; allSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; applicationSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; machineSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; machineOverridableSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; windowSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; resourceSettings.patternProperties[OVERRIDE_PROPERTY_PATTERN] = resourceLanguagePropertiesSchema; this._onDidSchemaChange.fire(); } updatePropertyDefaultValue(key, property) { const configurationdefaultOverride = this.configurationDefaultsOverrides.get(key)?.configurationDefaultOverrideValue; let defaultValue = undefined; let defaultSource = undefined; if (configurationdefaultOverride && (!property.disallowConfigurationDefault || !configurationdefaultOverride.source) // Prevent overriding the default value if the property is disallowed to be overridden by configuration defaults from extensions ) { defaultValue = configurationdefaultOverride.value; defaultSource = configurationdefaultOverride.source; } if (types.isUndefined(defaultValue)) { defaultValue = property.defaultDefaultValue; defaultSource = undefined; } if (types.isUndefined(defaultValue)) { defaultValue = getDefaultValue(property.type); } property.default = defaultValue; property.defaultValueSource = defaultSource; } } const OVERRIDE_IDENTIFIER_PATTERN = `\\[([^\\]]+)\\]`; const OVERRIDE_IDENTIFIER_REGEX = new RegExp(OVERRIDE_IDENTIFIER_PATTERN, 'g'); export const OVERRIDE_PROPERTY_PATTERN = `^(${OVERRIDE_IDENTIFIER_PATTERN})+$`; export const OVERRIDE_PROPERTY_REGEX = new RegExp(OVERRIDE_PROPERTY_PATTERN); export function overrideIdentifiersFromKey(key) { const identifiers = []; if (OVERRIDE_PROPERTY_REGEX.test(key)) { let matches = OVERRIDE_IDENTIFIER_REGEX.exec(key); while (matches?.length) { const identifier = matches[1].trim(); if (identifier) { identifiers.push(identifier); } matches = OVERRIDE_IDENTIFIER_REGEX.exec(key); } } return distinct(identifiers); } export function getDefaultValue(type) { const t = Array.isArray(type) ? type[0] : type; switch (t) { case 'boolean': return false; case 'integer': case 'number': return 0; case 'string': return ''; case 'array': return []; case 'object': return {}; default: return null; } } const configurationRegistry = new ConfigurationRegistry(); Registry.add(Extensions.Configuration, configurationRegistry); export function validateProperty(property, schema) { if (!property.trim()) { return nls.localize('config.property.empty', "Cannot register an empty property"); } if (OVERRIDE_PROPERTY_REGEX.test(property)) { return nls.localize('config.property.languageDefault', "Cannot register '{0}'. This matches property pattern '\\\\[.*\\\\]$' for describing language specific editor settings. Use 'configurationDefaults' contribution.", property); } if (configurationRegistry.getConfigurationProperties()[property] !== undefined) { return nls.localize('config.property.duplicate', "Cannot register '{0}'. This property is already registered.", property); } if (schema.policy?.name && configurationRegistry.getPolicyConfigurations().get(schema.policy?.name) !== undefined) { return nls.localize('config.policy.duplicate', "Cannot register '{0}'. The associated policy {1} is already registered with {2}.", property, schema.policy?.name, configurationRegistry.getPolicyConfigurations().get(schema.policy?.name)); } return null; }