No matter with what graphics technology you choose to implement a diagram for your Xtext-based language, there are a few things you should know in order to connect the diagram view/editor to the rest of the infrastructure.
Let us first have a look how to implement a context menu action for the Xtext editor to show the element at the current cursor position in the diagram. In Eclipse, you have to implement a handler for such an action. Xtext offers a bunch of classes making your life easier. EditorUtils detects the Xtext editor for the handler’s action and EObjectAtOffsetHelper finds the semantic model element at a given text offset.
As the document – and thereby the model parsed from it – is subject to editing, you have to make sure nobody is changing it while you traverse it to derive your diagram. You can execute code in a read transaction on the XtextDocument by wrapping it in an IUnitOfWork that you pass to the XtextDocument#readOnly method.
The following snipped shows an example handler written in Xtend.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class ShowInViewHandler extends AbstractHandler { | |
@Inject EObjectAtOffsetHelper eObjectAtOffsetHelper | |
override Object execute(ExecutionEvent event) | |
throws ExecutionException { | |
val editor = EditorUtils.getActiveXtextEditor(event) | |
if (editor != null) { | |
val selection = editor.selectionProvider.selection | |
as ITextSelection | |
editor.document.readOnly [ | |
val selectedElement = eObjectAtOffsetHelper | |
.resolveElementAt(it, selection.offset) | |
if (selectedElement != null) { | |
... // open in view | |
} | |
] | |
} | |
return null | |
} | |
} |
To navigate from a diagram element back to it’s element definition in the text, you need to store some back reference. The model element itself is not suitable, as its lifecycle is bound to the one of the editor, and the parser is even free to expunge model elements whenever the document is re-parsed. As Xtext is based on EMF, each model element has a unique URI. You should always use these URIs to store references to model elements beyond the boundaries of a read/write transaction. An element’s URI is provided by EMF's EcoreUtil#getURI method.
Having the URI, the navigation to the respective model element’s definition is easiest to implement using Xtext’s IURIEditorOpener. If your diagram view supports selection listeners, the respective Xtend code could look like this:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@Inject IURIEditorOpener editorOpener | |
... | |
selectionListener = [ event | | |
val selection = event.selection as IStructuredSelection | |
if(selection.size == 1) { | |
val selectedElement = selection.firstElement | |
if(selectedElement instanceof URI) | |
editorOpener.open(selectedElement, true) | |
} | |
] |