The NetBeans Platform is a powerful Rapid Application Development tool. It provides a solid framework for loose-coupling via the Lookup class and it comes with a robust modular development environment where dependencies must be explicit and well-defined. In addition, its GUI Builder drastically reduces time spent designing Swing components while the Window System provides some pretty awesome standard features, such as tabs, drag-and-drop window placement, and individual window min-/maximization.
However there is a major shortcoming in the platform. This problem is not a bug or an implementation issue (although there are certainly plenty of those in the platform), rather it is a glaring design problem which is unforgivably inconsistent with the dev-team's dual-mantras of loose-coupling and modular development. I'm talking about their actions paradigm.
Let's say you that want to create an action that will open a new TopComponent when you double click on a
However there is a major shortcoming in the platform. This problem is not a bug or an implementation issue (although there are certainly plenty of those in the platform), rather it is a glaring design problem which is unforgivably inconsistent with the dev-team's dual-mantras of loose-coupling and modular development. I'm talking about their actions paradigm.
Let's say you that want to create an action that will open a new TopComponent when you double click on a
ProductNode
. According to every tutorial/example about actions that I've come across, you implement such an action like so:
package com.potetm.actions;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import org.openide.awt.ActionRegistration;
import org.openide.awt.ActionReference;
import org.openide.awt.ActionReferences;
import org.openide.awt.ActionID;
import org.openide.util.NbBundle.Messages;
@ActionID(category = "File", id = "com.potetm.actions.OpenWindowAction")
@ActionRegistration(displayName = "#CTL_OpenWindowAction")
@ActionReferences({
@ActionReference(path = "Toolbars/File", position = 0)
})
@Messages("CTL_OpenWindowAction="Open Window")
public final class OpenWindowAction implements ActionListener {
public void actionPerformed(ActionEvent e) {
WindowManager.getDefault().findTopComponent("OtherWindowTopComponent").open();
}
}
getPreferredAction()
method in your ProductNode
.The problem with this implementation is that
OpenWindowAction
, which OtherWindowTopComponent
neither knows about nor cares about, is now telling OtherWindowTopComponent
when it should open. Now, to be fair, while this is never ideal, the current OpenWindowAction
action might suffice for this trivial example. After all, it's calling a method common to all TopComponents meaning it doesn't know anything at all about this particular TopComponent. Not only that, but it doesn't even have to explicitly depend on OtherWindowTopComponent.class
due to the fact that WindowManger.findTopComponent(...)
only takes a string. (Some might argue that that is, in itself, a problem, but for the sake of argument, I'll list it as an advantage.)However, this whole line of reasoning falls apart as soon as you want an action to do anything more complicated than simply opening a window. Suppose, for example, that you also want this newly opened window to display detailed information about
ProductNode
's Product
. The easiest solution under the current paradigm is to create an OtherWindowTopComponent.setProduct(Product)
method, which is essentially what is recommended in The NetBeans Platform 6.9 Developer's Guide. (Scroll down to the section on "Creating Global Actions".) Now your actionPerformed(...)
method looks like this:public void actionPerformed(ActionEvent e) {
Product product = ((Node) e.getSource()).getLookup().lookup(Product.class);
OtherWindowTopComponent otherWindowTopComponent =
(OtherWindowTopComponent) WindowManager.getDefault().findTopComponent("OtherWindowTopComponent");
otherWindowTopComponent.open();
otherWindowTopComponent.setProduct(product);
}
Now that's significantly more intrusive, and yet this example is still relatively trivial. It's not difficult to see how an action which requires many different objects working in tandem would turn into a God Object in every sense of the term.I found this truly incredible. Especially considering the fact that every tutorial, video, and how-to on the merits of the NetBeans Platform includes a lengthy discussion on loose coupling and modularity. It is undoubtedly true that these two attributes lead to robust, easily maintainable code. The tragic thing is that actions in NetBeans are massive violations of those principles. They violate the principles of modularity because any module that contains actions could potentially be forced to depend on every other module due soley to the action class. Even if you put every action in the application into a single module (which is itself a poor idea), that single module would inevitably depend on every other module for one reason or another. While this is allowed in the NetBeans Platform (as long as no module depended on the actions module), it is a truly terrible idea. It's one thing to have every module in an application depend on a single module. That just means that that module provides a particular service which every other module happens to need. However, if the roles were reversed, if one module were to depend on every other, that would mean that there were no clear relationships between that module and the others. It would seem (and would indeed be true) that that module just started depending on other modules willy-nilly whenever it was convenient.
NetBeans actions' violation of the principles of loose-coupling is even more severe. The loose-coupling paradigm states that each class should explicitly depend on as few other classes as possible. More specifically, as it applies to the NetBeans Platform, that means that you use Lookups to transfer information from one class to another instead of explicitly calling a particular method in another class. The act of changing a Lookup notifies any interested parties of the change without either party explicitly knowing anything about the other. You could, for example, take any TopComponent out of the application and it would still run, just without the functionality that the missing TopComponent provided. The same is not true for actions. An action is notified of an event, and then starts explicitly calling methods on every other class that it needs to complete its action. If you take any TopComponent that it was using out of the application, not only would the action be broken, it wouldn't compile.
This poor programming structure is further exacerbated by the fact that NetBeans manages the lifecycles of the actions which it creates and their lifespans are extremely short. If it were possible to manually manage the lifecycle of an action, you could instantiate it on application start and immediately set up an observer pattern, with interested parties being notified by the action when a hotkey or button is pressed. Unfortunately, given the circumstances I have been forced to hack my way to good programming practice.
My solution was to set up an observer pattern with a class that can be looked up via the
@ServiceProvider
annotation. The NetBeans-managed action would then notify this class when an action has occured, which would in turn message every party signed up to be notified. See the diagram below:The two classes in the dotted-box are essentially two objects doing the job of one. The
ActionProvider
serves as the "observable action" for the application. It will inform interested parties when a hotkey/button has been pressed. It in turn is informed by the ActionDelegate
, which is instantiated and managed by the NetBeans Platform. ActionDelegate
also has all of the annotations necessary to make the action show up in menus, be called for hotkeys, etc.I actually simplified the whole operation by making an
AbstractActionDelegate
and an AbstractActionProvider
which do all the work for every action. The AbstractActionDelegate
uses reflection to look up its corresponding ActionProvider
. I'm not a huge fan of reflection, however this also enforces a naming convention. I also made the actionPerformed(...) method on both classes protected
so that I could pass on the original ActionEvent
from the NetBeans Platform if necessary*.I plan on uploading the code for those classes to my website pretty soon. I'll update this post with a link when I do.
------------------------------------------------------------------------------------------
*Ordinarily you want to make an
ActionEvent
's source object be the ActionProvider
from which it came. This is so that your ActionListeners
can distinguish between actions (because you don't know what object NetBeans will list as the event source). However, especially in the case of actions triggered by Nodes
, you need the Node
to be the event source because the node contains the object you want in its lookup.EDIT: I finally uploaded the source code to github.
No comments:
Post a Comment