codeEEmoji is a plugin made for Intellij Idea and useful for Java programming. The plugin defines new sets of inlay hints in the context of code augmentation. The new inlay hints use emojis in an innovative way to help the developer. Emojis are displayed for anti-pattern cases such as naming violations or bad programming practices. The plugin can also show emojis in the editor indicating the presence of modifiers for classes, fields, methods, parameters or local variables that are being used. Likewise, the developer can indicate emojis to display for all these elements, according to a rule, for example, for a class being instantiated that implements a specific interface, for a method being invoked that is annotated by a certain annotation , a local variable of a given type, and so on. Furthermore, the plugin has the capability to display implicit annotations for the JakartaEE and Spring frameworks.
- Setup and Use
- Cases of Naming Violation
- Short Descriptive Name
- Getter More Than Accessor
- Is Returns More Than a Boolean
- Setter Method Returns
- Expecting But Not Getting a Single Instance
- Validation Method Does Not Confirm
- Getter Does Not Return
- Not Answered Question
- Transform Method Does Not Return
- Expecting But Not Getting a Collection
- Says One But Contains Many
- Name Suggests Boolean By Type Does Not
- Says Many But Contains One
- Name Contains Only Special Characters
- Cases of Showing Modifiers
- Cases of Showing Specifics of Projects
- Cases of Showing Implicit Annotations
- External Services API
- How to Extend
- Acknowledgements
- References
- IntelliJ IDEA 2023.* (Ultimate, Community or Educational)
- JDK 17 (Eclipse Temurin - as a suggestion)
JetBrains Marketplace - codEEmoji
- Download the latest release zip file available from Releases.
- Open Intellij IDEA and navigate to "File>Settings>Plugins" menu. Click on the gear icon and the "* *Install Plugin From Disk..." option. Select the downloaded zip file and click "OK**".
- Restart IDE.
The plugin creates new Inlay hints. All new inlay hints are enabled by default when installing the plugin. To disable inlay hints or configure options that are available for each one, go to "File>Settings>Editor>Inlay Hints". Click "Other>Java".
Instead of a descriptive name, the variable's name consists of a few letters.
Configuration options: Number of letters
Impacted identifiers: Fields, Method Parameters and Local Variables
A getter that doesn't just return the corresponding attribute but also takes other actions. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
A method's name is a predicate that denotes a true/false value that will be returned. The return type, however, is a more complex type than boolean. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
A setter method that has a return type other than void. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
Despite the fact that a method's name suggests it will return a single object, it will actually return a collection. Adapted from Arnaoudova et al.( 2016).
Impacted identifiers: Method Names
A validation method (such as one with the words validate, check, or ensure) does not confirm the validation; that is, it neither provides a return value indicating whether the validation was successful. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
When a method's name begins with get or return, for example, it might be assumed that it returns something, but the return type is actually void. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
A method's name takes the form of a predicate, but its return type is not boolean. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
While there is no return value, a method's name implies that an object has been transformed. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Method Names
Even though a method's name suggests a collection should be returned, nothing or just one object is instead given. Adapted from Arnaoudova et al.( 2016).
Impacted identifiers: Method Names
An attribute's type suggests that it stores a collection of objects, contrary to the name, which suggests a single instance. Adapted from Peruma et al.(2021).
Impacted identifiers: Fields, Method Parameters and Local Variables
An attribute's name implies that its value is true or false, yet its defining type is not boolean. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Fields, Method Parameters and Local Variables
The name of an attribute suggests multiple instances, but its type suggests a single one. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Fields, Method Parameters and Local Variables
The identifier's name is made up entirely of non-alphanumeric characters. Adapted from Arnaoudova et al.(2016).
Impacted identifiers: Fields, Method Parameters and Local Variables
This inlay hint allows you to configure the display of emojis for class, field and method modifiers. Emojis are displayed when an element is used in the code, indicating its modifiers. The figure below shows the configuration screen with the options enabled during installation. Then a code snippet is displayed, where all options have been enabled.
Impacted identifiers: Classes, Fields and Methods
This inlay hint is displayed according to the specifics of the project. It must be configured by the developer from a file in the root of the project named "codeemoji.json". It allows indicating rules for displaying emojis according to specific features for each element, as follows:
- Element: Class
- Features: Annotations, Extends and Implements
- Element: Field
- Features: Annotations, Types
- Element: Method
- Features: Annotations, Returns
- Element: Parameter
- Features: Annotations, Types
- Element: Local Variable
- Features: Annotations, Types
For each element, it is possible to optionally indicate an emoji from the unicode sequence that represents it. The " codeemoji.json" file follows a simple description pattern, as shown in the following a partial example:
Complete example here.
In the configuration screen of this inlay hint, the rules currently defined for the open project are displayed. See the following figure.
Here's an example of usage from a code snipped:
Unlike other cases, this inlay hint is useful for displaying implicit annotations when frameworks are used. It can be displayed for an entirely implicit annotation, for implicit annotation parameters, or even for implicit values for annotation parameters.
Implicit information is understood to be information that is optional, but with a different standard value. For example, a class mapped with @Entity will have @Column implicit for almost all fields (restrictions apply), where the name parameter of the annotation will have the name of the field as its value.
So far the plugin implements some JPA mapping annotations (Java Persistence API - javax and jakarta packages), and some useful Spring framework annotations.
In the configuration screen of this inlay hint, the annotations currently implemented by plugin are displayed. See the following figure.
Here's an example of usage with JPA from a code snipped:
Here's an examples of usage with Spring from a code snipped:
The codeEEmoji plugin is prepared to work with information provided from external services. It provides extension points for creating background services that can obtain information about a source code element for which the insertion of an inlay hint is being evaluated.
This API is experimental and the plugin currently does not contain any concrete services that use it. However, for future work it may be useful for cases of inlay hints that involve external services such as code versioners, quality analyzers, artificial intelligence tools for code prediction, among others.
codeEEmoji is already prepared to enable or disable these services, as they can reduce the performance of the IDE, as they are transversal to the framework. Therefore, it is up to the user to use these services or not. Services can be configured using the plugin's global settings. See Figure below.
The codEEmoji plugin is under to the terms and conditions of the GNU General Public License version 3.0. The source code can be accessed on the GitHub platform, whereby the repository contains two distinct branches: "main" for official releases and "develop" for ongoing development activities. The software is extensively developed using the IntelliJ Platform Plugin SDK. For individuals without familiarity with the process of developing plugins for the IntelliJ IDEA IDE, it is recommended to visit the official website in order to acquire a foundational understanding of the subject. An appropriate first reference for individuals with further expertise would be the section pertaining to Inlay hints. It is noteworthy to acknowledge that the inclusion of the Inlay hints feature represents a recent addition to the IntelliJ IDEA IDE. Additionally, it is worth mentioning that the codEEmoji plugin was created utilizing numerous APIs that have been designated as @Experimental. The APIs have the potential to undergo modifications in the future, which could result in compatibility challenges. Nevertheless, the present condition of the plugin indicates that it is entirely compatible with versions 2023.1.x and 2023.2.x, provided that JDK 17 or a higher version is utilized.
As mentioned in the reference page for implementing Inlay hints in the IDE, "Inlay hints render small pieces of information directly into the editor and give developers additional code insight without disturbing the workflow. A well-known example is parameter hints that usually display the name of the function parameters as given in its declaration". Inlay hints can be of the type inline (inlays displayed in the code between code tokens) or block (inlays displayed above a code block) and must be implemented by a provider class that is registered in the plugin configuration file. All cases implemented in the codEEmoji plugin are inlay hints that extend or implement the interface InlayHintsProvider.
The codEEmoji plugin provides two abstract classes (CEProvider and CEProviderMulti) that implement the interface InlayHintsProvider. They are starting points for implementing a case for adding an inlay hint. See figure that follow.
As can be seen in the diagram above, a class of type InlayHintProvider must implement a series of methods which will allow you to configure the inlay hint case in the IDE, allowing you to configure it in the appropriate menus for a programming language. The most important point concerns the getCollectorFor(PsiFile, Editor, S, InlayHintsSink) method. This method returns a class of type InlayHintsCollector. This class that is responsible for scanning the source code elements and execute the logic to identify the point where the addition of an inlay hint is necessary.
Generally a class of type InlayHintProvider is linked to a class of type InlayHintCollector. At this point, the class CEProviderMulti differentiates itself. It extends the CEProvider class and allows a list of coupled collectors.
The codEEmoji provides a rich API for implementing classes of type InlayHintsCollector. The figure below displays the class diagram available for this purpose.
The InlayHintsCollector framework interface can be implemented in the plugin by the CECollector and CECollectorMulti. The interface defines the collect(PsiElement, Editor, InlayHintsSink) method. CECollectorMulti allows you to implement this method using a list of collectors, useful for use with CEProviderMulti.
The CECollector abstract class is the main class for implementing a collector. It extends the abstract class CEInlayBuilder which contains all the methods for manipulating inlay hints. Child classes must implement processCollect(PsiElement, Editor, InlayHintsSink).
The CECollector has three main extensions: CESimpleCollector, CEProjectCollector and CEImplicitCollector. The first is for general use. The other two extensions are specific cases for working with implicit annotations and specifications of projects. The following sections explore each case.
Child classes of CESimpleCollector must implement the needsHint(H, Map) method. This method is parameterized by type of element to be analyzed and answers whether it is necessary to add an inlay hint according to the logic of the concrete child class.
The codEEmoji provides six abstract classes that directly extend CESimpleCollector that allow you to abstract the collect of elements of type Class, Method and Variable (Field, Parameter and Local Variable), in addition to elements that are referenced in the source code: Reference Class, Reference Field and Reference Method.
Example of use:
public class GetMethodDoesNotReturn extends CEProvider<NoSettings> {
@Override
public String getPreviewText() {
return """
public class Customer {
public void getName() {
doSomething();
}
}""";
}
@Override
public @NotNull InlayHintsCollector buildCollector(@NotNull Editor editor) {
return new CEMethodCollector(editor, getKeyId(), CONFUSED) {
@Override
public boolean needsHint(@NotNull PsiMethod element, @NotNull Map<?, ?> externalInfo) {
return (element.getName().startsWith("get")
|| element.getName().startsWith("return"))
&& Objects.equals(element.getReturnType(), PsiTypes.voidType());
}
};
}
}
For implementations of reference type elements, the plugin already provides abstract classes to work with references to modifiers in classes, methods, fields, and interface methods.
Example of use:
public class ShowingModifiers extends CEProviderMulti<ShowingModifiersSettings> {
//source code omitted...
@Override
public @NotNull List<InlayHintsCollector> buildCollectors(@NotNull Editor editor) {
List<InlayHintsCollector> list = new ArrayList<>();
//fields
list.addAll(
Arrays.asList(
new CEModifierFieldCollector(editor, getKeyId(), PUBLIC_SYMBOL, PUBLIC, getSettings().query(PUBLIC_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), DEFAULT_SYMBOL, DEFAULT, getSettings().query(DEFAULT_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), FINAL_VAR_SYMBOL, FINAL, getSettings().query(FINAL_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), PROTECTED_SYMBOL, PROTECTED, getSettings().query(PROTECTED_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), PRIVATE_SYMBOL, PRIVATE, getSettings().query(PRIVATE_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), STATIC_SYMBOL, STATIC, getSettings().query(STATIC_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), VOLATILE_SYMBOL, VOLATILE, getSettings().query(VOLATILE_FIELD)),
new CEModifierFieldCollector(editor, getKeyId(), TRANSIENT_SYMBOL, TRANSIENT, getSettings().query(TRANSIENT_FIELD))
)
);
//source code omitted...
return list;
}
}
The plugin ensures that before invoking the needsHint(H, Map) method, the element under collect analysis is sent to everyone analyzers with external services using processExternalInfo(H) method. To do this, a Map is passed to fill in the respective information for each service using a CEExternalAnalyzer singleton and the list of implementations of CEEExternalService interface. This way, in the concrete child class, the needsHint(H, Map) method receives a Map with information that can subsidize the addition or not of an inlay hint.
CEProjectCollector provides ways to work with a specific case of collector. It allows you to collect elements configurable according to the codeemoji.json configuration file directly in the user's project. The plugin already provides classes for collecting elements of type Class, Method and Variable. The diagram below shows these collectors and displays the available interfaces.
Example of use:
public class ShowingSpecifics extends CEProviderMulti<ShowingSpecificsSettings> {
//source code omitted...
@Override
public @NotNull List<InlayHintsCollector> buildCollectors(@NotNull Editor editor) {
List<InlayHintsCollector> list = new ArrayList<>();
list.add(new CEProjectClassCollector(editor, getKeyId()));
list.add(new CEProjectMethodCollector(editor, getKeyId()));
list.add(new CEProjectVariableCollector(editor, FIELD, getKeyId()));
list.add(new CEProjectVariableCollector(editor, PARAMETER, getKeyId()));
list.add(new CEProjectVariableCollector(editor, LOCALVARIABLE, getKeyId()));
return list;
}
@Override
public @NotNull ImmediateConfigurable createConfigurable(@NotNull ShowingSpecificsSettings settings) {
return new ShowingSpecificsConfigurable(settings);
}
}
CEImplicitCollector is also a specific type of collector in the plugin. It allows you to detect the need for implicit annotations hints to be added via block-type inlay hint. It also allows you to detect implicit annotation attributes as well as their respective implicit values.
The plugin already provides implementation for several annotations of the JPA and Spring, being fully extensible to other frameworks. In the diagram below, an example of JPA with @Entity and @Column. CEJPAEntityCollector defines a collector to work with the main annotation @Entity. The collector has a list of classes that implement the CEImplicit interface. This way, each item in this list is invoked to investigate whether to process an inlay hint.
Example of use:
public class ImplicitAnnotations extends CEProviderMulti<ImplicitAnnotationsSettings> {
//source code omitted...
@Override
public @NotNull List<InlayHintsCollector> buildCollectors(@NotNull Editor editor) {
final int codePoint = 0x1F4AD;
String keyId = getKeyId();
return new ArrayList<>(
Arrays.asList(
new CEJPAEntityCollector(editor, keyId, codePoint, "javax.persistence"),
new CEJPAEntityCollector(editor, keyId, codePoint, "jakarta.persistence"),
new CEJPAEmbeddableCollector(editor, keyId, codePoint, "javax.persistence"),
new CEJPAEmbeddableCollector(editor, keyId, codePoint, "jakarta.persistence"),
new CESpringConfigurationCollector(editor, keyId, codePoint),
new CESpringControllerCollector(editor, keyId, codePoint),
new CESpringRestControllerCollector(editor, keyId, codePoint)
));
}
//source code omitted...
}
The plugin is fully ready for internationalization. The CEBundle.properties bundle is provided. All strings for tooltips, settings, warnings, information and exceptions can be internationalized centrally. At the moment, all these messages are in the English language. To extend it to another language, simply create a file named with CEBundle_XX.properties, where XX stands for the language acronym. The IDE will automatically detect the language used and carry out the appropriate configuration.
This work was supported by the Free University of Bozen-Bolzano - UNIBZ.
Arnaoudova, Venera, Massimiliano Di Penta, and Giuliano Antoniol. "Linguistic antipatterns: What they are and how mainers perceive them." Empirical Software Engineering 21 (2016): 104-158.