]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.modeling.ui/src/org/simantics/modeling/ui/diagram/SheetClass.java.keep
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.modeling.ui / src / org / simantics / modeling / ui / diagram / SheetClass.java.keep
diff --git a/bundles/org.simantics.modeling.ui/src/org/simantics/modeling/ui/diagram/SheetClass.java.keep b/bundles/org.simantics.modeling.ui/src/org/simantics/modeling/ui/diagram/SheetClass.java.keep
new file mode 100644 (file)
index 0000000..6d5d561
--- /dev/null
@@ -0,0 +1,617 @@
+/*******************************************************************************\r
+ * Copyright (c) 2007- VTT Technical Research Centre of Finland.\r
+ * All rights reserved. This program and the accompanying materials\r
+ * are made available under the terms of the Eclipse Public License v1.0\r
+ * which accompanies this distribution, and is available at\r
+ * http://www.eclipse.org/legal/epl-v10.html\r
+ *\r
+ * Contributors:\r
+ *     VTT Technical Research Centre of Finland - initial API and implementation\r
+ *******************************************************************************/\r
+package org.simantics.modeling.ui.diagram;\r
+\r
+import java.awt.BasicStroke;\r
+import java.awt.Color;\r
+import java.awt.Font;\r
+import java.awt.FontMetrics;\r
+import java.awt.Graphics2D;\r
+import java.awt.Rectangle;\r
+import java.awt.Shape;\r
+import java.awt.geom.AffineTransform;\r
+import java.awt.geom.Path2D;\r
+import java.awt.geom.Point2D;\r
+import java.awt.geom.Rectangle2D;\r
+import java.util.Collection;\r
+import java.util.EnumSet;\r
+import java.util.Map;\r
+\r
+import javax.swing.table.TableModel;\r
+\r
+import org.simantics.g2d.diagram.IDiagram;\r
+import org.simantics.g2d.element.ElementClass;\r
+import org.simantics.g2d.element.ElementHints;\r
+import org.simantics.g2d.element.ElementUtils;\r
+import org.simantics.g2d.element.IElement;\r
+import org.simantics.g2d.element.SceneGraphNodeKey;\r
+import org.simantics.g2d.element.handler.ElementHandler;\r
+import org.simantics.g2d.element.handler.FillColor;\r
+import org.simantics.g2d.element.handler.InternalSize;\r
+import org.simantics.g2d.element.handler.LifeCycle;\r
+import org.simantics.g2d.element.handler.Move;\r
+import org.simantics.g2d.element.handler.Outline;\r
+import org.simantics.g2d.element.handler.Rotate;\r
+import org.simantics.g2d.element.handler.Scale;\r
+import org.simantics.g2d.element.handler.SceneGraph;\r
+import org.simantics.g2d.element.handler.StaticSymbol;\r
+import org.simantics.g2d.element.handler.Text;\r
+import org.simantics.g2d.element.handler.Transform;\r
+import org.simantics.g2d.element.handler.impl.BorderColorImpl;\r
+import org.simantics.g2d.element.handler.impl.FillColorImpl;\r
+import org.simantics.g2d.element.handler.impl.SimpleElementLayers;\r
+import org.simantics.g2d.element.handler.impl.StaticSymbolImpl;\r
+import org.simantics.g2d.element.handler.impl.TextEditorImpl;\r
+import org.simantics.g2d.element.handler.impl.TextImpl;\r
+import org.simantics.g2d.elementclass.MonitorHandler;\r
+import org.simantics.g2d.image.Image;\r
+import org.simantics.g2d.image.ProviderUtils;\r
+import org.simantics.g2d.image.impl.AbstractImage;\r
+import org.simantics.g2d.utils.Alignment;\r
+import org.simantics.scenegraph.Node;\r
+import org.simantics.scenegraph.g2d.G2DParentNode;\r
+import org.simantics.spreadsheet.Model;\r
+import org.simantics.utils.datastructures.cache.IFactory;\r
+import org.simantics.utils.datastructures.cache.IProvider;\r
+import org.simantics.utils.datastructures.cache.ProvisionException;\r
+import org.simantics.utils.datastructures.hints.IHintContext.Key;\r
+import org.simantics.utils.datastructures.hints.IHintContext.KeyOf;\r
+\r
+/**\r
+ * @author Tuukka Lehtonen\r
+ */\r
+public class SheetClass {\r
+\r
+    static final Font        FONT                      = Font.decode("Helvetica 10");\r
+\r
+    public static final Key  KEY_MODEL = new KeyOf(Model.class, "MODEL");\r
+    public static final Key  KEY_TABLE_MODEL = new KeyOf(TableModel.class, "TABLE_MODEL");\r
+\r
+    public static final Key  KEY_MONITOR_RESOURCE_PATH = new KeyOf(Collection.class, "MONITOR_RESOURCE_PATH");\r
+    public static final Key  KEY_MONITOR_SUBSTITUTIONS = new KeyOf(Map.class, "MONITOR_SUBSTITUTIONS");\r
+    public static final Key  KEY_MONITOR_GC            = new KeyOf(Graphics2D.class, "MONITOR_GC");\r
+    public static final Key  KEY_MONITOR_HEIGHT        = new KeyOf(Double.class, "MONITOR_HEIGHT");\r
+    public static final Key  KEY_MONITOR_FONT_METRICS  = new KeyOf(FontMetrics.class, "MONITOR_FONT_METRICS");\r
+    public static final Key  KEY_NUMBER_FORMAT            = new KeyOf(String.class, "NUMBER_FORMAT");\r
+\r
+    /**\r
+     * If this hint is defined, the monitor will force its x-axis to match this\r
+     * angle. If this hint doesn't exist, the monitor will not force x-axis\r
+     * orientation.\r
+     */\r
+    public static final Key  KEY_DIRECTION             = new KeyOf(Double.class, "MONITOR_DIRECTION");\r
+\r
+    public static final Key  KEY_SG_NODE             = new SceneGraphNodeKey(Node.class, "MONITOR_SG_NODE");\r
+\r
+    final static BasicStroke STROKE                    = new BasicStroke(1.0f);\r
+\r
+    public final static Alignment DEFAULT_HORIZONTAL_ALIGN  = Alignment.CENTER;\r
+    public final static Alignment DEFAULT_VERTICAL_ALIGN    = Alignment.CENTER;\r
+    public final static String DEFAULT_NUMBER_FORMAT    = "0.0###";\r
+\r
+    public final static Color     DEFAULT_FILL_COLOR        = new Color(224, 224, 224);\r
+    public final static Color     DEFAULT_BORDER_COLOR      = Color.BLACK;\r
+\r
+    public final static double    DEFAULT_HORIZONTAL_MARGIN = 5.0;\r
+    public final static double    DEFAULT_VERTICAL_MARGIN   = 2.5;\r
+\r
+    static Alignment getHorizontalAlignment(IElement e) {\r
+        return ElementUtils.getHintOrDefault(e, ElementHints.KEY_HORIZONTAL_ALIGN, DEFAULT_HORIZONTAL_ALIGN);\r
+    }\r
+\r
+    static Alignment getVerticalAlignment(IElement e) {\r
+        return ElementUtils.getHintOrDefault(e, ElementHints.KEY_VERTICAL_ALIGN, DEFAULT_VERTICAL_ALIGN);\r
+    }\r
+\r
+    static String getNumberFormat(IElement e) {\r
+        return ElementUtils.getHintOrDefault(e, KEY_NUMBER_FORMAT, DEFAULT_NUMBER_FORMAT);\r
+    }\r
+\r
+    static void setNumberFormat(IElement e, String format) {\r
+        ElementUtils.setOrRemoveHint(e, KEY_NUMBER_FORMAT, DEFAULT_NUMBER_FORMAT);\r
+    }\r
+\r
+    static double getHorizontalMargin(IElement e) {\r
+        return DEFAULT_HORIZONTAL_MARGIN;\r
+    }\r
+\r
+    static double getVerticalMargin(IElement e) {\r
+        return DEFAULT_VERTICAL_MARGIN;\r
+    }\r
+\r
+    static Font getFont(IElement e) {\r
+        return ElementUtils.getHintOrDefault(e, ElementHints.KEY_FONT, FONT);\r
+    }\r
+\r
+    public static class MonitorHandlerImpl implements MonitorHandler {\r
+        private static final long          serialVersionUID = -4258875745321808416L;\r
+        public static final MonitorHandler INSTANCE         = new MonitorHandlerImpl();\r
+    }\r
+\r
+    static class Initializer implements LifeCycle {\r
+        private static final long serialVersionUID = 4404942036933073584L;\r
+\r
+        IElement parentElement;\r
+        Map<String, String> substitutions;\r
+        Collection<Object> path;\r
+        boolean hack;\r
+\r
+        Initializer(IElement parentElement, Map<String, String> substitutions, Collection<Object> path, boolean hack) {\r
+            this.parentElement = parentElement;\r
+            this.substitutions = substitutions;\r
+            this.path = path;\r
+            this.hack = hack;\r
+        }\r
+\r
+        @Override\r
+        public void onElementActivated(IDiagram d, IElement e) {\r
+\r
+            if(!hack) {\r
+\r
+                hack = true;\r
+\r
+                Point2D parentPos = ElementUtils.getPos(parentElement);\r
+                Point2D thisPos = ElementUtils.getPos(e);\r
+\r
+                Move move = e.getElementClass().getSingleItem(Move.class);\r
+                move.moveTo(e, thisPos.getX() - parentPos.getX(), thisPos.getY() - parentPos.getY());\r
+\r
+            }\r
+\r
+        }\r
+        @Override\r
+        public void onElementCreated(IElement e) {\r
+            if(parentElement != null) e.setHint(ElementHints.KEY_PARENT_ELEMENT, parentElement);\r
+            if(substitutions != null) e.setHint(KEY_MONITOR_SUBSTITUTIONS, substitutions);\r
+            if(path != null) e.setHint(KEY_MONITOR_RESOURCE_PATH, path);\r
+\r
+            e.setHint(KEY_DIRECTION, 0.0);\r
+            e.setHint(KEY_NUMBER_FORMAT, DEFAULT_NUMBER_FORMAT);\r
+            //e.setHint(KEY_HORIZONTAL_ALIGN, Alignment.LEADING);\r
+            //e.setHint(KEY_VERTICAL_ALIGN, Alignment.LEADING);\r
+        }\r
+        @Override\r
+        public void onElementDeactivated(IDiagram d, IElement e) {\r
+        }\r
+        @Override\r
+        public void onElementDestroyed(IElement e) {\r
+        }\r
+    };\r
+\r
+    static String finalText(IElement e) {\r
+\r
+        Text t = e.getElementClass().getAtMostOneItemOfClass(Text.class);\r
+        assert(t != null);\r
+        String text = t.getText(e);\r
+        if(text == null) return null;\r
+        Map<String, String> substitutions = e.getHint(KEY_MONITOR_SUBSTITUTIONS);\r
+        if(substitutions != null) {\r
+            // TODO: slow as hell\r
+            for(Map.Entry<String, String> entry : substitutions.entrySet()) {\r
+                if (entry.getValue() != null) {\r
+                    text = text.replace(entry.getKey(), entry.getValue());\r
+                } else {\r
+                    text = text.replace(entry.getKey(), "<null>");\r
+                }\r
+            }\r
+        }\r
+\r
+        return text;\r
+    }\r
+\r
+    public static void update(IElement e) {\r
+        SheetSGNode node = e.getElementClass().getAtMostOneItemOfClass(SheetSGNode.class);\r
+        node.update(e);\r
+    }\r
+\r
+    public static void cleanup(IElement e) {\r
+        SheetSGNode node = e.getElementClass().getAtMostOneItemOfClass(SheetSGNode.class);\r
+        node.cleanup(e);\r
+    }\r
+\r
+    public static String editText(IElement e) {\r
+\r
+        Text t = e.getElementClass().getAtMostOneItemOfClass(Text.class);\r
+        assert(t != null);\r
+        String text = "#v1";\r
+        if(text == null) return null;\r
+        Map<String, String> substitutions = e.getHint(KEY_MONITOR_SUBSTITUTIONS);\r
+        if(substitutions != null) {\r
+            // TODO: slow as hell\r
+            for(Map.Entry<String, String> entry : substitutions.entrySet()) {\r
+                if (entry.getValue() != null) {\r
+                    text = text.replace(entry.getKey(), entry.getValue());\r
+                } else {\r
+                    text = text.replace(entry.getKey(), "<null>");\r
+                }\r
+            }\r
+        }\r
+\r
+        return text;\r
+    }\r
+\r
+    static final Rectangle2D DEFAULT_BOX = new Rectangle2D.Double(0, 0, 0, 0);\r
+\r
+    static Shape createMonitor(IElement e) {\r
+        Alignment hAlign = getHorizontalAlignment(e);\r
+        Alignment vAlign = getVerticalAlignment(e);\r
+        double hMargin = getHorizontalMargin(e);\r
+        double vMargin = getVerticalMargin(e);\r
+\r
+        String text = finalText(e);\r
+        if(text == null) {\r
+            return align(hMargin, vMargin, hAlign, vAlign, DEFAULT_BOX);\r
+        }\r
+\r
+        Graphics2D g = e.getHint(KEY_MONITOR_GC);\r
+        if(g == null) {\r
+            return align(hMargin, vMargin, hAlign, vAlign, DEFAULT_BOX);\r
+        }\r
+\r
+        Font f = getFont(e);\r
+        FontMetrics fm   = g.getFontMetrics(f);\r
+        Rectangle2D rect = fm.getStringBounds(text, g);\r
+\r
+        return align(hMargin, vMargin, hAlign, vAlign, rect);\r
+    }\r
+\r
+    static Shape align(double hMargin, double vMargin, Alignment hAlign, Alignment vAlign, Rectangle2D rect) {\r
+        //System.out.println("align: " + hMargin + ", " + vMargin + ", " + hAlign + ", " + vAlign + ": " + rect);\r
+        double tx = align(hMargin, hAlign, rect.getMinX(), rect.getMaxX());\r
+        double ty = align(vMargin, vAlign, rect.getMinY(), rect.getMaxY());\r
+        //System.out.println("    translate: " + tx + " "  + ty);\r
+        double nw = rect.getWidth() + 2*hMargin;\r
+        double nh = rect.getHeight() + 2*vMargin;\r
+        return makePath(tx + rect.getMinX(), ty + rect.getMinY(), nw, nh);\r
+    }\r
+\r
+    static double align(double margin, Alignment align, double min, double max) {\r
+        double s = max - min;\r
+        switch (align) {\r
+            case LEADING:\r
+                return -min;\r
+            case TRAILING:\r
+                return -s - 2 * margin - min;\r
+            case CENTER:\r
+                return -0.5 * s - margin - min;\r
+            default:\r
+                return 0;\r
+        }\r
+    }\r
+\r
+    static Path2D makePath(double x, double y, double w, double h) {\r
+        Path2D path = new Path2D.Double();\r
+        path.moveTo(x, y);\r
+        path.lineTo(x+w, y);\r
+        path.lineTo(x+w, y+h);\r
+        path.lineTo(x, y+h);\r
+        path.closePath();\r
+        return path;\r
+    }\r
+\r
+    public static final Shape BOX_SHAPE = new Rectangle(-1, -1, 2, 2);\r
+\r
+    public static class SheetSGNode implements SceneGraph, InternalSize, Outline {\r
+        \r
+        private static final long serialVersionUID = -5823585015844593347L;\r
+\r
+        static final SheetSGNode INSTANCE = new SheetSGNode();\r
+\r
+        @Override\r
+        public void init(final IElement e, final G2DParentNode parent) {\r
+            // Create node if it doesn't exist yet\r
+            SheetNode node = (SheetNode)e.getHint(KEY_SG_NODE);\r
+            if(node == null || node.getBounds() == null || node.getParent() != parent) {\r
+                node = parent.addNode(ElementUtils.generateNodeId(e), SheetNode.class);\r
+                e.setHint(KEY_SG_NODE, node);\r
+                Model model = e.getHint(KEY_MODEL);\r
+                TableModel tableModel = e.getHint(KEY_TABLE_MODEL);\r
+                node.init(model, tableModel);\r
+                System.out.println(parent);\r
+                node.setSize(500, 220);\r
+            }\r
+            update(e);\r
+        }\r
+\r
+        public void update(IElement e) {\r
+            \r
+//            String value = null;\r
+\r
+            final Text t = e.getElementClass().getAtMostOneItemOfClass(Text.class);\r
+            assert(t != null);\r
+            String text = finalText(e);\r
+\r
+//            System.out.println("monitorclass text = " + text);\r
+//\r
+//            if(text != null) {\r
+//                Map<String, String> substitutions = e.getHint(KEY_MONITOR_SUBSTITUTIONS);\r
+//                for(Map.Entry<String, String> entry : substitutions.entrySet()) {\r
+//                   System.out.println("substitute " + entry.getKey() + " -> " + entry.getValue());\r
+//                    text.replace(entry.getKey(), entry.getValue());\r
+//                }\r
+//            }\r
+//            key = text;\r
+//            if(substitutions != null) {\r
+//                value = substitutions.get("#v1");\r
+//                if(substitutions.containsKey("#u1")) {\r
+//                    value += substitutions.get("#u1");\r
+//                }\r
+//            }\r
+\r
+//            SheetNode node = (SheetNode) e.getHint(KEY_SG_NODE);\r
+//            if (node != null && text != null) {\r
+//                node.setText(text);\r
+//                Collection<?> path = e.getHint(KEY_MONITOR_RESOURCE_PATH);\r
+//                if (path != null && !path.isEmpty()) {\r
+//                    node.setEditable(true);\r
+//                } else {\r
+//                    node.setEditable(false);\r
+//                }\r
+//            }\r
+            \r
+        }\r
+\r
+        @Override\r
+        public void cleanup(IElement e) {\r
+            SheetNode node = (SheetNode)e.removeHint(KEY_SG_NODE);\r
+            if (node != null)\r
+                node.remove();\r
+        }\r
+\r
+        @Override\r
+        public Rectangle2D getBounds(IElement e, Rectangle2D size) {\r
+            Rectangle2D shape = new Rectangle2D.Double(0, 0, 0, 0);\r
+\r
+            SheetNode node = (SheetNode)e.getHint(KEY_SG_NODE);\r
+            if(node != null && node.getBounds() != null) {\r
+                shape = node.getBounds().getBounds2D();\r
+            }\r
+\r
+            if(size != null) size.setRect(shape);\r
+            return shape;\r
+        }\r
+\r
+        @Override\r
+        public Shape getElementShape(IElement e) {\r
+            Shape shape = new Rectangle2D.Double(0, 0, 0, 0);\r
+\r
+            SheetNode node = (SheetNode)e.getHint(KEY_SG_NODE);\r
+            if(node != null && node.getBounds() != null) {\r
+                shape = node.getBounds();\r
+            }\r
+\r
+            return shape;\r
+        }\r
+\r
+    }\r
+\r
+    public static class Transformer implements Transform, Move, Rotate, Scale {\r
+\r
+        private static final long serialVersionUID = -3704887325602085677L;\r
+\r
+        public static final Transformer INSTANCE = new Transformer(null);\r
+\r
+        Double aspectRatio;\r
+\r
+        public Transformer() {\r
+            this(null);\r
+        }\r
+\r
+        public Transformer(Double aspectRatio) {\r
+            this.aspectRatio = aspectRatio;\r
+        }\r
+\r
+        @Override\r
+        public Double getFixedAspectRatio(IElement e) {\r
+            return aspectRatio;\r
+        }\r
+\r
+        @Override\r
+        public Point2D getScale(IElement e) {\r
+            AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
+            return _getScale(at);\r
+        }\r
+\r
+        @Override\r
+        public void setScale(IElement e, Point2D newScale) {\r
+            // Doesn't work for monitors.\r
+            Point2D oldScale = getScale(e);\r
+            double sx = newScale.getX() / oldScale.getX();\r
+            double sy = newScale.getY() / oldScale.getY();\r
+            AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
+            at = new AffineTransform(at);\r
+            at.scale(sx, sy);\r
+            e.setHint(ElementHints.KEY_TRANSFORM, at);\r
+        }\r
+\r
+        @Override\r
+        public Point2D getMaximumScale(IElement e) {\r
+            return null;\r
+        }\r
+\r
+        @Override\r
+        public Point2D getMinimumScale(IElement e) {\r
+            return null;\r
+        }\r
+\r
+        private static Point2D _getScale(AffineTransform at) {\r
+            double m00 = at.getScaleX();\r
+            double m11 = at.getScaleY();\r
+            double m10 = at.getShearY();\r
+            double m01 = at.getShearX();\r
+            // Project unit vector to canvas\r
+            double sx = Math.sqrt(m00 * m00 + m10 * m10);\r
+            double sy = Math.sqrt(m01 * m01 + m11 * m11);\r
+            return new Point2D.Double(sx, sy);\r
+        }\r
+\r
+        @Override\r
+        public void rotate(IElement e, double theta, Point2D origin) {\r
+            if (Double.isNaN(theta)) return;\r
+            theta = Math.toDegrees(theta);\r
+            Double angle = e.getHint(KEY_DIRECTION);\r
+            double newAngle = angle != null ? angle+theta : theta;\r
+            newAngle = Math.IEEEremainder(newAngle, 360.0);\r
+            e.setHint(KEY_DIRECTION, newAngle);\r
+        }\r
+\r
+        @Override\r
+        public double getAngle(IElement e) {\r
+            Double angle = e.getHint(KEY_DIRECTION);\r
+            return angle != null ? Math.toRadians(angle) : 0;\r
+        }\r
+\r
+        @Override\r
+        public Point2D getPosition(IElement e) {\r
+            AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
+            Point2D p = new Point2D.Double(at.getTranslateX(), at.getTranslateY());\r
+            return p;\r
+        }\r
+\r
+        @Override\r
+        public void moveTo(IElement e, double x, double y) {\r
+            AffineTransform origAt = e.getHint(ElementHints.KEY_TRANSFORM);\r
+            double oldX = origAt.getTranslateX();\r
+            double oldY = origAt.getTranslateY();\r
+            AffineTransform move = AffineTransform.getTranslateInstance(x-oldX, y-oldY);\r
+            AffineTransform at2 = new AffineTransform(origAt);\r
+            at2.preConcatenate(move);\r
+            e.setHint(ElementHints.KEY_TRANSFORM, at2);\r
+        }\r
+\r
+        @Override\r
+        public AffineTransform getTransform(IElement e) {\r
+            AffineTransform at = e.getHint(ElementHints.KEY_TRANSFORM);\r
+\r
+            IElement parentElement = e.getHint(ElementHints.KEY_PARENT_ELEMENT);\r
+            if (parentElement == null)\r
+                return at;\r
+\r
+            Transform parentTransform = parentElement.getElementClass().getSingleItem(Transform.class);\r
+            assert(parentTransform!=null);\r
+\r
+            AffineTransform result = (AffineTransform)at.clone();\r
+            result.preConcatenate(parentTransform.getTransform(parentElement));\r
+\r
+            return result;\r
+        }\r
+\r
+        @Override\r
+        public void setTransform(IElement e, AffineTransform at) {\r
+            e.setHint(ElementHints.KEY_TRANSFORM, at.clone());\r
+        }\r
+\r
+//        @Override\r
+//        public void onElementActivated(IDiagram d, IElement e) {\r
+//        }\r
+//\r
+//        @Override\r
+//        public void onElementCreated(IElement e) {\r
+//            e.setHint(ElementHints.KEY_TRANSFORM, new AffineTransform());\r
+//        }\r
+//\r
+//        @Override\r
+//        public void onElementDeactivated(IDiagram d, IElement e) {\r
+//        }\r
+//\r
+//        @Override\r
+//        public void onElementDestroyed(IElement e) {\r
+//                     List<SceneGraph> nodeHandlers = e.getElementClass().getItemsByClass(SceneGraph.class);\r
+//                     for(SceneGraph n : nodeHandlers) {\r
+//                             System.out.println("element gone:"+e);\r
+//                             n.cleanup(e);\r
+//                     }\r
+//        }\r
+    }\r
+\r
+    static class MonitorImageFactory implements IFactory<Image> {\r
+\r
+        @Override\r
+        public Image get() throws ProvisionException {\r
+            return new AbstractImage() {\r
+                Shape path = align(DEFAULT_HORIZONTAL_MARGIN, DEFAULT_VERTICAL_MARGIN, DEFAULT_HORIZONTAL_ALIGN, DEFAULT_VERTICAL_ALIGN,\r
+                        new Rectangle2D.Double(0, 0, 50, 22));\r
+\r
+                @Override\r
+                public Rectangle2D getBounds() {\r
+                    return path.getBounds2D();\r
+                }\r
+\r
+                @Override\r
+                public EnumSet<Feature> getFeatures() {\r
+                    return EnumSet.of(Feature.Vector);\r
+                }\r
+\r
+                @Override\r
+                public Shape getOutline() {\r
+                    return path;\r
+                }\r
+\r
+                @Override\r
+                public Node init(G2DParentNode parent) {\r
+                    SheetNode node = parent.getOrCreateNode(""+hashCode(), SheetNode.class);\r
+//                    node.setText("");\r
+//                    node.setSize(50, 22);\r
+                    node.setTransform(new AffineTransform());\r
+                    return node;\r
+                }\r
+            };\r
+        }\r
+    }\r
+\r
+    static final IProvider<Image> MONITOR_IMAGE =\r
+        ProviderUtils.reference(\r
+                ProviderUtils.cache(\r
+                        ProviderUtils.rasterize(\r
+                                new MonitorImageFactory()\r
+                        )));\r
+\r
+    static final StaticSymbol MONITOR_SYMBOL = new StaticSymbolImpl( MONITOR_IMAGE.get() );\r
+\r
+    static final FillColor FILL_COLOR = new FillColorImpl(DEFAULT_FILL_COLOR);\r
+\r
+    public static final ElementClass MONITOR_CLASS =\r
+        ElementClass.compile(\r
+                MonitorHandlerImpl.INSTANCE,\r
+                Transformer.INSTANCE,\r
+                BorderColorImpl.BLACK,\r
+                FILL_COLOR,\r
+                SheetSGNode.INSTANCE,\r
+                TextImpl.INSTANCE,\r
+                TextEditorImpl.INSTANCE,\r
+                SimpleElementLayers.INSTANCE,\r
+                MONITOR_SYMBOL\r
+        );\r
+\r
+    // staticScale{X,Y} define the scale of the static monitor image\r
+    public static ElementClass create(IElement parentElement, Map<String, String> substitutions, Collection<Object> path, double staticScaleX, double staticScaleY, ElementHandler... extraHandlers) {\r
+        // Bit of a hack to be able to define the scale\r
+        IProvider<Image> staticMonitorSymbolProvider = ProviderUtils.reference(\r
+                ProviderUtils.cache(\r
+                        ProviderUtils\r
+                        .rasterize(\r
+                                new MonitorImageFactory())));\r
+        StaticSymbol staticMonitorSymbol = new StaticSymbolImpl( staticMonitorSymbolProvider.get() );\r
+        return ElementClass.compile(\r
+                new Initializer(parentElement, substitutions, path, parentElement != null ? false : true),\r
+                MonitorHandlerImpl.INSTANCE,\r
+                Transformer.INSTANCE,\r
+                BorderColorImpl.BLACK,\r
+                FILL_COLOR,\r
+                SheetSGNode.INSTANCE,\r
+                TextImpl.INSTANCE,\r
+                TextEditorImpl.INSTANCE,\r
+                SimpleElementLayers.INSTANCE,\r
+                staticMonitorSymbol\r
+        ).newClassWith(extraHandlers);\r
+    }\r
+\r
+}\r