- Hover provider showing entity information and type - Go-to-definition (F12) for entity references - Basic IFC file validation (ISO-10303-21 header check) - Entity parsing with regex-based detection - Proper CommonJS module system (avoiding ES module issues) This replaces the broken baseline from ifc-developer-tools which had: - Non-functional ES module configuration - Circular dependency issues - Parser crashes - Non-working PositionVisitor Built on Microsoft's LSP example template for a clean, maintainable foundation. Next: Add hierarchical entity dependency tree in hover tooltip."
209 lines
7.9 KiB
JavaScript
209 lines
7.9 KiB
JavaScript
"use strict";
|
|
/* --------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
* ------------------------------------------------------------------------------------------ */
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.SyncConfigurationFeature = exports.toJSONObject = exports.ConfigurationFeature = void 0;
|
|
const vscode_1 = require("vscode");
|
|
const vscode_languageserver_protocol_1 = require("vscode-languageserver-protocol");
|
|
const Is = require("./utils/is");
|
|
const UUID = require("./utils/uuid");
|
|
const features_1 = require("./features");
|
|
/**
|
|
* Configuration pull model. From server to client.
|
|
*/
|
|
class ConfigurationFeature {
|
|
constructor(client) {
|
|
this._client = client;
|
|
}
|
|
getState() {
|
|
return { kind: 'static' };
|
|
}
|
|
fillClientCapabilities(capabilities) {
|
|
capabilities.workspace = capabilities.workspace || {};
|
|
capabilities.workspace.configuration = true;
|
|
}
|
|
initialize() {
|
|
let client = this._client;
|
|
client.onRequest(vscode_languageserver_protocol_1.ConfigurationRequest.type, (params, token) => {
|
|
let configuration = (params) => {
|
|
let result = [];
|
|
for (let item of params.items) {
|
|
let resource = item.scopeUri !== void 0 && item.scopeUri !== null ? this._client.protocol2CodeConverter.asUri(item.scopeUri) : undefined;
|
|
result.push(this.getConfiguration(resource, item.section !== null ? item.section : undefined));
|
|
}
|
|
return result;
|
|
};
|
|
let middleware = client.middleware.workspace;
|
|
return middleware && middleware.configuration
|
|
? middleware.configuration(params, token, configuration)
|
|
: configuration(params, token);
|
|
});
|
|
}
|
|
getConfiguration(resource, section) {
|
|
let result = null;
|
|
if (section) {
|
|
let index = section.lastIndexOf('.');
|
|
if (index === -1) {
|
|
result = toJSONObject(vscode_1.workspace.getConfiguration(undefined, resource).get(section));
|
|
}
|
|
else {
|
|
let config = vscode_1.workspace.getConfiguration(section.substr(0, index), resource);
|
|
if (config) {
|
|
result = toJSONObject(config.get(section.substr(index + 1)));
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
let config = vscode_1.workspace.getConfiguration(undefined, resource);
|
|
result = {};
|
|
for (let key of Object.keys(config)) {
|
|
if (config.has(key)) {
|
|
result[key] = toJSONObject(config.get(key));
|
|
}
|
|
}
|
|
}
|
|
if (result === undefined) {
|
|
result = null;
|
|
}
|
|
return result;
|
|
}
|
|
clear() {
|
|
}
|
|
}
|
|
exports.ConfigurationFeature = ConfigurationFeature;
|
|
function toJSONObject(obj) {
|
|
if (obj) {
|
|
if (Array.isArray(obj)) {
|
|
return obj.map(toJSONObject);
|
|
}
|
|
else if (typeof obj === 'object') {
|
|
const res = Object.create(null);
|
|
for (const key in obj) {
|
|
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
res[key] = toJSONObject(obj[key]);
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
}
|
|
return obj;
|
|
}
|
|
exports.toJSONObject = toJSONObject;
|
|
class SyncConfigurationFeature {
|
|
constructor(_client) {
|
|
this._client = _client;
|
|
this.isCleared = false;
|
|
this._listeners = new Map();
|
|
}
|
|
getState() {
|
|
return { kind: 'workspace', id: this.registrationType.method, registrations: this._listeners.size > 0 };
|
|
}
|
|
get registrationType() {
|
|
return vscode_languageserver_protocol_1.DidChangeConfigurationNotification.type;
|
|
}
|
|
fillClientCapabilities(capabilities) {
|
|
(0, features_1.ensure)((0, features_1.ensure)(capabilities, 'workspace'), 'didChangeConfiguration').dynamicRegistration = true;
|
|
}
|
|
initialize() {
|
|
this.isCleared = false;
|
|
let section = this._client.clientOptions.synchronize?.configurationSection;
|
|
if (section !== undefined) {
|
|
this.register({
|
|
id: UUID.generateUuid(),
|
|
registerOptions: {
|
|
section: section
|
|
}
|
|
});
|
|
}
|
|
}
|
|
register(data) {
|
|
let disposable = vscode_1.workspace.onDidChangeConfiguration((event) => {
|
|
this.onDidChangeConfiguration(data.registerOptions.section, event);
|
|
});
|
|
this._listeners.set(data.id, disposable);
|
|
if (data.registerOptions.section !== undefined) {
|
|
this.onDidChangeConfiguration(data.registerOptions.section, undefined);
|
|
}
|
|
}
|
|
unregister(id) {
|
|
let disposable = this._listeners.get(id);
|
|
if (disposable) {
|
|
this._listeners.delete(id);
|
|
disposable.dispose();
|
|
}
|
|
}
|
|
clear() {
|
|
for (const disposable of this._listeners.values()) {
|
|
disposable.dispose();
|
|
}
|
|
this._listeners.clear();
|
|
this.isCleared = true;
|
|
}
|
|
onDidChangeConfiguration(configurationSection, event) {
|
|
if (this.isCleared) {
|
|
return;
|
|
}
|
|
let sections;
|
|
if (Is.string(configurationSection)) {
|
|
sections = [configurationSection];
|
|
}
|
|
else {
|
|
sections = configurationSection;
|
|
}
|
|
if (sections !== undefined && event !== undefined) {
|
|
let affected = sections.some((section) => event.affectsConfiguration(section));
|
|
if (!affected) {
|
|
return;
|
|
}
|
|
}
|
|
const didChangeConfiguration = async (sections) => {
|
|
if (sections === undefined) {
|
|
return this._client.sendNotification(vscode_languageserver_protocol_1.DidChangeConfigurationNotification.type, { settings: null });
|
|
}
|
|
else {
|
|
return this._client.sendNotification(vscode_languageserver_protocol_1.DidChangeConfigurationNotification.type, { settings: this.extractSettingsInformation(sections) });
|
|
}
|
|
};
|
|
let middleware = this._client.middleware.workspace?.didChangeConfiguration;
|
|
(middleware ? middleware(sections, didChangeConfiguration) : didChangeConfiguration(sections)).catch((error) => {
|
|
this._client.error(`Sending notification ${vscode_languageserver_protocol_1.DidChangeConfigurationNotification.type.method} failed`, error);
|
|
});
|
|
}
|
|
extractSettingsInformation(keys) {
|
|
function ensurePath(config, path) {
|
|
let current = config;
|
|
for (let i = 0; i < path.length - 1; i++) {
|
|
let obj = current[path[i]];
|
|
if (!obj) {
|
|
obj = Object.create(null);
|
|
current[path[i]] = obj;
|
|
}
|
|
current = obj;
|
|
}
|
|
return current;
|
|
}
|
|
let resource = this._client.clientOptions.workspaceFolder
|
|
? this._client.clientOptions.workspaceFolder.uri
|
|
: undefined;
|
|
let result = Object.create(null);
|
|
for (let i = 0; i < keys.length; i++) {
|
|
let key = keys[i];
|
|
let index = key.indexOf('.');
|
|
let config = null;
|
|
if (index >= 0) {
|
|
config = vscode_1.workspace.getConfiguration(key.substr(0, index), resource).get(key.substr(index + 1));
|
|
}
|
|
else {
|
|
config = vscode_1.workspace.getConfiguration(undefined, resource).get(key);
|
|
}
|
|
if (config) {
|
|
let path = keys[i].split('.');
|
|
ensurePath(result, path)[path[path.length - 1]] = toJSONObject(config);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
exports.SyncConfigurationFeature = SyncConfigurationFeature;
|