Node Storage concerns how the bounds (bounding box) of nodes are stored during the execution of the DiagramRenderer#draw
method. This is a performance feature to avoid the inefficiency of continually re-computing the bounds of nodes during a single diagram rendering pass. The principle behind the Node Storage feature is that the bounding box for a node should only be computed once for a given diagram rendering pass. Because bounding boxes are used in many calculations for node and edge placement, it is necessary to cache the computed value to avoid repeatedly recomputing it.
The feature is supported by a NodeStorage
object that maps individual nodes to their corresponding bounds Rectangle
. NodeStorage
objects are managed by node renderers. In this way, a given node renderer, say NoteNodeRenderer
, will cache the bounds for all the NoteNode
s of a given diagram.
The following class diagram illustrates how the design is realized with the illustration of a NoteNode
(a node that represents a note on a UML diagram).
DiagramRenderer
declares two methods to manage node caching. activateNodeStorages()
prepares and activates the cache by going through all its viewers and activating their cache. It should be called once before a diagram is drawn. deactivateAndClearNodeStorages()
clears the cache by going through all the viewers and clearing their cache. It should be called once after a diagram is drawn.getBounds(DiagramElement)
of interface DiagramElementRenderer
is implemented by a final method in AbstractNodeRenderer
that delegates the call to getBounds
of class NodeStorage
. This call takes as second argument a Function<Node,Rectangle>
that is the bounds calculator. This function object is use to compute the bounds in case of a cache miss. The design of the bounds calculation mechanism is described below.The following sequence diagram illustrates a scenario where a diagram that contains a single NoteNode
is drawn.
DiagramRenderer
aggregates many NodeVRenderer
objects, so the call to activateNodeStorage()
will be repeated once for each element renderer managed by the diagram renderer.DiagramRenderer#draw
eventually reaches NoteNodeRenderer#getBounds
, because this information is used to draw the node. This results in a call to NodeStorage#getBounds
where the second argument (::internalGetBounds
) is a reference to a instance method of a particular object, namely method internalGetBounds
of the NoteNodeRenderer
object.internalGetBounds
is called to compute and return the bounds.The NodeStorage
class is responsible for caching the bounds of different nodes, and computing these bounds for any node not in the cache. This computation requires specialized algorithms to compute the bounds of different nodes. To make it possible to reuse the NodeStorage
class for all different types of nodes, its method getBounds
takes as second argument a bounds calculator function object of type Function<Node, Rectangle>
. The concrete bounds calculators are implemented as protected methods of the NodeRenderer
subclasses that implement the abstract method AbstractNodeRenderer#internalGetBounds
. For example, in the class diagram above, NoteNodeRenderer
implements internalGetBounds
with the code to compute the bounds of NodeNode
s. With this design in place, the implementation of AbstractNodeRenderer#getBounds
is greatly simplified as:
@Override
public final Rectangle getBounds(Node pNode)
{
return aNodeStorage.getBounds(pNode, this::internalGetBounds);
}
Here, the expression this::internalGetBounds
is a reference to method internalGetBounds
of the NodeRenderer
object upon which getBounds
gets called. As illustrated in the sequence diagram, when there is a cache miss, this method will be the one called to compute the node’s bounds.