Concept Functions
Concept functions allow language designers to leave hooks for their language users, through which the users can provide code to leverage in the generated code. For example, most of the languages that MPS offers for language design, such as Editor , Constraints or Intentions, leverage Concept functions:
data:image/s3,"s3://crabby-images/b730d/b730d2d74c52e9d95a21bbd41a914f17e55c1eac" alt="cf1.png cf1.png"
You can also discover their usages down in the Inspector window:
data:image/s3,"s3://crabby-images/ac08f/ac08fe864e553a8d1804dacc9ced245175001fb2" alt="cf2.png cf2.png"
Concept functions are defined in jetbrains.mps.baselanguage and they contain BaseLanguage code, which upon generation becomes part of the generated Java code. This option can give your DSLs enormous flexibility.
We'll use the Robot Kaja sample project to experiment with Concept functions. The goal is to allow the Script authors to provide a function that will customize the Trace messages, which are reported to the user through the trace command:
data:image/s3,"s3://crabby-images/909e7/909e706908708ad3a91b26449811aaaed3641c99" alt="cf3.png cf3.png"
data:image/s3,"s3://crabby-images/663e1/663e17f7d2618f1030fbe4dad10d15a2626699b9" alt="cf4.png cf4.png"
The user will be able to customize the trace messages through a function that receives the original message as a parameter and returns a string that should be displayed instead:
data:image/s3,"s3://crabby-images/9c123/9c123880c5d81ff2da6ecf3ee0c30fe8daa9f399" alt="cf5.png cf5.png"
data:image/s3,"s3://crabby-images/64dea/64dea6e0c1a4011c0783de74a880356504a36b42" alt="cf6.png cf6.png"
First, a sub-concept of ConceptFunction must be created:
data:image/s3,"s3://crabby-images/107cb/107cba5bb1fe04e2bbcdc2bfbab61ac14e1f75e0" alt="cf7.png cf7.png"
data:image/s3,"s3://crabby-images/37704/377043c62e658b8bd10e58fee598f9a3564cbbbb" alt="cf8.png cf8.png"
The behavior aspect overrides a few methods inherited from ConceptFunction:
getExpectedReturnType() - declares what type should be returned from the function
getParameterConcepts() - lists the concepts that will represent parameters to this function
showName() - indicates, whether the name of the function should be displayed in the editor alongside the parameter list and the return type
getName() - the name of the function to display in the editor. As it defaults to the concept alias, you will usually only want to override this if needing a dynamic name
Since MyFunction requires an argument to hold the original trace message value, we also need to create a concept to represent that parameter, which extends the ConceptFunctionParameter concept and specifies its type through an overridden getType() behavior method:
data:image/s3,"s3://crabby-images/65b69/65b690e1c8d692ccb83cf193d3a89c438a424c78" alt="cf9.png cf9.png"
data:image/s3,"s3://crabby-images/ed60c/ed60cd36bd8a5f4620f17e0613b6aa7c55334520" alt="cf10.png cf10.png"
Once defined, the MyFunction concept can be added to Script:
data:image/s3,"s3://crabby-images/91df8/91df84d62dd213f9dcf03ab27ee55e8d4b259aa3" alt="cf11.png cf11.png"
data:image/s3,"s3://crabby-images/b3d3b/b3d3b1014d260de7d2b34bbfc07b0b5332e6cd84" alt="cf12.png cf12.png"
This will allow us to edit the function in the Script editor:
data:image/s3,"s3://crabby-images/6bbde/6bbdeb27b45eb89e94efa7eb3bf75e1aef026ad0" alt="cf16.png cf16.png"
When you hit enter, the editor will display the signature of the concept function and you will be able to edit its body:
data:image/s3,"s3://crabby-images/a4496/a4496d9783b0bbc2bfde0b4f7a8334551b608003" alt="cf17.png cf17.png"
Notice that the Inspector shows the description messages for the function as well as its parameters, when you place the caret at the concept function signature.
The last step that remains is to alter the generator so that the trace message customization can happen. We first need to modify the KajaFrame class, which is a super-class for all the classes that get generated from Robot Kaja Scripts:
data:image/s3,"s3://crabby-images/269a1/269a1eba74e49883982c8a919c9e045d3eed0ef7" alt="cf15.png cf15.png"
The trace() method needs to call the new customizeMessage() method in order to have the original trace message customized. The default implementation of customizeMessage() method returns the message without any alteration.
The generator template that defines how a class generated for a Script should look like, now has to generate a extra method that will override the customizeMessage() method in KajaFrame:
data:image/s3,"s3://crabby-images/f4bdf/f4bdf35c3d5426e5db6df57408026202c6da8789" alt="cf13.png cf13.png"
The overriding method only gets generated when the concept function exists in the Script. The generator uses the body of myFunction as a body of the generated customizeMessage() method.
data:image/s3,"s3://crabby-images/08adc/08adcd38f538f644073bb26c3f524245c12d69ae" alt="cf14.png cf14.png"
Now the concept function for customizing trace messages should be fully functional:
data:image/s3,"s3://crabby-images/9c123/9c123880c5d81ff2da6ecf3ee0c30fe8daa9f399" alt="cf5.png cf5.png"
Thanks for your feedback!