Description comments
We are going to learn how to leverage the capability of adding attributes to nodes. This feature is fully described in the Attributes section of the Structure chapter. In this cookbook we'll create a simple addition to the Calculator tutorial language that will allow the input and output fields of the calculator definitions to have descriptive comments attached to them. These descriptive comments will be propagated into the generated Java code in the form of "popup tooltips" attached to the Swing text fields corresponding to the input and output fields.
tip
Since MPS 2018.2 documentation comments can be seamlessly inserted into code in any language without explicit support in the language itself. The feature is described in Generic placeholders and generic comments. Adding support for comments into languages is no longer necessary, unless you want to greatly customize the way comments are used. Nevertheless, this cookbook focuses on attributes and is still highly relevant.
We chose the Calculator tutorial language as a testbed for our experiments. You can find the calculator-tutorial project included in the set of sample projects that comes with the MPS distribution. I recommend you skimmed through the tutorial to familiarize yourself with the language before we continue.
Let's start building the machinery for adding descriptive comments to input and output fields. Our approach will be based on annotations that when added to nodes will have impact on their visual appearance as well as on the generated code. There are several vital components that will enable our desired functionality:
A marker interface that indicates, which nodes can hold the description comments
The annotation itself to indicate whether a node holds the description comment and what the description is
An action providing a keyboard shortcut to quickly add and remove the description comment to/from a node under caret
Updating the code generation process in order to generate the visual "tooltips" for the described components
We start with a new interface to mark all commentable nodes:
data:image/s3,"s3://crabby-images/26f3d/26f3d14bd925cb157e6f37847fcdd9abb012db26" alt="Marking the nodes Marking the nodes"
Now both InputField and OutputField concepts have to implement the new ICanHaveDescription interface.
data:image/s3,"s3://crabby-images/d3281/d32815c5d8254652eee35c32c88fb5f376ba5738" alt="Implementing the ICanHaveDescription interface Implementing the ICanHaveDescription interface"
We are still missing the core pice of our puzzle - the actual annotation that will be attributed to input and output fields to specify the description comment. Let's create it now:
data:image/s3,"s3://crabby-images/6788c/6788c650051aec8b73e1a7a37c0a8f9edc0c2b04" alt="Creating the description Creating the description"
Once you make the concept extend NodeAttribute, MPS will hint you to provide further information detailing the attribution. Invoking a quick-fix through AltEnter will add the necessary information for you to customize:
data:image/s3,"s3://crabby-images/3c0e6/3c0e67eaf5bc2fca7c47a0e861fd9f97510eb3f0" alt="Invoking a quick-fix Invoking a quick-fix"
You need to specify additional qualities of the new annotation by setting values for role as well as attributed concepts. We set the role for the annotation as descriptionComment. This is the name to use in queries in order to find out, whether a node has been annotated with DescriptionAnnotation or not. We also indicate that the annotation can only be attributed to nodes implementing ICanHaveDescription, which in our case is InputField and OutputField.
data:image/s3,"s3://crabby-images/08ce8/08ce8d0577f526df368040aa9801b8ca2fa1cdc7" alt="Specifying additional qualities Specifying additional qualities"
The DescriptionAnnotation also has to store the actual text of the description - we'll use a string property descriptionText for this:
data:image/s3,"s3://crabby-images/448a2/448a275b606c1bbd458e115cd4c3bbe94f778345" alt="Adding the description text Adding the description text"
The editor of DescriptionAnnotation is the key element in defining the visual behavior of nodes with description comments attached to them. We prepend the annotated node with a "described" constant and append the actual description text wrapped in parentheses. The attributed node cell represents the editor of the actual node - the node that the attribute is attached to.
data:image/s3,"s3://crabby-images/7a511/7a5111dd860c2b646900caad65ef554c1e52837c" alt="Attaching to a node Attaching to a node"
To toggle the description comment of a node on and off, we will create new KeyMap.
data:image/s3,"s3://crabby-images/b9a93/b9a9326b7daf3c2d4a5e175d8a89066b5db309eb" alt="Creating a new key map Creating a new key map"
To edit the code we will use concepts from BaseLanguage and smodel, which should be available by default. If not, import them through Ctrl0L. We will also need the actions language in order to create initialized nodes easily. This one will most likely need to be imported explicitly (Control + L).
data:image/s3,"s3://crabby-images/f3b6d/f3b6d8c16d4f90af8bd52d1a3021cfa453e8e4b7" alt="Adding an import statement Adding an import statement"
Now we can complete the action:
data:image/s3,"s3://crabby-images/82ca4/82ca4bf5b361f158bc2bf407ea3d56357a4ba6f4" alt="Completing the action Completing the action"
Whenever Control + Alt + Shift + D is pressed on a node implementing ICanHaveDescription, the action will add or remove the DescriptionAnnotation to/from the node.
The KeyMap now needs to be added to all fields that should react to Control + Alt + Shift + D. This is to InputField and OutputField in our case. The editor cells that should react to the Control + Alt + Shift + D key combination must have the key map specified in the keycap property.
data:image/s3,"s3://crabby-images/1e914/1e914bf8a5ecf60b715768dc01d56419a7481e1a" alt="Adding the key map to all fields Adding the key map to all fields"
After compilation you are able to use Control + Alt + Shift + D both on input fields and output fields to toggle the description comment:
data:image/s3,"s3://crabby-images/eafc1/eafc1f272afee39400713f44e0a5868a66d68c6d" alt="Toggling the description comment Toggling the description comment"
The DescriptionAnnotation is currently lost during the generation process. Instead, we would like to generate the Swing "tooltips" for the text fields that represent the commented input and output fields.
data:image/s3,"s3://crabby-images/fd76d/fd76d70c43ae3a834bac543b97a6e8d99e5951cc" alt="Updating the generator Updating the generator"
When generating code we currently take all input or output fields in the calculator and generate JTextFields for them. In the constructor we also generate initialization code for each generated JTextField - this is the place, where we can attach a Swing "tooltip" the JTextFields that represent an input or output fields with description comments.
data:image/s3,"s3://crabby-images/f740f/f740f108d2c18f6677c03b13e6221d2e3e002646" alt="Creating JTextFields Creating JTextFields"
The $IF$ node macro will ensure that a "tooltip" is only generated for nodes with the DescriptionComment annotation attached.
data:image/s3,"s3://crabby-images/6b155/6b155ddb772eef57d0ac8c39cd8bac791e6d7637" alt="Adding a macro Adding a macro"
The property macro of the text of the "tooltip" will set the descriptionText of the DescriptionAnnotation attached to the current node:
data:image/s3,"s3://crabby-images/95adf/95adfb159e3cc7b6f29f87457059f1c1b1d5b350" alt="Property macro Property macro"
After rebuilding the language and the sandbox solution, the generated Calculator form will have "tooltips" attached to the fields that have been marked with the DescriptionAnnotation:
data:image/s3,"s3://crabby-images/fd76d/fd76d70c43ae3a834bac543b97a6e8d99e5951cc" alt="Resulting tooltip Resulting tooltip"
tip
You may now consider checking out the Requirement tracking language cookbook, which provides a commented tour round the Requirement tracking sample project showing another practical use of node attributes together with multiple projections and scoping.
We can finish our tour here. Both input fields and output fields can be described with DescriptionAnnotation and the generated code contains a "tooltip" for each such description.
Thanks for your feedback!