3 import java.util.HashMap;
\r
4 import java.util.HashSet;
\r
5 import java.util.Map;
\r
6 import java.util.Set;
\r
7 import java.util.concurrent.atomic.AtomicBoolean;
\r
9 import org.eclipse.swt.SWT;
\r
10 import org.eclipse.swt.layout.FillLayout;
\r
11 import org.eclipse.swt.widgets.Display;
\r
12 import org.eclipse.swt.widgets.Event;
\r
13 import org.eclipse.swt.widgets.Shell;
\r
15 import era.mi.gui.components.BasicGUIComponent;
\r
16 import era.mi.gui.wires.GUIWire;
\r
17 import era.mi.logic.Simulation;
\r
18 import net.haspamelodica.swt.helper.gcs.GeneralGC;
\r
19 import net.haspamelodica.swt.helper.gcs.TranslatedGC;
\r
20 import net.haspamelodica.swt.helper.swtobjectwrappers.Point;
\r
21 import net.haspamelodica.swt.helper.zoomablecanvas.ZoomableCanvas;
\r
22 import net.haspamelodica.swt.helper.zoomablecanvas.helper.ZoomableCanvasOverlay;
\r
23 import net.haspamelodica.swt.helper.zoomablecanvas.helper.ZoomableCanvasUserInput;
\r
26 * Standalone simulation visualizer.
\r
28 * @author Daniel Kirschten
\r
30 public class LogicUI
\r
32 private final Display display;
\r
33 private final Shell shell;
\r
34 private final ZoomableCanvas canvas;
\r
35 private final Set<BasicGUIComponent> components;
\r
36 private final Map<BasicGUIComponent, Point> componentPositions;
\r
37 private final Set<GUIWire> wires;
\r
41 display = new Display();
\r
42 shell = new Shell(display);
\r
43 shell.setLayout(new FillLayout());
\r
44 canvas = new ZoomableCanvas(shell, SWT.NONE);
\r
46 components = new HashSet<>();
\r
47 componentPositions = new HashMap<>();
\r
48 wires = new HashSet<>();
\r
50 canvas.addZoomedRenderer(gc -> components.forEach(c -> drawComponent(gc, c)));
\r
51 canvas.addZoomedRenderer(gc -> wires.forEach(w -> w.render(gc)));
\r
52 ZoomableCanvasUserInput userInput = new ZoomableCanvasUserInput(canvas);
\r
53 userInput.buttonDrag = 3;
\r
54 userInput.buttonZoom = 2;
\r
55 userInput.enableUserInput();
\r
56 new ZoomableCanvasOverlay(canvas, null).enableScale();
\r
57 canvas.addListener(SWT.MouseDown, this::mouseDown);
\r
60 * Add a component to be drawn.
\r
61 * Returns the given component for convenience.
\r
63 * @author Daniel Kirschten
\r
65 public <C extends BasicGUIComponent> C addComponent(C component, double x, double y)
\r
67 components.add(component);
\r
68 componentPositions.put(component, new Point(x, y));
\r
72 * Add a graphical wire between the given connection points of the given components.
\r
73 * The given components have to be added and the given connection points have to be connected logically first.
\r
75 * @author Daniel Kirschten
\r
77 public void addWire(BasicGUIComponent component1, int component1ConnectionIndex, BasicGUIComponent component2, int component2ConnectionIndex, Point... path)
\r
79 wires.add(new GUIWire(canvas::redrawThreadsafe, component1, component1ConnectionIndex, componentPositions.get(component1), component2, component2ConnectionIndex, componentPositions.get(component2), path));
\r
81 private void drawComponent(GeneralGC gc, BasicGUIComponent component)
\r
83 TranslatedGC tgc = new TranslatedGC(gc, componentPositions.get(component));
\r
84 component.render(tgc);
\r
85 tgc.setBackground(display.getSystemColor(SWT.COLOR_BLUE));
\r
87 private void mouseDown(Event e)
\r
91 Point click = canvas.displayToWorldCoords(e.x, e.y);
\r
92 for(BasicGUIComponent component : components)
\r
93 if(component.getBounds().translate(componentPositions.get(component)).contains(click))
\r
95 if(component.clicked(click.x, click.y))
\r
103 * Start the simulation timeline, and open the UI shell.
\r
104 * Returns when the shell is closed.
\r
108 AtomicBoolean running = new AtomicBoolean(true);
\r
109 Thread simulationThread = new Thread(() ->
\r
111 while(running.get())
\r
113 //always execute to keep timeline from "hanging behind" for too long
\r
114 Simulation.TIMELINE.executeUpTo(System.currentTimeMillis(), System.currentTimeMillis() + 10);
\r
116 if(Simulation.TIMELINE.hasNext())
\r
117 sleepTime = Simulation.TIMELINE.nextEventTime() - System.currentTimeMillis();
\r
123 Thread.sleep(sleepTime);
\r
124 } catch(InterruptedException e)
\r
125 {} //it is normal execution flow to be interrupted
\r
128 simulationThread.start();
\r
129 Simulation.TIMELINE.addEventAddedListener(event ->
\r
131 if(event.getTiming() <= System.currentTimeMillis())
\r
132 simulationThread.interrupt();
\r
136 while(!shell.isDisposed())
\r
137 if(!display.readAndDispatch())
\r
139 running.set(false);
\r
140 simulationThread.interrupt();
\r