@eclipse-glsp/client
Version:
A sprotty-based client for GLSP
148 lines (130 loc) • 5.88 kB
text/typescript
/********************************************************************************
* Copyright (c) 2023 Business Informatics Group (TU Wien) and others.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the Eclipse
* Public License v. 2.0 are satisfied: GNU General Public License, version 2
* with the GNU Classpath Exception which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
********************************************************************************/
import {
Action,
codiconCSSString,
CreateEdgeOperation,
GModelElement,
GModelRoot,
IActionHandler,
isConnectable,
LabeledAction,
name,
SetUIExtensionVisibilityAction,
toArray,
TriggerEdgeCreationAction
} from '@eclipse-glsp/sprotty';
import { injectable } from 'inversify';
import { CloseReason, toActionArray } from '../../../base/auto-complete/auto-complete-widget';
import { AutocompleteSuggestion, IAutocompleteSuggestionProvider } from '../../../base/auto-complete/autocomplete-suggestion-providers';
import { EnableDefaultToolsAction } from '../../../base/tool-manager/tool';
import { GEdge } from '../../../model';
import { SearchAutocompletePalette } from '../search/search-palette';
import { SetEdgeTargetSelectionAction } from './action';
import { EdgeAutocompleteContext } from './edge-autocomplete-context';
export namespace EdgeAutocompletePaletteMetadata {
export const ID = 'edge-autocomplete-palette';
}
export class EdgeAutocompletePalette extends SearchAutocompletePalette implements IActionHandler {
protected context?: EdgeAutocompleteContext;
protected readonly targetSuggestionProvider = new PossibleEdgeTargetAutocompleteSuggestionProvider();
override id(): string {
return EdgeAutocompletePaletteMetadata.ID;
}
handle(action: Action): Action | void {
if (TriggerEdgeCreationAction.is(action)) {
this.context = {
trigger: action,
role: 'source'
};
this.targetSuggestionProvider.setContext(action, this.context);
}
}
protected override onBeforeShow(containerElement: HTMLElement, root: Readonly<GModelRoot>, ...contextElementIds: string[]): void {
super.onBeforeShow(containerElement, root, ...contextElementIds);
this.autocompleteWidget.inputField.placeholder = `Search for ${this.context?.role} elements`;
}
protected override getSuggestionProviders(root: Readonly<GModelRoot>, input: string): IAutocompleteSuggestionProvider[] {
return [this.targetSuggestionProvider];
}
protected reload(): void {
const context = this.context;
this.hide();
this.context = context;
this.actionDispatcher.dispatch(
SetUIExtensionVisibilityAction.create({
extensionId: EdgeAutocompletePaletteMetadata.ID,
visible: true
})
);
}
protected override executeSuggestion(input: LabeledAction | Action[] | Action): void {
const action = toActionArray(input)[0] as SetEdgeTargetSelectionAction;
if (this.context?.role === 'source') {
this.context.sourceId = action.elementId;
this.context.role = 'target';
this.reload();
} else if (this.context?.role === 'target') {
this.context.targetId = action.elementId;
}
if (this.context?.sourceId !== undefined && this.context?.targetId !== undefined) {
this.actionDispatcher.dispatchAll([
CreateEdgeOperation.create({
elementTypeId: this.context.trigger.elementTypeId,
sourceElementId: this.context.sourceId,
targetElementId: this.context.targetId,
args: this.context.trigger.args
}),
EnableDefaultToolsAction.create()
]);
this.hide();
}
}
protected override autocompleteHide(reason: CloseReason): void {
if (reason !== 'submission') {
this.hide();
}
}
}
export class PossibleEdgeTargetAutocompleteSuggestionProvider implements IAutocompleteSuggestionProvider {
protected proxyEdge?: GEdge;
protected context?: EdgeAutocompleteContext;
setContext(triggerAction: TriggerEdgeCreationAction, edgeAutocompleteContext: EdgeAutocompleteContext): void {
this.proxyEdge = new GEdge();
this.proxyEdge.type = triggerAction.elementTypeId;
this.context = edgeAutocompleteContext;
}
isAllowedSource(element: GModelElement | undefined, role: 'source' | 'target'): boolean {
return element !== undefined && this.proxyEdge !== undefined && isConnectable(element) && element.canConnect(this.proxyEdge, role);
}
async retrieveSuggestions(root: Readonly<GModelRoot>, text: string): Promise<AutocompleteSuggestion[]> {
const context = this.context;
if (this.context === undefined) {
return [];
}
const nodes = toArray(root.index.all().filter(element => this.isAllowedSource(element, context!.role))) as GEdge[];
return nodes.map(node => ({
element: node,
action: {
label: `[${node.type}] ${name(node) ?? '<no-name>'}`,
actions: [SetEdgeTargetSelectionAction.create(node.id, context!.role)],
icon: codiconCSSString('arrow-both')
}
}));
}
}