]> gerrit.simantics Code Review - simantics/platform.git/blob - bundles/org.simantics.diagram/src/org/simantics/diagram/elements/MonitorClass.java
Sync git svn branch with SVN repository r33269.
[simantics/platform.git] / bundles / org.simantics.diagram / src / org / simantics / diagram / elements / MonitorClass.java
1 /*******************************************************************************\r
2  * Copyright (c) 2007, 2010 Association for Decentralized Information Management\r
3  * in Industry THTH ry.\r
4  * All rights reserved. This program and the accompanying materials\r
5  * are made available under the terms of the Eclipse Public License v1.0\r
6  * which accompanies this distribution, and is available at\r
7  * http://www.eclipse.org/legal/epl-v10.html\r
8  *\r
9  * Contributors:\r
10  *     VTT Technical Research Centre of Finland - initial API and implementation\r
11  *******************************************************************************/\r
12 package org.simantics.diagram.elements;\r
13 \r
14 import java.awt.BasicStroke;\r
15 import java.awt.Color;\r
16 import java.awt.Font;\r
17 import java.awt.FontMetrics;\r
18 import java.awt.Graphics2D;\r
19 import java.awt.Rectangle;\r
20 import java.awt.Shape;\r
21 import java.awt.geom.AffineTransform;\r
22 import java.awt.geom.Path2D;\r
23 import java.awt.geom.Point2D;\r
24 import java.awt.geom.Rectangle2D;\r
25 import java.util.EnumSet;\r
26 import java.util.Map;\r
27 \r
28 import javax.vecmath.Vector2d;\r
29 \r
30 import org.simantics.db.layer0.variable.RVI;\r
31 import org.simantics.g2d.diagram.IDiagram;\r
32 import org.simantics.g2d.element.ElementClass;\r
33 import org.simantics.g2d.element.ElementHints;\r
34 import org.simantics.g2d.element.ElementUtils;\r
35 import org.simantics.g2d.element.IElement;\r
36 import org.simantics.g2d.element.SceneGraphNodeKey;\r
37 import org.simantics.g2d.element.handler.ElementHandler;\r
38 import org.simantics.g2d.element.handler.FillColor;\r
39 import org.simantics.g2d.element.handler.InternalSize;\r
40 import org.simantics.g2d.element.handler.LifeCycle;\r
41 import org.simantics.g2d.element.handler.Move;\r
42 import org.simantics.g2d.element.handler.Outline;\r
43 import org.simantics.g2d.element.handler.Rotate;\r
44 import org.simantics.g2d.element.handler.Scale;\r
45 import org.simantics.g2d.element.handler.SceneGraph;\r
46 import org.simantics.g2d.element.handler.StaticSymbol;\r
47 import org.simantics.g2d.element.handler.Text;\r
48 import org.simantics.g2d.element.handler.TextEditor;\r
49 import org.simantics.g2d.element.handler.TextEditor.Modifier;\r
50 import org.simantics.g2d.element.handler.Transform;\r
51 import org.simantics.g2d.element.handler.impl.BorderColorImpl;\r
52 import org.simantics.g2d.element.handler.impl.FillColorImpl;\r
53 import org.simantics.g2d.element.handler.impl.ParentImpl;\r
54 import org.simantics.g2d.element.handler.impl.SimpleElementLayers;\r
55 import org.simantics.g2d.element.handler.impl.StaticSymbolImpl;\r
56 import org.simantics.g2d.element.handler.impl.TextColorImpl;\r
57 import org.simantics.g2d.element.handler.impl.TextEditorImpl;\r
58 import org.simantics.g2d.element.handler.impl.TextFontImpl;\r
59 import org.simantics.g2d.element.handler.impl.TextImpl;\r
60 import org.simantics.g2d.elementclass.MonitorHandler;\r
61 import org.simantics.g2d.image.Image;\r
62 import org.simantics.g2d.image.ProviderUtils;\r
63 import org.simantics.g2d.image.impl.AbstractImage;\r
64 import org.simantics.g2d.utils.Alignment;\r
65 import org.simantics.scenegraph.Node;\r
66 import org.simantics.scenegraph.g2d.G2DParentNode;\r
67 import org.simantics.scenegraph.g2d.nodes.ShapeNode;\r
68 import org.simantics.scl.runtime.function.Function1;\r
69 import org.simantics.utils.datastructures.cache.IFactory;\r
70 import org.simantics.utils.datastructures.cache.IProvider;\r
71 import org.simantics.utils.datastructures.cache.ProvisionException;\r
72 import org.simantics.utils.datastructures.hints.IHintContext.Key;\r
73 import org.simantics.utils.datastructures.hints.IHintContext.KeyOf;\r
74 import org.simantics.utils.strings.format.MetricsFormat;\r
75 import org.simantics.utils.strings.format.MetricsFormatList;\r
76 \r
77 /**\r
78  * @author Tuukka Lehtonen\r
79  */\r
80 public class MonitorClass {\r
81 \r
82     static final Font        FONT                      = Font.decode("Helvetica 12");\r
83 \r
84     /**\r
85      * Back-end specific object describing the monitored component.\r
86      */\r
87     public static final Key  KEY_MONITOR_COMPONENT     = new KeyOf(Object.class, "MONITOR_COMPONENT");\r
88 \r
89     /**\r
90      * Existence of this hint indicates that the monitor is showing a value that\r
91      * does not originate from the owner diagram.\r
92      */\r
93     public static final Key  KEY_MONITOR_IS_EXTERNAL   = new KeyOf(Boolean.class, "MONITOR_IS_EXTERNAL");\r
94 \r
95     /**\r
96      * The valuation suffix string describing the monitored variable of the\r
97      * component described by {@link #KEY_MONITOR_COMPONENT}.\r
98      */\r
99     public static final Key  KEY_MONITOR_SUFFIX        = new KeyOf(String.class, "MONITOR_SUFFIX");\r
100 \r
101     public static final Key  KEY_MONITOR_SUBSTITUTIONS = new KeyOf(Map.class, "MONITOR_SUBSTITUTIONS");\r
102     public static final Key  KEY_MONITOR_GC            = new KeyOf(Graphics2D.class, "MONITOR_GC");\r
103     public static final Key  KEY_MONITOR_HEIGHT        = new KeyOf(Double.class, "MONITOR_HEIGHT");\r
104     public static final Key  KEY_NUMBER_FORMAT         = new KeyOf(MetricsFormat.class, "NUMBER_FORMAT");\r
105     public static final Key  KEY_TOOLTIP_TEXT          = new KeyOf(String.class, "TOOLTIP_TEXT");\r
106 \r
107     public static final Key  KEY_EXPRESSION             = new KeyOf(String.class, "EXPRESSION");\r
108     public static final Key  KEY_INPUT_VALIDATOR             = new KeyOf(Object.class, "INPUT_VALIDATOR");\r
109     public static final Key  KEY_RVI             = new KeyOf(RVI.class, "RVI");\r
110 \r
111     /**\r
112      * If this hint is defined, the monitor will force its x-axis to match this\r
113      * angle. If this hint doesn't exist, the monitor will not force x-axis\r
114      * orientation.\r
115      */\r
116     public static final Key  KEY_DIRECTION             = new KeyOf(Double.class, "MONITOR_DIRECTION");\r
117 \r
118     public static final Key  KEY_BORDER_WIDTH          = new KeyOf(Double.class, "MONITOR_BORDER");\r
119 \r
120     public static final Key  KEY_SG_NODE               = new SceneGraphNodeKey(TextNode.class, "MONITOR_SG_NODE");\r
121     public static final Key  KEY_SG_NODE2              = new SceneGraphNodeKey(ShapeNode.class, "MONITOR_SG_NODE2");\r
122 \r
123     final static BasicStroke STROKE                    = new BasicStroke(1.0f);\r
124 \r
125     public final static Alignment DEFAULT_HORIZONTAL_ALIGN  = Alignment.CENTER;\r
126     public final static Alignment DEFAULT_VERTICAL_ALIGN    = Alignment.CENTER;\r
127     public final static MetricsFormat DEFAULT_NUMBER_FORMAT = MetricsFormatList.METRICS_DECIMAL;\r
128 \r
129     public final static Color     DEFAULT_FILL_COLOR        = new Color(224, 224, 224);\r
130     public final static Color     DEFAULT_BORDER_COLOR      = Color.BLACK;\r
131 \r
132     public final static double    DEFAULT_HORIZONTAL_MARGIN = 5.0;\r
133     public final static double    DEFAULT_VERTICAL_MARGIN   = 2.5;\r
134 \r
135     static Alignment getHorizontalAlignment(IElement e) {\r
136         return ElementUtils.getHintOrDefault(e, ElementHints.KEY_HORIZONTAL_ALIGN, DEFAULT_HORIZONTAL_ALIGN);\r
137     }\r
138 \r
139     static Alignment getVerticalAlignment(IElement e) {\r
140         return ElementUtils.getHintOrDefault(e, ElementHints.KEY_VERTICAL_ALIGN, DEFAULT_VERTICAL_ALIGN);\r
141     }\r
142 \r
143     static MetricsFormat getNumberFormat(IElement e) {\r
144         return ElementUtils.getHintOrDefault(e, KEY_NUMBER_FORMAT, DEFAULT_NUMBER_FORMAT);\r
145     }\r
146 \r
147     static void setNumberFormat(IElement e, MetricsFormat f) {\r
148         ElementUtils.setOrRemoveHint(e, KEY_NUMBER_FORMAT, f);\r
149     }\r
150 \r
151     static double getHorizontalMargin(IElement e) {\r
152         return DEFAULT_HORIZONTAL_MARGIN;\r
153     }\r
154 \r
155     static double getVerticalMargin(IElement e) {\r
156         return DEFAULT_VERTICAL_MARGIN;\r
157     }\r
158 \r
159     static Font getFont(IElement e) {\r
160         return ElementUtils.getHintOrDefault(e, ElementHints.KEY_FONT, FONT);\r
161     }\r
162 \r
163     public static class MonitorHandlerImpl implements MonitorHandler {\r
164         private static final long          serialVersionUID = -4258875745321808416L;\r
165         public static final MonitorHandler INSTANCE         = new MonitorHandlerImpl();\r
166     }\r
167 \r
168     static class Initializer implements LifeCycle {\r
169         private static final long serialVersionUID = 4404942036933073584L;\r
170 \r
171         IElement parentElement;\r
172         Map<String, String> substitutions;\r
173         Object component;\r
174         String suffix;\r
175         boolean hack;\r
176 \r
177         Initializer(IElement parentElement, Map<String, String> substitutions, Object component, String suffix, boolean hack) {\r
178             this.parentElement = parentElement;\r
179             this.substitutions = substitutions;\r
180             this.component = component;\r
181             this.suffix = suffix;\r
182             this.hack = hack;\r
183         }\r
184 \r
185         @Override\r
186         public void onElementActivated(IDiagram d, IElement e) {\r
187             if(!hack) {\r
188                 hack = true;\r
189 \r
190                 Point2D parentPos = ElementUtils.getPos(parentElement);\r
191                 Point2D thisPos = ElementUtils.getPos(e);\r
192 \r
193                 Move move = e.getElementClass().getSingleItem(Move.class);\r
194                 move.moveTo(e, thisPos.getX() - parentPos.getX(), thisPos.getY() - parentPos.getY());\r
195             }\r
196         }\r
197         @Override\r
198         public void onElementCreated(IElement e) {\r
199             if(parentElement != null) e.setHint(ElementHints.KEY_PARENT_ELEMENT, parentElement);\r
200             if(substitutions != null) e.setHint(KEY_MONITOR_SUBSTITUTIONS, substitutions);\r
201             if(component != null) e.setHint(KEY_MONITOR_COMPONENT, component);\r
202             if(suffix != null) e.setHint(KEY_MONITOR_SUFFIX, suffix);\r
203             e.setHint(KEY_DIRECTION, 0.0);\r
204             e.setHint(KEY_NUMBER_FORMAT, DEFAULT_NUMBER_FORMAT);\r
205             //e.setHint(KEY_HORIZONTAL_ALIGN, Alignment.LEADING);\r
206             //e.setHint(KEY_VERTICAL_ALIGN, Alignment.LEADING);\r
207         }\r
208 \r
209         @Override\r
210         public void onElementDestroyed(IElement e) {\r
211         }\r
212 \r
213         @Override\r
214         public void onElementDeactivated(IDiagram d, IElement e) {\r
215         }\r
216     };\r
217 \r
218     static String finalText(IElement e) {\r
219         String text = e.getElementClass().getSingleItem(Text.class).getText(e);\r
220         if (text == null)\r
221             return null;\r
222         return substitute(text, e);\r
223     }\r
224 \r
225     public static String editText(IElement e) {\r
226         return substitute("#v1", e);\r
227     }\r
228 \r
229     private static String formValue(IElement e) {\r
230         // TODO: consider using substitute\r
231         Map<String, String> substitutions = e.getHint(KEY_MONITOR_SUBSTITUTIONS);\r
232         if (substitutions != null) {\r
233             String value = substitutions.get("#v1");\r
234             if (substitutions.containsKey("#u1") && substitutions.get("#u1").length() > 0) {\r
235                 value += " " + substitutions.get("#u1");\r
236             }\r
237             return value;\r
238         } else {\r
239                 return ElementUtils.getText(e);\r
240         }\r
241     }\r
242 \r
243     static String substitute(String text, IElement e) {\r
244         Map<String, String> substitutions = e.getHint(KEY_MONITOR_SUBSTITUTIONS);\r
245         return substitute(text, substitutions);\r
246     }\r
247 \r
248     static String substitute(String text, Map<String, String> substitutions) {\r
249         if (substitutions != null) {\r
250             // TODO: slow as hell\r
251             for(Map.Entry<String, String> entry : substitutions.entrySet()) {\r
252                 if (entry.getValue() != null) {\r
253                     text = text.replace(entry.getKey(), entry.getValue());\r
254                 } else {\r
255                     text = text.replace(entry.getKey(), "<null>");\r
256                 }\r
257             }\r
258         }\r
259         return text;\r
260     }\r
261 \r
262     public static void update(IElement e) {\r
263         MonitorSGNode node = e.getElementClass().getSingleItem(MonitorSGNode.class);\r
264         node.update(e);\r
265     }\r
266 \r
267     public static void cleanup(IElement e) {\r
268         MonitorSGNode node = e.getElementClass().getSingleItem(MonitorSGNode.class);\r
269         node.cleanup(e);\r
270     }\r
271 \r
272     public static boolean hasModifier(IElement e) { \r
273         TextEditor editor = e.getElementClass().getAtMostOneItemOfClass(TextEditor.class);\r
274         Modifier modifier = editor != null ? editor.getModifier(e) : null;\r
275         return modifier != null;\r
276     }\r
277 \r
278     static final Rectangle2D DEFAULT_BOX = new Rectangle2D.Double(0, 0, 0, 0);\r
279 \r
280     static Shape createMonitor(IElement e) {\r
281         Alignment hAlign = getHorizontalAlignment(e);\r
282         Alignment vAlign = getVerticalAlignment(e);\r
283         double hMargin = getHorizontalMargin(e);\r
284         double vMargin = getVerticalMargin(e);\r
285 \r
286         String text = finalText(e);\r
287         if(text == null) {\r
288             return align(hMargin, vMargin, hAlign, vAlign, DEFAULT_BOX);\r
289         }\r
290 \r
291         Graphics2D g = e.getHint(KEY_MONITOR_GC);\r
292         if(g == null) {\r
293             return align(hMargin, vMargin, hAlign, vAlign, DEFAULT_BOX);\r
294         }\r
295 \r
296         Font f = getFont(e);\r
297         FontMetrics fm   = g.getFontMetrics(f);\r
298         Rectangle2D rect = fm.getStringBounds(text, g);\r
299 \r
300         return align(hMargin, vMargin, hAlign, vAlign, rect);\r
301     }\r
302 \r
303     static Shape align(double hMargin, double vMargin, Alignment hAlign, Alignment vAlign, Rectangle2D rect) {\r
304         //System.out.println("align: " + hMargin + ", " + vMargin + ", " + hAlign + ", " + vAlign + ": " + rect);\r
305         double tx = align(hMargin, hAlign, rect.getMinX(), rect.getMaxX());\r
306         double ty = align(vMargin, vAlign, rect.getMinY(), rect.getMaxY());\r
307         //System.out.println("    translate: " + tx + " "  + ty);\r
308         double nw = rect.getWidth() + 2*hMargin;\r
309         double nh = rect.getHeight() + 2*vMargin;\r
310         return makePath(tx + rect.getMinX(), ty + rect.getMinY(), nw, nh);\r
311     }\r
312 \r
313     static double align(double margin, Alignment align, double min, double max) {\r
314         double s = max - min;\r
315         switch (align) {\r
316             case LEADING:\r
317                 return -min;\r
318             case TRAILING:\r
319                 return -s - 2 * margin - min;\r
320             case CENTER:\r
321                 return -0.5 * s - margin - min;\r
322             default:\r
323                 return 0;\r
324         }\r
325     }\r
326 \r
327     static Path2D makePath(double x, double y, double w, double h) {\r
328         Path2D path = new Path2D.Double();\r
329         path.moveTo(x, y);\r
330         path.lineTo(x+w, y);\r
331         path.lineTo(x+w, y+h);\r
332         path.lineTo(x, y+h);\r
333         path.closePath();\r
334         return path;\r
335     }\r
336 \r
337     public static final Shape BOX_SHAPE = new Rectangle(-1, -1, 2, 2);\r
338 \r
339     public static class MonitorSGNode implements SceneGraph, InternalSize, Outline {\r
340         private static final long serialVersionUID = -106278359626957687L;\r
341 \r
342         static final MonitorSGNode INSTANCE = new MonitorSGNode();\r
343 \r
344         @SuppressWarnings("unchecked")\r
345         @Override\r
346         public void init(final IElement e, final G2DParentNode parent) {\r
347             // Create node if it doesn't exist yet\r
348             TextNode node = e.getHint(KEY_SG_NODE);\r
349             String nodeId = null;\r
350             if(node == null || node.getBounds() == null || node.getParent() != parent) {\r
351                 nodeId = ElementUtils.generateNodeId(e);\r
352                 node = parent.addNode(nodeId, TextNode.class);\r
353                 e.setHint(KEY_SG_NODE, node);\r
354 \r
355                 node.setTextListener(new ITextListener() {\r
356 \r
357                     @Override\r
358                     public void textChanged() {}\r
359 \r
360                     @Override\r
361                     public void textEditingStarted() {}\r
362 \r
363                     @Override\r
364                     public void textEditingCancelled() {}\r
365 \r
366                     boolean isEndingEdit = false;\r
367 \r
368                     @Override\r
369                     public void textEditingEnded() {\r
370                         TextNode node = e.getHint(KEY_SG_NODE);\r
371                         if (node == null)\r
372                             return;\r
373 \r
374                         // Prevent recursive execution which will happen\r
375                         // due to the endEdit(node) invocation at the end.\r
376                         if (isEndingEdit)\r
377                             return;\r
378                         isEndingEdit = true;\r
379 \r
380                         try {\r
381                             TextEditor editor = e.getElementClass().getAtMostOneItemOfClass(TextEditor.class);\r
382                             if (editor != null) {\r
383                                 Modifier modifier = editor.getModifier(e);\r
384                                 if (modifier != null) {\r
385                                     String newValue = node.getText();\r
386                                     String error = modifier.isValid(e, newValue);\r
387                                     if (error == null) {\r
388                                         // Only modify if the modification was not\r
389                                         // cancelled and the value is valid.\r
390                                         modifier.modify(e, newValue);\r
391                                     } else {\r
392                                         // TODO: show error somehow, possibly through status bar\r
393 \r
394                                         // Make sure that the monitor content gets\r
395                                         // reset to its previous value.\r
396                                         node.setText(formValue(e));\r
397                                     }\r
398                                 }\r
399                             }\r
400                         } finally {\r
401                             isEndingEdit = false;\r
402                         }\r
403                     }\r
404                 });\r
405 \r
406                 Object validator = e.getHint(KEY_INPUT_VALIDATOR);\r
407                 if(validator != null) {\r
408                     node.setValidator((Function1<String, String>)validator);\r
409                 }\r
410                 \r
411                 RVI rvi = e.getHint(KEY_RVI);\r
412                 if(rvi != null) {\r
413                     node.setRVI(rvi);\r
414                 }\r
415 \r
416                 Double border_width = (Double)e.getHint(KEY_BORDER_WIDTH);\r
417                 if(border_width == null) border_width = 0.1;\r
418 \r
419                 node.setBorderWidth(border_width);\r
420 \r
421 //                Rectangle2D bounds = (Rectangle2D)e.getHint(ElementHints.KEY_BOUNDS);\r
422 //                if(bounds != null) node.setBounds(bounds);\r
423                 Font font = ElementUtils.getTextFont(e);\r
424                 Color color = ElementUtils.getTextColor(e);\r
425                 String text = ElementUtils.getText(e);\r
426                 node.init(text, font, color, 0, 0, 1.0);\r
427             }\r
428 \r
429             Boolean isExternal = e.getHint(KEY_MONITOR_IS_EXTERNAL);\r
430             if (Boolean.TRUE.equals(isExternal)) {\r
431                 ShapeNode shape = e.getHint(KEY_SG_NODE2);\r
432                 if (shape == null || shape.getParent() != parent) {\r
433                     if (nodeId == null)\r
434                         nodeId = ElementUtils.generateNodeId(e);\r
435                     nodeId += "-ext";\r
436                     shape = parent.addNode(nodeId, ShapeNode.class);\r
437                     shape.setZIndex(-1);\r
438                     shape.setColor(Color.BLACK);\r
439                     shape.setFill(true);\r
440                     shape.setStroke(null);\r
441                     shape.setShape( arrow(4, 2, 0) );\r
442                     e.setHint(KEY_SG_NODE2, shape);\r
443                 }\r
444             } else {\r
445                 ShapeNode shape = e.getHint(KEY_SG_NODE2);\r
446                 if (shape != null)\r
447                     shape.remove();\r
448             }\r
449 \r
450             update(e);\r
451         }\r
452 \r
453         public void update(IElement e) {\r
454             String value = null;\r
455 \r
456             final Text t = e.getElementClass().getAtMostOneItemOfClass(Text.class);\r
457             assert(t != null);\r
458 \r
459             value = formValue(e);\r
460 \r
461             TextNode node = (TextNode) e.getHint(KEY_SG_NODE);\r
462             if (node != null && value != null) {\r
463 \r
464                 node.setText(value);\r
465                 Object component = e.getHint(KEY_MONITOR_COMPONENT);\r
466                 if (component != null && hasModifier(e)) {\r
467                     node.setEditable(true);\r
468                 } else {\r
469                     node.setEditable(false);\r
470                 }\r
471 \r
472                 // FIXME: set only if changed .. (but quickfix is not to clone)\r
473                 Font font = ElementUtils.getTextFont(e);\r
474                 if (node.getFont() != font) { // Don't update if we have a same object\r
475                     node.setFont(font);\r
476                 }\r
477                 Color color = ElementUtils.getTextColor(e);\r
478                 node.setColor(new Color(color.getRed(), color.getGreen(), color.getBlue(), color.getAlpha()));\r
479 \r
480                 AffineTransform at = ElementUtils.getTransform(e);\r
481                 if(at != null)\r
482                     node.setTransform(at);\r
483 \r
484                 Alignment halign = e.getHint(ElementHints.KEY_HORIZONTAL_ALIGN);\r
485                 if (halign != null)\r
486                     node.setHorizontalAlignment((byte) halign.ordinal());\r
487                 Alignment valign = e.getHint(ElementHints.KEY_VERTICAL_ALIGN);\r
488                 if (valign != null)\r
489                     node.setVerticalAlignment((byte) valign.ordinal());\r
490 \r
491 //                String tt = (String) e.getHint(KEY_TOOLTIP_TEXT);\r
492 //                if (tt != null)\r
493 //                    node.setToolTipText(new String(tt));\r
494 \r
495                 ShapeNode shape = e.getHint(KEY_SG_NODE2);\r
496                 if (shape != null) {\r
497                     AffineTransform at2 = new AffineTransform(at);\r
498                     Rectangle2D bounds = node.getBoundsInLocal();\r
499                     at2.translate(bounds.getMinX(), bounds.getCenterY());\r
500                     shape.setTransform(at2);\r
501                 }\r
502             }\r
503         }\r
504 \r
505         @Override\r
506         public void cleanup(IElement e) {\r
507             TextNode node = (TextNode)e.removeHint(KEY_SG_NODE);\r
508             if (node != null)\r
509                 node.remove();\r
510         }\r
511 \r
512         @Override\r
513         public Rectangle2D getBounds(IElement e, Rectangle2D size) {\r
514             TextNode node = (TextNode)e.getHint(KEY_SG_NODE);\r
515             if (node != null) {\r
516                 Rectangle2D bounds = node.getBoundsInLocal();\r
517                 if (bounds != null) {\r
518                     if (size == null)\r
519                         size = new Rectangle2D.Double(0, 0, 0, 0);\r
520                     size.setRect(bounds);\r
521                 }\r
522             }\r
523             return size;\r
524         }\r
525 \r
526         @Override\r
527         public Shape getElementShape(IElement e) {\r
528             Shape shape = new Rectangle2D.Double(0, 0, 0, 0);\r
529 \r
530             TextNode node = (TextNode)e.getHint(KEY_SG_NODE);\r
531             if(node != null && node.getBoundsInLocal() != null) {\r
532                 shape = node.getBoundsInLocal();\r
533             }\r
534 \r
535             return shape;\r
536         }\r
537 \r
538     }\r
539 \r
540     public static class Transformer implements Transform, Move, Rotate, Scale, LifeCycle {\r
541 \r
542         private static final long serialVersionUID = -3704887325602085677L;\r
543 \r
544         public static final Transformer INSTANCE = new Transformer(null);\r
545 \r
546         Double aspectRatio;\r
547 \r
548         public Transformer() {\r
549             this(null);\r
550         }\r
551 \r
552         public Transformer(Double aspectRatio) {\r
553             this.aspectRatio = aspectRatio;\r
554         }\r
555 \r
556         @Override\r
557         public Double getFixedAspectRatio(IElement e) {\r
558             return aspectRatio;\r
559         }\r
560 \r
561         @Override\r
562         public Point2D getScale(IElement e) {\r
563             AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
564             return _getScale(at);\r
565         }\r
566 \r
567         @Override\r
568         public void setScale(IElement e, Point2D newScale) {\r
569             // Doesn't work for monitors.\r
570             Point2D oldScale = getScale(e);\r
571             double sx = newScale.getX() / oldScale.getX();\r
572             double sy = newScale.getY() / oldScale.getY();\r
573             AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
574             at = new AffineTransform(at);\r
575             at.scale(sx, sy);\r
576             e.setHint(ElementHints.KEY_TRANSFORM, at);\r
577         }\r
578 \r
579         @Override\r
580         public Point2D getMaximumScale(IElement e) {\r
581             return null;\r
582         }\r
583 \r
584         @Override\r
585         public Point2D getMinimumScale(IElement e) {\r
586             return null;\r
587         }\r
588 \r
589         private static Point2D _getScale(AffineTransform at) {\r
590             double m00 = at.getScaleX();\r
591             double m11 = at.getScaleY();\r
592             double m10 = at.getShearY();\r
593             double m01 = at.getShearX();\r
594             // Project unit vector to canvas\r
595             double sx = Math.sqrt(m00 * m00 + m10 * m10);\r
596             double sy = Math.sqrt(m01 * m01 + m11 * m11);\r
597             return new Point2D.Double(sx, sy);\r
598         }\r
599 \r
600         @Override\r
601         public void rotate(IElement e, double theta, Point2D origin) {\r
602             if (Double.isNaN(theta)) return;\r
603             theta = Math.toDegrees(theta);\r
604             Double angle = e.getHint(KEY_DIRECTION);\r
605             double newAngle = angle != null ? angle+theta : theta;\r
606             newAngle = Math.IEEEremainder(newAngle, 360.0);\r
607             e.setHint(KEY_DIRECTION, newAngle);\r
608         }\r
609 \r
610         @Override\r
611         public double getAngle(IElement e) {\r
612             Double angle = e.getHint(KEY_DIRECTION);\r
613             return angle != null ? Math.toRadians(angle) : 0;\r
614         }\r
615         @Override\r
616         public Point2D getPosition(IElement e) {\r
617             AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
618             Point2D p = new Point2D.Double(at.getTranslateX(), at.getTranslateY());\r
619             return p;\r
620         }\r
621 \r
622         @Override\r
623         public void moveTo(IElement e, double x, double y) {\r
624             AffineTransform origAt = e.getHint(ElementHints.KEY_TRANSFORM);\r
625 \r
626             AffineTransform result = new AffineTransform(origAt);\r
627             result.preConcatenate(AffineTransform.getTranslateInstance(x - origAt.getTranslateX(), y - origAt.getTranslateY()));\r
628             e.setHint(ElementHints.KEY_TRANSFORM, result);\r
629         }\r
630 \r
631         @Override\r
632         public AffineTransform getTransform(IElement e) {\r
633             AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
634 \r
635             IElement parentElement = e.getHint(ElementHints.KEY_PARENT_ELEMENT);\r
636             if (parentElement == null)\r
637                 return at;\r
638 \r
639             Transform parentTransform = parentElement.getElementClass().getSingleItem(Transform.class);\r
640             assert(parentTransform!=null);\r
641 \r
642             AffineTransform result = (AffineTransform)at.clone();\r
643             AffineTransform parentAT = parentTransform.getTransform(parentElement);\r
644             result.preConcatenate(AffineTransform.getTranslateInstance(parentAT.getTranslateX(), parentAT.getTranslateY()));\r
645 \r
646             return result;\r
647         }\r
648 \r
649         @Override\r
650         public void setTransform(IElement e, AffineTransform at) {\r
651             e.setHint(ElementHints.KEY_TRANSFORM, at.clone());\r
652         }\r
653 \r
654         @Override\r
655         public void onElementActivated(IDiagram d, IElement e) {\r
656         }\r
657 \r
658         @Override\r
659         public void onElementCreated(IElement e) {\r
660             e.setHint(ElementHints.KEY_TRANSFORM, new AffineTransform());\r
661         }\r
662 \r
663         @Override\r
664         public void onElementDeactivated(IDiagram d, IElement e) {\r
665         }\r
666 \r
667         @Override\r
668         public void onElementDestroyed(IElement e) {\r
669 //            List<SceneGraph> nodeHandlers = e.getElementClass().getItemsByClass(SceneGraph.class);\r
670 //            for(SceneGraph n : nodeHandlers) {\r
671 //                System.out.println("element gone:"+e);\r
672 //                n.cleanup(e);\r
673 //            }\r
674         }\r
675     }\r
676 \r
677     static double getOrientationDelta(IElement e, AffineTransform tr) {\r
678         Double angle = e.getHint(KEY_DIRECTION);\r
679         if (angle == null || Double.isNaN(angle))\r
680             return Double.NaN;\r
681         double angrad = Math.toRadians(angle);\r
682 \r
683         Vector2d forcedAxis = new Vector2d(Math.cos(angrad), Math.sin(angrad));\r
684         Vector2d x = new Vector2d(tr.getScaleX(), tr.getShearX());\r
685         forcedAxis.normalize();\r
686         x.normalize();\r
687         double cosa = forcedAxis.dot(x);\r
688         double delta = Math.acos(cosa);\r
689         return delta;\r
690     }\r
691 \r
692     static class MonitorImageFactory implements IFactory<Image> {\r
693         private double staticScaleX = 1, staticScaleY = 1;\r
694 \r
695         public MonitorImageFactory(double staticScaleX, double staticScaleY) {\r
696             this.staticScaleX = staticScaleX;\r
697             this.staticScaleY = staticScaleY;\r
698         }\r
699 \r
700         @Override\r
701         public Image get() throws ProvisionException {\r
702             return new Img();\r
703         }\r
704 \r
705         public class Img extends AbstractImage {\r
706 \r
707             Shape path = align(DEFAULT_HORIZONTAL_MARGIN, DEFAULT_VERTICAL_MARGIN, DEFAULT_HORIZONTAL_ALIGN, DEFAULT_VERTICAL_ALIGN,\r
708                     new Rectangle2D.Double(0, 0, 50*staticScaleX, 22*staticScaleY));\r
709 \r
710             @Override\r
711             public int hashCode() {\r
712                 long temp = Double.doubleToLongBits(staticScaleX);\r
713                 int result = (int) (temp ^ (temp >>> 32));\r
714                 temp = Double.doubleToLongBits(staticScaleY);\r
715                 result = 31 * result + (int) (temp ^ (temp >>> 32));\r
716                 return result;\r
717             }\r
718 \r
719             @Override\r
720             public boolean equals(Object obj) {\r
721                 if (this == obj)\r
722                     return true;\r
723                 if (obj == null)\r
724                     return false;\r
725                 if (getClass() != obj.getClass())\r
726                     return false;\r
727                 MonitorImageFactory other = (MonitorImageFactory) obj;\r
728                 return Double.doubleToLongBits(staticScaleX) == Double.doubleToLongBits(other.staticScaleX)\r
729                         && Double.doubleToLongBits(staticScaleY) != Double.doubleToLongBits(other.staticScaleY);\r
730             }\r
731 \r
732             @Override\r
733             public Rectangle2D getBounds() {\r
734                 return path.getBounds2D();\r
735             }\r
736 \r
737             @Override\r
738             public EnumSet<Feature> getFeatures() {\r
739                 return EnumSet.of(Feature.Vector);\r
740             }\r
741 \r
742             @Override\r
743             public Shape getOutline() {\r
744                 return path;\r
745             }\r
746 \r
747             @Override\r
748             public Node init(G2DParentNode parent) {\r
749                 TextNode node = parent.getOrCreateNode(""+hashCode(), TextNode.class);\r
750                 node.init("Drop Me", FONT, Color.BLACK, 0, 0, 0.2);\r
751                 //node.setSize(50, 22);\r
752                 node.setBorderWidth(1);\r
753                 node.setTransform(AffineTransform.getScaleInstance(staticScaleX, staticScaleY));\r
754                 node.setEditable(false);\r
755                 return node;\r
756             }\r
757         };\r
758     }\r
759 \r
760     static final IProvider<Image> MONITOR_IMAGE =\r
761         ProviderUtils.reference(\r
762                 ProviderUtils.cache(\r
763                         ProviderUtils.rasterize(\r
764                                 new MonitorImageFactory(0.5, 0.5)\r
765                         )));\r
766 \r
767     static final StaticSymbol MONITOR_SYMBOL = new StaticSymbolImpl( MONITOR_IMAGE.get() );\r
768 \r
769     static final FillColor FILL_COLOR = new FillColorImpl(DEFAULT_FILL_COLOR);\r
770 \r
771     static final ElementClass MONITOR_CLASS_BASE =\r
772         ElementClass.compile(\r
773                 MonitorHandlerImpl.INSTANCE,\r
774                 Transformer.INSTANCE,\r
775                 BorderColorImpl.BLACK,\r
776                 FILL_COLOR,\r
777                 MonitorSGNode.INSTANCE,\r
778 //                ClickableImpl.INSTANCE,\r
779                 TextImpl.INSTANCE,\r
780                 TextEditorImpl.INSTANCE,\r
781                 TextFontImpl.DEFAULT,\r
782                 TextColorImpl.BLACK,\r
783                 SimpleElementLayers.INSTANCE,\r
784                 ParentImpl.INSTANCE\r
785         );\r
786 \r
787     // staticScale{X,Y} define the scale of the static monitor image\r
788     public static ElementClass create(double staticScaleX, double staticScaleY, ElementHandler... extraHandlers) {\r
789         // Bit of a hack to be able to define the scale\r
790         IProvider<Image> staticMonitorSymbolProvider = ProviderUtils.reference(\r
791                 ProviderUtils.cache(\r
792                         ProviderUtils\r
793                         .rasterize(\r
794                                 new MonitorImageFactory(staticScaleX, staticScaleY))));\r
795         StaticSymbol staticMonitorSymbol = new StaticSymbolImpl( staticMonitorSymbolProvider.get() );\r
796         return MONITOR_CLASS_BASE.newClassWith(staticMonitorSymbol).newClassWith(extraHandlers);\r
797     }\r
798 \r
799     // staticScale{X,Y} define the scale of the static monitor image\r
800     public static ElementClass create(IElement parentElement, Map<String, String> substitutions, Object component, String suffix, double staticScaleX, double staticScaleY, ElementHandler... extraHandlers) {\r
801         return create(staticScaleX, staticScaleY, extraHandlers)\r
802                 .newClassWith(\r
803                         new Initializer(parentElement, substitutions, component,\r
804                                 suffix, parentElement != null ? false : true));\r
805     }\r
806 \r
807     public static Shape arrow(double length, double width, double space) {\r
808         Path2D.Double path = new Path2D.Double();\r
809         path.moveTo(-space, 0);\r
810         path.lineTo(-length - space, -width);\r
811         path.lineTo(-length - space, +width);\r
812         path.closePath();\r
813         return path;\r
814     }\r
815 \r
816 }\r