KIML
This exercise will introduce the usage of the Eclipse Plugin Development Environment for developing new layout algorithms to be used in Eclipse diagram editors. Replace each <login> by your own login name (e.g. msp), and each <Login> by your login name with capitalized first letter (e.g. Msp). For any questions contact msp.
- Install and set up Eclipse
- Check out the following KIELER plugins:
- de.cau.cs.kieler.core
- de.cau.cs.kieler.core.kgraph
- de.cau.cs.kieler.core.kgraph.edit
- de.cau.cs.kieler.core.ui
- de.cau.cs.kieler.keg
- de.cau.cs.kieler.keg.diagram
- de.cau.cs.kieler.kiml.layout
- de.cau.cs.kieler.kiml.ui
- de.cau.cs.kieler.kiml.viewer
- de.cau.cs.kieler.klay.layered
- Create a new plugin
- File -> New -> Other... -> Plug-in Development -> Plug-in Project
- Project name: de.cau.cs.rtprak.<login>.exercise1
- Next -> set version to 0.1.0.qualifier, provider to Christian-Albrechts-Universität zu Kiel, and execution environment to J2SE-1.5 (do this for all plugins that you create!)
- The checkboxes in the Options group can be deactivated -> Finish
- Check in the new plugin project into https://rtsys.informatik.uni-kiel.de/svn/teaching/prak/10ss-layout/exercises/<login>/
- Activate Checkstyle: right-click the project -> Properties -> Checkstyle -> Checkstyle active for this project
- Open the file META-INF/MANIFEST.MF -> Dependencies tab
- Add the plugins de.cau.cs.kieler.core and de.cau.cs.kieler.kiml.layout to the list of dependencies, then save the file
- Create a layout provider class with the New -> Class wizard
- Package: de.cau.cs.rtprak.<login>.exercise1
- Name: <Login>LayoutProvider
- Superclass: de.cau.cs.kieler.kiml.AbstractLayoutProvider
Implement the layout provider class
- You should now see a checkstyle warning for the doLayout method
- Add the keyword final to each method argument to eliminate the warning
- Add the following constant to the class:
/** default value for spacing between nodes. */privatestaticfinalfloat=15.0f;
DEFAULT_SPACING
- Write the following lines at the beginning of the doLayout method (see IKielerProgressMonitor and KimlLayoutUtil):
.begin("<Login> Layouter",1);=.getData(.class);float=.getProperty(.SPACING);if(<0){=;}float=.getProperty(.BORDER_SPACING);if(<0){=;}
progressMonitor KShapeLayout parentLayout layoutNodeKShapeLayout objectSpacing parentLayoutLayoutOptions objectSpacing objectSpacing DEFAULT_SPACING borderSpacing parentLayoutLayoutOptions borderSpacing borderSpacing DEFAULT_SPACING
- Write the following line at the end of the doLayout method:
.done();
progressMonitor
Implement the rest of the layouter such that the nodes of the input graph are all put in a row
- See the KGraph and KLayoutData data structures: the input is a KNode and holds the nodes of the graph in its list of children
- Iterate over the nodes in the getChildren() list of the layoutNode input
- Retrieve the size of a node using the following code:
=.getData(.class);float=.getWidth();float=.getHeight();
KShapeLayout nodeLayout nodeKShapeLayout width nodeLayout height nodeLayout
- Set the position (x, y) of a node's upper left corner using the following code:
.setXpos();.setYpos();
nodeLayoutx nodeLayouty
- objectSpacing shall be the spacing to be left between each pair of nodes
- borderSpacing shall be the spacing to be left to the borders of the drawing: the first node's coordinates shall be (borderSpacing, borderSpacing)
- At the end of the method, set the width and height of parentLayout so that it is large enough to hold the whole drawing, including borders
- Edges may be ignored for now
- You should now see a checkstyle warning for the doLayout method
- Open the file META-INF/MANIFEST.MF -> Extensions tab
- Add an extension for de.cau.cs.kieler.kiml.layout.layoutProviders
- Right-click the extension -> New -> layoutProvider
- Set name to <Login> Test Layouter, class to de.cau.cs.rtprak.<login>.exercise1.<Login>LayoutProvider
- Right-click the new layoutProvider -> New -> knownOption, set option to de.cau.cs.kieler.layout.options.minSpacing
- Add another knownOption, set to de.cau.cs.kieler.layout.options.borderSpacing
- Run -> Run Configurations... -> right-click Eclipse Application -> New
- Name: Layout
- For testing the layouter, a new workspace location will be created; you may configure its destination in Workspace Data -> Location
- Add the program arguments -debug -consoleLog in the Arguments tab
- Go to Plug-ins tab, select Launch with: plug-ins selected below only
- Deselect All, activate Workspace checkbox, Add Required Plug-ins, Apply, Run
- Test the layouter in the new Eclipse instance:
- New -> Project... -> General -> Project, name test
- Right-click test project -> New -> Other... -> Graphs Diagram
- Create a graph using the palette on the right
- Window -> Show View -> Other... -> KIELER -> Layout
- While the graph diagram is open, set Layout Provider or Type in the Layout view to <Login> Test Layouter
- Open the additional views Layout Graph and Layout Time
- Trigger layout with KIELER -> Layout or Ctrl+R L (first Ctrl+R, then L) or the button in the toolbar
- See the direct input and output of your algorithm in the Layout Graph view: Pre-Layout is the input, Post-Layout is the output
- See the execution time analysis in the Layout Time view
Implement another class EdgeRouter with superclass de.cau.cs.kieler.core.alg.AbstractAlgorithm
- Add the following method:
/** * Route the edges that are connected with the children of the given node. * * @param parentNode the parent node of the input graph */publicvoidrouteEdges(final){().begin("Edge Routing",1);().done();}
KNode parentNode getMonitor getMonitor
- Add the following code to the end of the doLayout method in your layout provider:
=new();.reset(.subTask(1));.routeEdges();
EdgeRouter edgeRouter EdgeRouter edgeRouterprogressMonitor edgeRouterlayoutNode
Implement the routeEdges method:
- Each edge shall be drawn with three line segments: one vertical segment starting below the source node, one horizonzal segment, and another vertical segment ending below the target node.
- The horizontal segments of two different edges shall not have the same y-coordinate; for consecutive edges, the distance between their horizontal segments shall equal objectSpacing
- See the attached image test-drawing.png for an example
- Find the edges using getOutgoingEdges() or getIncomingEdges() on a node
- Get the edge layout of an edge to set bend points using this code:
=.getData(.class);
KEdgeLayout edgeLayout edgeKEdgeLayout
- Create a bend point using this code:
=.eINSTANCE.createKPoint();
KPoint point KLayoutDataFactory
- Use the getBendPoints() list on the edgeLayout to add bend points (clear the list first to remove points from the previous layout)
- Set the values of the points returned by getSourcePoint() and getTargetPoint() according to the positions where the edge leaves its source node and reches its target node
- Add the following method:
- Use your previous run configuration to test the edge router.