UNPKG

@eclipse-glsp/client

Version:

A sprotty-based client for GLSP

214 lines (187 loc) 8.21 kB
/******************************************************************************** * Copyright (c) 2019-2024 EclipseSource 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, CommandExecutionContext, CommandReturn, DeleteMarkersAction, GParentElement, IActionDispatcher, IActionHandler, ICommand, Marker, SetMarkersAction, TYPES, hasArrayProp } from '@eclipse-glsp/sprotty'; import { inject, injectable, optional } from 'inversify'; import { EditorContextService } from '../../base/editor-context-service'; import { IFeedbackActionDispatcher, IFeedbackEmitter } from '../../base/feedback/feedback-action-dispatcher'; import { FeedbackCommand } from '../../base/feedback/feedback-command'; import { FeedbackEmitter } from '../../base/feedback/feedback-emitter'; import { removeCssClasses } from '../../utils/gmodel-util'; import { GIssueMarker, createGIssue, getGIssueMarker, getOrCreateGIssueMarker, getSeverity } from './issue-marker'; /** * Feedback emitter sending actions for visualizing model validation feedback and * re-establishing this feedback visualization after the model has been updated. */ @injectable() export class ValidationFeedbackEmitter implements IFeedbackEmitter { @inject(TYPES.IFeedbackActionDispatcher) protected feedbackActionDispatcher: IFeedbackActionDispatcher; protected registeredFeedbackByReason: Map<string, FeedbackEmitter> = new Map(); /** * Register the action that should be emitted for visualizing validation feedback. * @param action the action that should be emitted when the model is updated and that will visualize the model validation feedback. * @param reason the reason for this validation feedback. */ registerValidationFeedbackAction(action: ApplyMarkersAction, reason = ''): void { // De-register feedback and clear existing markers with the same reason const emitter = this.registeredFeedbackByReason.get(reason)?.dispose() ?? this.feedbackActionDispatcher.createEmitter(); emitter.add(action, () => DeleteMarkersAction.create(action.markers)).submit(); this.registeredFeedbackByReason.set(reason, emitter); } } /** * Manages current markers for the outside of the GLSP. * * Typically this is rebound by the surrounding tool, e.g. Theia, to be aware of * and propagate current markers. */ @injectable() export abstract class ExternalMarkerManager { languageLabel: string; protected actionDispatcher?: IActionDispatcher; connect(actionDispatcher: IActionDispatcher): void { this.actionDispatcher = actionDispatcher; } removeMarkers(markers: Marker[]): void { if (this.actionDispatcher) { this.actionDispatcher.dispatch(DeleteMarkersAction.create(markers)); } } abstract setMarkers(markers: Marker[], reason?: string, sourceUri?: string): void; } @injectable() export class SetMarkersActionHandler implements IActionHandler { @inject(ValidationFeedbackEmitter) protected validationFeedbackEmitter: ValidationFeedbackEmitter; @inject(ExternalMarkerManager) @optional() protected externalMarkerManager?: ExternalMarkerManager; @inject(EditorContextService) protected editorContextService: EditorContextService; handle(action: SetMarkersAction): void | Action | ICommand { const markers: Marker[] = action.markers; this.setMarkers(markers, action.reason); } async setMarkers(markers: Marker[], reason: string | undefined): Promise<void> { const uri = this.editorContextService.sourceUri; this.externalMarkerManager?.setMarkers(markers, reason, uri); const applyMarkersAction = ApplyMarkersAction.create(markers); this.validationFeedbackEmitter.registerValidationFeedbackAction(applyMarkersAction, reason); } } /** * Action for applying makers to the graphical model. */ export interface ApplyMarkersAction extends Action { kind: typeof ApplyMarkersAction.KIND; markers: Marker[]; } export namespace ApplyMarkersAction { export const KIND = 'applyMarkers'; export function is(object: any): object is ApplyMarkersAction { return Action.hasKind(object, KIND) && hasArrayProp(object, 'markers'); } export function create(markers: Marker[]): ApplyMarkersAction { return { kind: KIND, markers }; } } /** * Handles {@link ApplyMarkersAction}s by creating the corresponding {@link GIssueMarker}s and * adding them to the graphical model. */ @injectable() export class ApplyMarkersCommand extends FeedbackCommand { static KIND = ApplyMarkersAction.KIND; constructor(@inject(TYPES.Action) protected action: ApplyMarkersAction) { super(); } execute(context: CommandExecutionContext): CommandReturn { this.action.markers.forEach(marker => { const modelElement = context.root.index.getById(marker.elementId); if (modelElement instanceof GParentElement) { const issueMarker = getOrCreateGIssueMarker(modelElement); const issue = createGIssue(marker); issueMarker.issues.push(issue); if (issueMarker instanceof GIssueMarker) { issueMarker.computeProjectionCssClasses(); } addMaxSeverityCSSClassToIssueParent(modelElement, issueMarker); } }); return context.root; } } function addMaxSeverityCSSClassToIssueParent(modelElement: GParentElement, issueMarker: GIssueMarker): void { const maxSeverityCSSClass = getSeverity(issueMarker); if (!modelElement.cssClasses) { modelElement.cssClasses = [maxSeverityCSSClass]; } else { modelElement.cssClasses = modelElement.cssClasses.filter((value: string) => !value.match('info|warning|error')); modelElement.cssClasses.push(maxSeverityCSSClass); } } function removeCSSClassFromIssueParent(modelElement: GParentElement, issueMarker: GIssueMarker): void { const severity = getSeverity(issueMarker); removeCssClasses(modelElement, [severity]); } /** * Command for handling `DeleteMarkersAction` */ @injectable() export class DeleteMarkersCommand extends FeedbackCommand { static KIND = DeleteMarkersAction.KIND; constructor(@inject(TYPES.Action) protected action: DeleteMarkersAction) { super(); } execute(context: CommandExecutionContext): CommandReturn { this.action.markers.forEach(marker => { const modelElement = context.root.index.getById(marker.elementId); if (modelElement instanceof GParentElement) { const issueMarker = getGIssueMarker(modelElement); if (issueMarker) { removeCSSClassFromIssueParent(modelElement, issueMarker); for (let index = 0; index < issueMarker.issues.length; ++index) { const issue = issueMarker.issues[index]; if (issue.message === marker.description) { issueMarker.issues.splice(index--, 1); } } if (issueMarker.issues.length === 0) { modelElement.remove(issueMarker); } else { addMaxSeverityCSSClassToIssueParent(modelElement, issueMarker); } } } }); return context.root; } }