TimePanel.java 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389
  1. package holeg.ui.view.main;
  2. import holeg.preferences.ColorPreference;
  3. import holeg.preferences.ImagePreference;
  4. import holeg.ui.controller.Control;
  5. import holeg.ui.model.GuiSettings;
  6. import holeg.ui.view.image.Import;
  7. import holeg.utility.listener.LostFocusListener;
  8. import java.awt.BorderLayout;
  9. import java.awt.Color;
  10. import java.awt.Dimension;
  11. import java.awt.FlowLayout;
  12. import java.awt.GridBagConstraints;
  13. import java.awt.GridBagLayout;
  14. import java.awt.Point;
  15. import java.awt.event.ActionEvent;
  16. import java.awt.event.ActionListener;
  17. import java.awt.event.KeyEvent;
  18. import java.awt.event.MouseAdapter;
  19. import java.awt.event.MouseEvent;
  20. import java.awt.event.MouseListener;
  21. import java.util.Hashtable;
  22. import java.util.concurrent.ScheduledFuture;
  23. import java.util.concurrent.ScheduledThreadPoolExecutor;
  24. import java.util.concurrent.TimeUnit;
  25. import javax.swing.AbstractAction;
  26. import javax.swing.Box;
  27. import javax.swing.ImageIcon;
  28. import javax.swing.JButton;
  29. import javax.swing.JLabel;
  30. import javax.swing.JPanel;
  31. import javax.swing.JSlider;
  32. import javax.swing.JTextField;
  33. import javax.swing.KeyStroke;
  34. import javax.swing.SwingConstants;
  35. import javax.swing.Timer;
  36. import javax.swing.event.ChangeEvent;
  37. import javax.swing.event.ChangeListener;
  38. import javax.swing.plaf.basic.BasicSliderUI;
  39. /**
  40. * This Class represents a Panel where the User can start and stop the Simulation. He Can also reset
  41. * the Simulation and click through every Iteration.
  42. */
  43. public class TimePanel extends JPanel implements ActionListener {
  44. private static final int MAX_ITERATIONS = 100000;
  45. /*
  46. * variable for calculating the performance
  47. */
  48. final JButton playBtn = new JButton();
  49. final JButton timeResetBtn = new JButton();
  50. final JButton timeForwardBtn = new JButton();
  51. final JButton timeBackwardBtn = new JButton();
  52. final JLabel iterationsLabel = new JLabel("Iterations:", SwingConstants.CENTER);
  53. private final Control control;
  54. private final JSlider speedSlider = new JSlider();
  55. private final Timer timer;
  56. JTextField iterationsField;
  57. JLabel hint = new JLabel("Invalid", SwingConstants.RIGHT);
  58. JSlider timeSlider = new JSlider() {
  59. {
  60. // Make the slider jump to mouse position on left click
  61. MouseListener[] listeners = getMouseListeners();
  62. for (MouseListener l : listeners) {
  63. removeMouseListener(l); // remove UI-installed TrackListener
  64. }
  65. final BasicSliderUI ui = (BasicSliderUI) getUI();
  66. BasicSliderUI.TrackListener tl = ui.new TrackListener() {
  67. // this is where we jump to absolute value of click
  68. @Override
  69. public void mouseClicked(MouseEvent e) {
  70. Point p = e.getPoint();
  71. int value = ui.valueForXPosition(p.x);
  72. setValue(value);
  73. }
  74. // disable check that will invoke scrollDueToClickInTrack
  75. @Override
  76. public boolean shouldScroll(int dir) {
  77. return false;
  78. }
  79. };
  80. addMouseListener(tl);
  81. }
  82. };
  83. private ScheduledFuture<?> futureTask;
  84. private int dragResetIteration = 0;
  85. private boolean running = false;
  86. /**
  87. * Constructor
  88. *
  89. * @param cont the Controller
  90. */
  91. public TimePanel(Control cont) {
  92. super();
  93. this.control = cont;
  94. // One Iteration
  95. timer = new Timer(0, clicked -> timerAction());
  96. // Time Slider. Panels and Buttons
  97. this.setLayout(new BorderLayout(0, 0));
  98. this.setBorder(null);
  99. // Slider
  100. timeSlider.setPaintTicks(true);
  101. timeSlider.setPaintLabels(true);
  102. timeSlider.setMajorTickSpacing(
  103. (int) Math.ceil(((double) cont.getModel().getMaxIterations()) / 20));
  104. timeSlider.setMinorTickSpacing(
  105. (int) Math.ceil(((double) cont.getModel().getMaxIterations()) / 100));
  106. timeSlider.setToolTipText("Time Slider");
  107. timeSlider.setMaximum(cont.getModel().getMaxIterations() - 1);
  108. timeSlider.setValue(0);
  109. timeSlider.addChangeListener(
  110. changeEvent -> control.getModel().setCurrentIteration(timeSlider.getValue()));
  111. this.setBorder(null);
  112. timeSlider.addChangeListener(changeEvent -> {
  113. control.updateStateForIteration(timeSlider.getValue());
  114. });
  115. timeSlider.addMouseListener(new MouseAdapter() {
  116. @Override
  117. public void mousePressed(MouseEvent e) {
  118. dragResetIteration = cont.getModel().getCurrentIteration();
  119. }
  120. });
  121. timeSlider.addMouseMotionListener(new MouseAdapter() {
  122. @Override
  123. public void mouseDragged(MouseEvent e) {
  124. if (dragResetIteration != cont.getModel().getCurrentIteration()) {
  125. if (running) {
  126. play();
  127. }
  128. }
  129. }
  130. });
  131. // Panel
  132. JPanel timeBtnPanel = new JPanel();
  133. timeBtnPanel.setBorder(null);
  134. timeBtnPanel.setLayout(new FlowLayout(FlowLayout.CENTER));
  135. // Buttons
  136. playBtn.setToolTipText("Play");
  137. playBtn.setContentAreaFilled(false);
  138. playBtn.setBorderPainted(false);
  139. playBtn.setBorder(null);
  140. playBtn.setIcon(new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Play, 30, 30)));
  141. playBtn.addActionListener(clicked -> play());
  142. timeResetBtn.setToolTipText("Reset");
  143. timeResetBtn.setContentAreaFilled(false);
  144. timeResetBtn.setBorder(null);
  145. timeResetBtn.setIcon(
  146. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Reset, 30, 30)));
  147. timeResetBtn.addActionListener(new ActionListener() {
  148. public void actionPerformed(ActionEvent ae) {
  149. timeSlider.setValue(timeSlider.getMinimum());
  150. control.getModel().setCurrentIteration(timeSlider.getValue());
  151. control.resetSimulation();
  152. control.updateStateForCurrentIteration();
  153. if (running) {
  154. play();
  155. }
  156. }
  157. });
  158. timeForwardBtn.setToolTipText("Forward");
  159. timeForwardBtn.setContentAreaFilled(false);
  160. timeForwardBtn.setBorder(null);
  161. timeForwardBtn.setIcon(
  162. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Forward, 30, 30)));
  163. timeForwardBtn.addActionListener(new ActionListener() {
  164. public void actionPerformed(ActionEvent ae) {
  165. timeSlider.setValue(timeSlider.getValue() + 1);
  166. control.getModel().setCurrentIteration(timeSlider.getValue());
  167. }
  168. });
  169. timeBackwardBtn.setToolTipText("Backward");
  170. timeBackwardBtn.setBorder(null);
  171. timeBackwardBtn.setIcon(
  172. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Backward, 30, 30)));
  173. timeBackwardBtn.addActionListener(new ActionListener() {
  174. public void actionPerformed(ActionEvent ae) {
  175. timeSlider.setValue(timeSlider.getValue() - 1);
  176. control.getModel().setCurrentIteration(timeSlider.getValue());
  177. }
  178. });
  179. timeBtnPanel.add(playBtn);
  180. timeBtnPanel.add(Box.createRigidArea(new Dimension(10, 0)));
  181. timeBtnPanel.add(timeResetBtn);
  182. timeBtnPanel.add(Box.createRigidArea(new Dimension(10, 0)));
  183. timeBtnPanel.add(timeForwardBtn);
  184. // Speed Panel
  185. JPanel speedPanel = new JPanel();
  186. JLabel simSpeedLabel = new JLabel("Speed:");
  187. speedPanel.add(simSpeedLabel);
  188. speedPanel.add(speedSlider);
  189. speedSlider.setPaintTicks(true);
  190. speedSlider.setPaintLabels(true);
  191. speedSlider.setMaximum(6);
  192. speedSlider.setMinimum(0);
  193. speedSlider.setValue(1);
  194. speedSlider.setPaintLabels(true);
  195. Hashtable<Integer, JLabel> table = new Hashtable<Integer, JLabel>();
  196. table.put(0, new JLabel("1x"));
  197. table.put(1, new JLabel("2x"));
  198. table.put(2, new JLabel("4x"));
  199. table.put(3, new JLabel("8x"));
  200. table.put(4, new JLabel("16x"));
  201. table.put(5, new JLabel("32x"));
  202. table.put(6, new JLabel("64x"));
  203. speedSlider.setLabelTable(table);
  204. speedSlider.addChangeListener(new ChangeListener() {
  205. @Override
  206. public void stateChanged(ChangeEvent e) {
  207. /*
  208. * Shifting Powers of two: e.g. 1<<0 -> 1 step per Second 1<<3 -> 8 steps per
  209. * Second and so on,
  210. */
  211. int calculationsPerSecond = 1 << speedSlider.getValue();
  212. GuiSettings.timerSpeed = (1024 >> speedSlider.getValue());
  213. speedSlider.setToolTipText("Speed: " + calculationsPerSecond + " Calculations per Second.");
  214. }
  215. });
  216. speedSlider.setToolTipText("Change the Number of Calculations per Secons");
  217. // Buttons and Speed Panel
  218. JPanel btnAndSpeedPanel = new JPanel();
  219. btnAndSpeedPanel.setLayout(new BorderLayout(0, 0));
  220. btnAndSpeedPanel.setBorder(null);
  221. btnAndSpeedPanel.add(timeBtnPanel, BorderLayout.NORTH);
  222. btnAndSpeedPanel.add(speedPanel, BorderLayout.CENTER);
  223. JPanel iterationsPanel = new JPanel();
  224. iterationsPanel.setLayout(new GridBagLayout());
  225. GridBagConstraints c = new GridBagConstraints();
  226. c.anchor = GridBagConstraints.CENTER;
  227. c.fill = GridBagConstraints.HORIZONTAL;
  228. c.gridx = 0;
  229. c.gridy = 0;
  230. iterationsPanel.add(iterationsLabel, c);
  231. hint.setForeground(Color.red);
  232. hint.setText(" ");
  233. iterationsField = new JTextField(6);// Considering hundreds of thousands in an extreme case
  234. iterationsField.setText("" + cont.getModel().getMaxIterations());
  235. iterationsField.setToolTipText("0-" + MAX_ITERATIONS);
  236. iterationsField.addActionListener(this);
  237. ScheduledThreadPoolExecutor s = new ScheduledThreadPoolExecutor(1);
  238. iterationsField.addCaretListener((e) -> {
  239. try {
  240. iterationsField.setBackground(Color.WHITE);// red stings
  241. if (futureTask != null) {
  242. futureTask.cancel(true);
  243. }
  244. futureTask = s.schedule((Runnable) this::updateIterationsInput, 1, TimeUnit.SECONDS);
  245. hint.setText(" ");
  246. } catch (NumberFormatException n) {
  247. iterationsField.setBackground(ColorPreference.TimePanel.Invalid);// red stings
  248. hint.setText("Invalid");
  249. }
  250. });
  251. iterationsField.addFocusListener((LostFocusListener) (e) -> updateIterationsInput());
  252. c.gridy = 1;
  253. iterationsPanel.add(iterationsField, c);
  254. c.gridy = 2;
  255. iterationsPanel.add(hint, c);
  256. // iterationsPanel.add(new JLabel(), BorderLayout.SOUTH);
  257. JPanel timePanel = new JPanel();
  258. timePanel.setLayout(new BorderLayout());
  259. ;
  260. timePanel.add(iterationsPanel, BorderLayout.WEST);
  261. timePanel.add(timeSlider, BorderLayout.CENTER);
  262. this.add(btnAndSpeedPanel, BorderLayout.WEST);
  263. add(timePanel);
  264. // Disable Keys
  265. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0, false), "UP_ARROW");
  266. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0, false), "DOWN_ARROW");
  267. timeSlider.getInputMap()
  268. .put(KeyStroke.getKeyStroke(KeyEvent.VK_PAGE_DOWN, 0, false), "PAGE_DOWN");
  269. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_PAGE_UP, 0, false), "PAGE_UP");
  270. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_END, 0, false), "END");
  271. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_HOME, 0, false), "HOME");
  272. // Left arrow Key
  273. timeSlider.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_LEFT, 0, false), "LEFT_ARROW");
  274. timeSlider.getActionMap().put("LEFT_ARROW", new AbstractAction() {
  275. @Override
  276. public void actionPerformed(ActionEvent e) {
  277. timeSlider.setValue(timeSlider.getValue() - 1);
  278. control.resetSimulation();
  279. control.calculateStateForCurrentIteration();
  280. }
  281. });
  282. // Right arrow Key
  283. timeSlider.getInputMap()
  284. .put(KeyStroke.getKeyStroke(KeyEvent.VK_RIGHT, 0, false), "Right_ARROW");
  285. timeSlider.getActionMap().put("Right_ARROW", new AbstractAction() {
  286. @Override
  287. public void actionPerformed(ActionEvent e) {
  288. timeSlider.setValue(timeSlider.getValue() + 1);
  289. }
  290. });
  291. }
  292. public void timerAction() {
  293. timeSlider.setValue(timeSlider.getValue() + 1);
  294. control.getModel().setCurrentIteration(timeSlider.getValue());
  295. timer.setDelay(GuiSettings.timerSpeed);
  296. if (timeSlider.getValue() >= control.getModel().getMaxIterations() - 1) {
  297. running = false;
  298. playBtn.setIcon(
  299. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Play, 30, 30)));
  300. timer.stop();
  301. }
  302. }
  303. public void play() {
  304. if (control.getModel().getCurrentIteration() == control.getModel().getMaxIterations() - 1) {
  305. timeSlider.setValue(timeSlider.getMinimum());
  306. }
  307. running = !running;
  308. if (running) {
  309. timer.setDelay(GuiSettings.timerSpeed);
  310. timer.start();
  311. playBtn.setIcon(
  312. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Pause, 30, 30)));
  313. } else {
  314. timer.stop();
  315. playBtn.setIcon(
  316. new ImageIcon(Import.loadImage(ImagePreference.Button.TimePanel.Play, 30, 30)));
  317. }
  318. }
  319. @Override
  320. public void actionPerformed(ActionEvent arg0) {
  321. updateIterationsInput();
  322. }
  323. /**
  324. * Update the Text field and apply valid changes to update Iterations from to model.
  325. * <p>
  326. * Executed by user input.
  327. */
  328. private void updateIterationsInput() {
  329. try {
  330. int iterations = Integer.parseInt(iterationsField.getText());
  331. // iterationsLblHint.reset();
  332. boolean resetField = true;
  333. if (iterations < 1) {
  334. iterations = 1;
  335. } else if (iterations > MAX_ITERATIONS) {
  336. iterations = MAX_ITERATIONS;
  337. } else {
  338. resetField = false;
  339. }
  340. if (resetField) {
  341. iterationsField.setText("" + iterations);
  342. }
  343. control.getModel().setIterations(Integer.parseInt(iterationsField.getText()));
  344. timeSlider.setMaximum(control.getModel().getMaxIterations() - 1);
  345. timeSlider.setLabelTable(null);// Otherwise the ticks won't update
  346. timeSlider.setMajorTickSpacing((int) Math.ceil(((double) iterations) / 20));
  347. timeSlider.setMinorTickSpacing(
  348. (int) Math.ceil(((double) iterations) / 100));// Even though the final mark
  349. // can't actually be
  350. // reached.
  351. } catch (NumberFormatException ignored) {
  352. }
  353. }
  354. }