Building an interpreter cookbook
Check out the Shapes sample project, which is bundled with MPS since version 3.1. It can do some fancy tricks in the editor.
tip
The sample projects are automatically installed by MPS upon its first run and they should be located in $USER_HOME /MPSSamples.
By default, the editor shows plain code that consists of commands to draw visual shapes of given sizes and colors on a canvas.
data:image/s3,"s3://crabby-images/ada6f/ada6f0e6e232418446140be7a7121f051bf18a73" alt="I1.png I1.png"
The code can be generated into Java and run. This will start a new Java process running the application that just has been generated from the code above:
data:image/s3,"s3://crabby-images/6c932/6c932bf29494cd34f0147b46628561dfee3fb932" alt="I4.png I4.png"
data:image/s3,"s3://crabby-images/852a3/852a37ee184970ce05909840650c4700183f4e2e" alt="I5.png I5.png"
MPS can, however, also interpret the code without generating Java. Press AltEnter anywhere in the code to invoke the Intentions pop-up menu and choose "Preview Scene". You'll get a new frame containing your scene, which is interpreting the code:
data:image/s3,"s3://crabby-images/9596b/9596bad2f27f132cf932b6d237d5c958826a6265" alt="I6.png I6.png"
data:image/s3,"s3://crabby-images/7ef4c/7ef4cddc6b155a3adee1a034ce03e2a5096b169c" alt="I7.png I7.png"
With the ability to interpret code the editor can now serve the results of the interpreter back to the developer through the editor. It can either interpret individual shapes and draw them in the editor next to the code that defines each of them:
data:image/s3,"s3://crabby-images/5596b/5596bcb54c588844091066a4b8d4055d05d104b5" alt="I2.png I2.png"
Or the whole scene can be drawn next to the code and react instantaneously to the changes made to the code.
data:image/s3,"s3://crabby-images/3e950/3e95093e7dfa1dae28251a17ac48fe8fc76be6ec" alt="I3.png I3.png"
To try these capabilities yourself, right-click the editor, pick "Push Editor Hints" and select the hints that will indicate to the editor, which of the preview capabilities you want to enable:
data:image/s3,"s3://crabby-images/3d1ea/3d1eaa085f277a6745c5ed6d55309f03e2533391" alt="I8.png I8.png"
data:image/s3,"s3://crabby-images/7508a/7508abc56dc07fe6907078228ffcf22ac6555789" alt="I9.png I9.png"
This was a quick teaser, now we can have a look at how to build such simple interpreting capabilities into your language.
MPS is primarily focused on code generation. Programs represented as ASTs get translated into code in a target language, which then may get compiled and run. There are, however, situations when interpreting your code directly may be a better option. You could give the developers some sort of Preview of what the code does and thus providing instant feedback to their code changes. You could detect and report errors that only show at run-time Or you could just want to speed up code-run-fix cycle turnaround by bypassing the generation and compilation phases.
MPS does not come with any infrastructure for building interpreters at the moment, but the Behavior aspect of your language gives you some power to build interpreters on your own. Ideally, just like with code generators, you would create a runtime solution holding the interpreter runtime classes. The Behavior aspect of your language concepts would then cooperate with the runtime classed to navigate the AST properly and interpret all nodes in the mode.
We'll have a look at how the Shapes language enabled interpreting its code.
Let's start with the scenario when the user invokes the interpreter explicitly by an Intention. First, the intention needs to be created:
data:image/s3,"s3://crabby-images/2319d/2319d905617dd0660cfdc10ec205f8995586e61e" alt="I10.png I10.png"
The intention invokes the interpret() method on the Canvas node, which is defined in the Behavior aspect of Canvas.
data:image/s3,"s3://crabby-images/3dae5/3dae5335587dd1525a37e0a1ca731136f9c4d823" alt="I21.png I21.png"
The method builds a new Frame that will hold a panel with a customized paintComponent() method. The panel is being reused by other preview functionality, so it has been extracted into a separate helper class PreviewFactory. For other than trivial cases the class should be placed into a runtime solution, but here we've just put it directly into the Behavior aspect of the language.
data:image/s3,"s3://crabby-images/f5152/f5152cd65897f6aeee861d3520951b65972fb7d8" alt="I22.png I22.png"
The current model is traversed with the code: thisCanvas.shapes.forEach({~it => it.drawShape(graphics)});. This is where the interpreting happens. The drawShape() method is implemented by each Shape subconcept so that they can get rendered on the screen. Notice that the traversal code is wrapped inside a ReadAction to be guaranteed a read permission to the model.
The drawShape() method has to be implemented by all Shapes:
data:image/s3,"s3://crabby-images/dbb61/dbb61b0240c48f60a7350b159b511bb3350a5e67" alt="I12.png I12.png"
data:image/s3,"s3://crabby-images/d96bc/d96bc6a38a57dc237a682cb7cf76ee0489aa523e" alt="I13.png I13.png"
data:image/s3,"s3://crabby-images/3e9c5/3e9c55c2002e6b0201d4e19d9012bc06d9f2e7be" alt="I14.png I14.png"
This is enough to get our little interpreter off the ground.
To preview individual shapes next to their definition in code, we need to change the editor or define a new one with a different editor hint , so that it holds a swing component containing the shape's visualization. In the sample we chose to leverage multiple projections (refer to the Multiple Projections video to see how multiple projections work in MPS) and thus we created a new editor with the ShapePreview hint specified in the header. Only when the user enables the ShapePreview hint, this editor will be used instead of the default text-only one.
data:image/s3,"s3://crabby-images/93378/933788cded53499e72b4b782c05f2ed2b8380c9c" alt="I16.png I16.png"
The swing component defines a JPanel, which in the paintComponent() method obtains a read lock on the model and then draws the current Shape node at a specific location. Notice that a new drawShapeAt() method has been added to Shapes to draw the Shapes ignoring their defined position within the canvas.
data:image/s3,"s3://crabby-images/01374/013749439796b5e73207a5f107954d0e671b4a5b" alt="I17.png I17.png"
To preview a whole scene, we'll need to create a new editor for the Canvas concept and hook it to the ScenePreview editor hint.
data:image/s3,"s3://crabby-images/5c089/5c089a5bfdeda002b61d31423925a38bfe05ece7" alt="I23.png I23.png"
The swing component cell contains a customized JPanel, which is again created by the PreviewFactory class just like when we were interpreting on explicit user request earlier on.
Now, this is it. I hope you liked it.
Thanks for your feedback!