The final restructured version for automatic build using maven tycho
[Mograsim.git] / plugins / net.mograsim.logic.model / src / net / mograsim / logic / model / model / components / atomic / ModelManualSwitch.java
diff --git a/plugins/net.mograsim.logic.model/src/net/mograsim/logic/model/model/components/atomic/ModelManualSwitch.java b/plugins/net.mograsim.logic.model/src/net/mograsim/logic/model/model/components/atomic/ModelManualSwitch.java
new file mode 100644 (file)
index 0000000..7a99504
--- /dev/null
@@ -0,0 +1,185 @@
+package net.mograsim.logic.model.model.components.atomic;
+
+import org.eclipse.swt.graphics.Color;
+
+import net.haspamelodica.swt.helper.gcs.GeneralGC;
+import net.haspamelodica.swt.helper.swtobjectwrappers.Font;
+import net.haspamelodica.swt.helper.swtobjectwrappers.Point;
+import net.haspamelodica.swt.helper.swtobjectwrappers.Rectangle;
+import net.mograsim.logic.core.LogicObserver;
+import net.mograsim.logic.core.components.CoreManualSwitch;
+import net.mograsim.logic.core.types.Bit;
+import net.mograsim.logic.core.types.BitVector;
+import net.mograsim.logic.core.types.BitVectorFormatter;
+import net.mograsim.logic.model.model.LogicModelModifiable;
+import net.mograsim.logic.model.model.components.ModelComponent;
+import net.mograsim.logic.model.model.wires.Pin;
+import net.mograsim.logic.model.model.wires.PinUsage;
+import net.mograsim.logic.model.modeladapter.LogicCoreAdapter;
+import net.mograsim.logic.model.modeladapter.componentadapters.ManualSwitchAdapter;
+import net.mograsim.logic.model.serializing.IdentifyParams;
+import net.mograsim.logic.model.serializing.IndirectModelComponentCreator;
+import net.mograsim.preferences.Preferences;
+
+public class ModelManualSwitch extends ModelComponent
+{
+       private static final double width = 20;
+       private static final double height = 15;
+       private static final double fontHeight = 5;
+       private static final double heightMiniButtons = 4; // 0 is disabled
+
+       public final int logicWidth;
+       private final Pin outputPin;
+
+       private final LogicObserver logicObs;
+       private CoreManualSwitch manualSwitch;
+
+       public ModelManualSwitch(LogicModelModifiable model, int logicWidth)
+       {
+               this(model, logicWidth, null);
+       }
+
+       public ModelManualSwitch(LogicModelModifiable model, int logicWidth, String name)
+       {
+               super(model, name, false);
+               this.logicWidth = logicWidth;
+               logicObs = (i) -> model.requestRedraw();
+
+               setSize(width, height);
+               addPin(this.outputPin = new Pin(model, this, "", logicWidth, PinUsage.OUTPUT, width, height / 2));
+
+               init();
+       }
+
+       @Override
+       public void render(GeneralGC gc, Rectangle visibleRegion)
+       {
+               Color foreground = Preferences.current().getColor("net.mograsim.logic.model.color.foreground");
+               if (foreground != null)
+                       gc.setForeground(foreground);
+               gc.drawRectangle(getBounds());
+               String label = BitVectorFormatter.formatAsString(manualSwitch == null ? null : manualSwitch.getValues());
+               Font oldFont = gc.getFont();
+               Font labelFont = new Font(oldFont.getName(), fontHeight, oldFont.getStyle());
+               gc.setFont(labelFont);
+               Point textExtent = gc.textExtent(label);
+               Color textColor = Preferences.current().getColor("net.mograsim.logic.model.color.text");
+               if (textColor != null)
+                       gc.setForeground(textColor);
+               gc.drawText(label, getPosX() + (width - textExtent.x) / 2, getPosY() + (height - textExtent.y) / 2, true);
+               gc.setFont(oldFont);
+
+               if (manualSwitch != null && logicWidth > 1 && heightMiniButtons > 0 && visibleRegion.y < getPosY() + heightMiniButtons)
+               {
+                       double x = getPosX();
+                       double y = getPosY();
+                       gc.drawLine(x, y + heightMiniButtons, x + width, y + heightMiniButtons);
+                       Color c = gc.getBackground();
+                       gc.setBackground(gc.getForeground());
+                       BitVector bv = manualSwitch.getValues();
+                       double part = width / bv.length();
+                       for (int i = 0; i < bv.length(); i++)
+                       {
+                               double start = x + part * i;
+                               if (i != 0)
+                                       gc.drawLine(start, y, start, y + heightMiniButtons);
+                               if (bv.getMSBit(i) == Bit.ONE)
+                               {
+//                                     gc.fillRectangle(start, y, part, heightMiniButtons); // alternative, but not always visible what Bit is where 
+                                       gc.drawLine(start, y, start + part, y + heightMiniButtons);
+                                       gc.drawLine(start + part, y, start, y + heightMiniButtons);
+                               }
+                       }
+                       gc.setBackground(c);
+               }
+       }
+
+       public void setCoreModelBinding(CoreManualSwitch logicSwitch)
+       {
+               if (this.manualSwitch != null)
+                       this.manualSwitch.deregisterObserver(logicObs);
+               this.manualSwitch = logicSwitch;
+               if (logicSwitch != null)
+                       logicSwitch.registerObserver(logicObs);
+       }
+
+       public boolean hasCoreModelBinding()
+       {
+               return manualSwitch != null;
+       }
+
+       @Override
+       public Object getHighLevelState(String stateID)
+       {
+               switch (stateID)
+               {
+               case "out":
+                       if (manualSwitch != null)
+                               return manualSwitch.getValues();
+                       return null;
+               default:
+                       return super.getHighLevelState(stateID);
+               }
+       }
+
+       @Override
+       public void setHighLevelState(String stateID, Object newState)
+       {
+               switch (stateID)
+               {
+               case "out":
+                       if (manualSwitch != null)
+                               manualSwitch.setState((BitVector) newState);
+                       break;
+               default:
+                       super.setHighLevelState(stateID, newState);
+                       break;
+               }
+       }
+
+       @Override
+       public boolean clicked(double x, double y)
+       {
+               if (manualSwitch != null)
+               {
+                       if (heightMiniButtons > 0 && y - getPosY() < heightMiniButtons)
+                       {
+                               int part = (int) ((x - getPosX()) * logicWidth / width);
+                               manualSwitch.setState(manualSwitch.getValues().withBitChanged(part, Bit::not));
+                       } else
+                       {
+                               manualSwitch.toggle();
+                       }
+               }
+               return true;
+       }
+
+       public CoreManualSwitch getManualSwitch()
+       {
+               return manualSwitch;
+       }
+
+       public Pin getOutputPin()
+       {
+               return outputPin;
+       }
+
+       @Override
+       public String getIDForSerializing(IdentifyParams idParams)
+       {
+               return "ManualSwitch";
+       }
+
+       @Override
+       public Integer getParamsForSerializing(IdentifyParams idParams)
+       {
+               return logicWidth;
+       }
+
+       static
+       {
+               LogicCoreAdapter.addComponentAdapter(new ManualSwitchAdapter());
+               IndirectModelComponentCreator.setComponentSupplier(ModelManualSwitch.class.getName(),
+                               (m, p, n) -> new ModelManualSwitch(m, p.getAsInt(), n));
+       }
+}
\ No newline at end of file