Add burnComplete function to progress listener
[sw/motorsim] / gui / com / billkuker / rocketry / motorsim / visual / workbench / MotorEditor.java
1 package com.billkuker.rocketry.motorsim.visual.workbench;\r
2 \r
3 import java.awt.BorderLayout;\r
4 import java.awt.Dimension;\r
5 import java.awt.event.ActionEvent;\r
6 import java.awt.event.ActionListener;\r
7 import java.awt.event.ComponentEvent;\r
8 import java.awt.event.ComponentListener;\r
9 import java.awt.event.FocusEvent;\r
10 import java.awt.event.FocusListener;\r
11 import java.beans.PropertyChangeEvent;\r
12 import java.beans.PropertyChangeListener;\r
13 import java.beans.PropertyVetoException;\r
14 import java.util.Collection;\r
15 import java.util.HashMap;\r
16 import java.util.List;\r
17 import java.util.Map;\r
18 import java.util.Vector;\r
19 \r
20 import javax.measure.quantity.Length;\r
21 import javax.measure.unit.SI;\r
22 import javax.swing.Box;\r
23 import javax.swing.BoxLayout;\r
24 import javax.swing.DefaultComboBoxModel;\r
25 import javax.swing.JButton;\r
26 import javax.swing.JComboBox;\r
27 import javax.swing.JFrame;\r
28 import javax.swing.JLabel;\r
29 import javax.swing.JPanel;\r
30 import javax.swing.JProgressBar;\r
31 import javax.swing.JSplitPane;\r
32 import javax.swing.JTabbedPane;\r
33 import javax.swing.JTextArea;\r
34 import javax.swing.JTextField;\r
35 import javax.swing.SwingUtilities;\r
36 import javax.swing.UIManager;\r
37 import javax.swing.WindowConstants;\r
38 \r
39 import org.apache.log4j.Logger;\r
40 import org.jscience.physics.amount.Amount;\r
41 \r
42 import com.billkuker.rocketry.motorsim.Burn;\r
43 import com.billkuker.rocketry.motorsim.Chamber;\r
44 import com.billkuker.rocketry.motorsim.ChangeListening;\r
45 import com.billkuker.rocketry.motorsim.ConvergentDivergentNozzle;\r
46 import com.billkuker.rocketry.motorsim.CylindricalChamber;\r
47 import com.billkuker.rocketry.motorsim.Fuel;\r
48 import com.billkuker.rocketry.motorsim.Grain;\r
49 import com.billkuker.rocketry.motorsim.Motor;\r
50 import com.billkuker.rocketry.motorsim.Nozzle;\r
51 import com.billkuker.rocketry.motorsim.RocketScience;\r
52 import com.billkuker.rocketry.motorsim.cases.Schedule40;\r
53 import com.billkuker.rocketry.motorsim.fuel.KNSU;\r
54 import com.billkuker.rocketry.motorsim.grain.CSlot;\r
55 import com.billkuker.rocketry.motorsim.grain.CoredCylindricalGrain;\r
56 import com.billkuker.rocketry.motorsim.grain.EndBurner;\r
57 import com.billkuker.rocketry.motorsim.grain.Finocyl;\r
58 import com.billkuker.rocketry.motorsim.grain.Moonburner;\r
59 import com.billkuker.rocketry.motorsim.grain.MultiGrain;\r
60 import com.billkuker.rocketry.motorsim.grain.RodAndTubeGrain;\r
61 import com.billkuker.rocketry.motorsim.grain.Star;\r
62 import com.billkuker.rocketry.motorsim.visual.BurnPanel;\r
63 import com.billkuker.rocketry.motorsim.visual.Editor;\r
64 import com.billkuker.rocketry.motorsim.visual.GrainPanel;\r
65 import com.billkuker.rocketry.motorsim.visual.HardwarePanel;\r
66 \r
67 public class MotorEditor extends JTabbedPane implements PropertyChangeListener {\r
68         private static final long serialVersionUID = 1L;\r
69         private static Logger log = Logger.getLogger(MotorEditor.class);\r
70         Motor motor;\r
71         GrainEditor grainEditor;\r
72         BurnTab bt;\r
73         Burn burn;\r
74 \r
75         private Vector<BurnWatcher> burnWatchers = new Vector<BurnWatcher>();\r
76         private DefaultComboBoxModel availableFuels = new DefaultComboBoxModel();\r
77         \r
78         public void addFuel(Fuel f){\r
79                 availableFuels.addElement(f);\r
80         }\r
81 \r
82         //private static final int XML_TAB = 0;\r
83         private static final int CASING_TAB = 0;\r
84         private static final int GRAIN_TAB = 1;\r
85         private static final int BURN_TAB = 2;\r
86 \r
87         private List<Class<? extends Grain>> grainTypes = new Vector<Class<? extends Grain>>();\r
88         {\r
89                 grainTypes.add(CoredCylindricalGrain.class);\r
90                 grainTypes.add(Finocyl.class);\r
91                 grainTypes.add(Star.class);\r
92                 grainTypes.add(Moonburner.class);\r
93                 grainTypes.add(RodAndTubeGrain.class);\r
94                 grainTypes.add(CSlot.class);\r
95                 grainTypes.add(EndBurner.class);\r
96         }\r
97 \r
98         private abstract class Chooser<T> extends JPanel {\r
99                 private static final long serialVersionUID = 1L;\r
100                 private List<Class<? extends T>> types;\r
101                 private Map<Class<? extends T>, T> old = new HashMap<Class<? extends T>, T>();\r
102 \r
103                 @SuppressWarnings("unchecked")\r
104                 public Chooser(T initial, List<Class<? extends T>> ts) {\r
105                         types = ts;\r
106                         if ( initial != null )\r
107                                 old.put((Class<? extends T>)initial.getClass(), initial);\r
108                         setLayout(new BoxLayout(this, BoxLayout.Y_AXIS));\r
109                         for (final Class<? extends T> c : types) {\r
110                                 JButton b = new JButton(c.getSimpleName());\r
111                                 add(b);\r
112                                 b.addActionListener(new ActionListener() {\r
113                                         public void actionPerformed(ActionEvent e) {\r
114                                                 try {\r
115                                                         T val = old.get(c);\r
116                                                         if ( val == null ){\r
117                                                                 System.err.println("CREATED NEW =========================");\r
118                                                                 val = c.newInstance();\r
119                                                                 old.put(c, val);\r
120                                                         }\r
121                                                         choiceMade(val);\r
122                                                 } catch (InstantiationException e1) {\r
123                                                         e1.printStackTrace();\r
124                                                 } catch (IllegalAccessException e1) {\r
125                                                         e1.printStackTrace();\r
126                                                 }\r
127                                         }\r
128                                 });\r
129                         }\r
130                 }\r
131 \r
132                 protected abstract void choiceMade(T o);\r
133         }\r
134 \r
135         private class BurnTab extends JPanel {\r
136                 private static final long serialVersionUID = 1L;\r
137                 private Thread currentThread;\r
138                 \r
139                 public BurnTab() {\r
140                         setLayout(new BorderLayout());\r
141                         setName("Simulation Results");\r
142                         reBurn();\r
143                 }\r
144                 \r
145                 private class BurnCanceled extends RuntimeException{\r
146                         private static final long serialVersionUID = 1L;\r
147                 };\r
148 \r
149                 public void reBurn() {\r
150                         removeAll();\r
151                         currentThread = new Thread() {\r
152                                 public void run() {\r
153                                         final Thread me = this;\r
154                                         final JProgressBar bar = new JProgressBar(0, 100);\r
155                                         add(bar, BorderLayout.NORTH);\r
156                                         final JLabel progress = new JLabel();\r
157                                         add(progress, BorderLayout.CENTER);\r
158                                         try {\r
159                                                 final Burn b = new Burn(motor);\r
160                                                 b.addBurnProgressListener(\r
161                                                                 new Burn.BurnProgressListener() {\r
162                                                                         @Override\r
163                                                                         public void burnComplete(){};\r
164                                                                         @Override\r
165                                                                         public void setProgress(float f) {\r
166                                                                                 int pct = (int)(f*100);\r
167                                                                                 bar.setValue(pct);\r
168                                                                                 Amount<Length> web = motor.getGrain().webThickness();\r
169                                                                                 Amount<Length> remaining = web.times(1.0 - f);\r
170                                                                                 \r
171                                                                                 progress.setText("Progress: " + pct + "% (" + RocketScience.ammountToRoundedString(remaining) + " web thickness remaining)");\r
172                                                                                 if ( currentThread != me ){\r
173                                                                                         throw new BurnCanceled();\r
174                                                                                 }\r
175                                                                         }\r
176                                                                 });\r
177                                                 b.burn();\r
178 \r
179                                                 final BurnPanel bp = new BurnPanel(b);\r
180                                                 SwingUtilities.invokeLater(new Thread() {\r
181                                                         public void run() {\r
182                                                                 remove(bar);\r
183                                                                 remove(progress);\r
184                                                                 add(bp, BorderLayout.CENTER);\r
185 \r
186                                                                 for (BurnWatcher bw : burnWatchers)\r
187                                                                         bw.replace(burn, b);\r
188                                                                 burn = b;\r
189 \r
190                                                                 revalidate();\r
191                                                         }\r
192                                                 });\r
193                                         } catch (BurnCanceled c){\r
194                                                 log.info("Burn Canceled!");\r
195                                         } catch (Exception e) {\r
196                                                 remove(bar);\r
197                                                 JTextArea t = new JTextArea(e.getMessage());\r
198                                                 t.setEditable(false);\r
199                                                 add(t);\r
200                                         }\r
201                                 }\r
202                         };\r
203                         currentThread.start();\r
204                 }\r
205         }\r
206 \r
207         private class GrainEditor extends JSplitPane {\r
208                 private static final long serialVersionUID = 1L;\r
209 \r
210                 public GrainEditor(final Grain g) {\r
211                         super(JSplitPane.HORIZONTAL_SPLIT);\r
212                         setName("Grain Geometry");\r
213                         setRightComponent(new GrainPanel(g));\r
214                         if (g instanceof Grain.Composite) {\r
215                                 final JPanel p = new JPanel();\r
216                                 p.setLayout(new BoxLayout(p, BoxLayout.Y_AXIS));\r
217                                 p.add(new Editor(g));\r
218                                 for (Grain gg : ((Grain.Composite) g).getGrains()) {\r
219                                         final int grainEditorIndex = p.getComponentCount() + 1;\r
220                                         p.add(new Chooser<Grain>(gg, grainTypes) {\r
221                                                 private static final long serialVersionUID = 1L;\r
222 \r
223                                                 @Override\r
224                                                 protected void choiceMade(Grain ng) {\r
225                                                         if (g instanceof MultiGrain) {\r
226                                                                 ((MultiGrain) g).setGrain(ng);\r
227                                                                 p.remove(grainEditorIndex);\r
228                                                                 p.add(new Editor(ng), grainEditorIndex);\r
229                                                                 p.remove(0);\r
230                                                                 p.add(new Editor(g), 0);\r
231                                                         }\r
232                                                 }\r
233                                         });\r
234                                         p.add(new Editor(gg));\r
235                                         if (gg instanceof ChangeListening.Subject) {\r
236                                                 ((ChangeListening.Subject) gg)\r
237                                                                 .addPropertyChangeListener(MotorEditor.this);\r
238                                         }\r
239                                 }\r
240                                 setLeftComponent(p);\r
241                         } else {\r
242                                 setLeftComponent(new Editor(g));\r
243                         }\r
244                         // setDividerLocation(.25);\r
245                         // setResizeWeight(.25);\r
246                         if (g instanceof ChangeListening.Subject) {\r
247                                 ((ChangeListening.Subject) g)\r
248                                                 .addPropertyChangeListener(MotorEditor.this);\r
249                         }\r
250                 }\r
251         }\r
252 \r
253         private class CaseEditor extends JSplitPane implements ComponentListener {\r
254                 private static final long serialVersionUID = 1L;\r
255 \r
256                 public CaseEditor(Nozzle n, Chamber c) {\r
257                         super(JSplitPane.VERTICAL_SPLIT);\r
258                         setName("General Parameters");\r
259                         this.addComponentListener(this);\r
260                         \r
261                         JPanel parts = new JPanel();\r
262                         parts.setLayout(new BoxLayout(parts, BoxLayout.X_AXIS));\r
263                         setTopComponent(parts);\r
264                         setBottomComponent(new HardwarePanel(n, c));\r
265                         \r
266                         JPanel nameAndFuel = new JPanel();\r
267                         nameAndFuel.setLayout(new BoxLayout(nameAndFuel, BoxLayout.Y_AXIS));\r
268 \r
269                         nameAndFuel.add(new JLabel("Name:"));\r
270                         nameAndFuel.add(new JTextField(motor.getName()) {\r
271                                 private static final long serialVersionUID = 1L;\r
272                                 {\r
273                                         setMinimumSize(new Dimension(200, 20));\r
274                                         setMaximumSize(new Dimension(Short.MAX_VALUE, 20));\r
275                                         final JTextField t = this;\r
276                                         addFocusListener(new FocusListener() {\r
277 \r
278                                                 @Override\r
279                                                 public void focusLost(FocusEvent e) {\r
280                                                         String n = t.getText();\r
281                                                         if (!"".equals(n) && !n.equals(motor.getName())) {\r
282                                                                 motor.setName(n);\r
283                                                         } else {\r
284                                                                 t.setText(motor.getName());\r
285                                                         }\r
286                                                 }\r
287 \r
288                                                 @Override\r
289                                                 public void focusGained(FocusEvent e) {\r
290 \r
291                                                 }\r
292                                         });\r
293 \r
294                                 }\r
295                         });\r
296                         nameAndFuel.add(new JLabel("Fuel:"));\r
297                         nameAndFuel.add( new JComboBox(availableFuels){\r
298                                 {\r
299                                         this.setSelectedItem(motor.getFuel());\r
300                                 }\r
301                                 private static final long serialVersionUID = 1L;\r
302                                 {\r
303                                         setMinimumSize(new Dimension(200, 20));\r
304                                         setMaximumSize(new Dimension(Short.MAX_VALUE, 20));\r
305                                         addActionListener(new ActionListener(){\r
306                                                 @Override\r
307                                                 public void actionPerformed(ActionEvent e) {\r
308                                                         motor.setFuel((Fuel)getSelectedItem());\r
309                                                         System.out.println("FUEL CHANGED");\r
310                                                 }});\r
311                                 }\r
312                         });\r
313                         nameAndFuel.add(Box.createVerticalGlue());\r
314                         parts.add(nameAndFuel);\r
315                         \r
316                         JPanel casing = new JPanel();\r
317                         casing.setLayout(new BoxLayout(casing, BoxLayout.Y_AXIS));\r
318                         casing.add(new JLabel("Casing:"));\r
319                         casing.add(new Editor(c));\r
320                         parts.add(casing);\r
321                         \r
322                         JPanel nozzle = new JPanel();\r
323                         nozzle.setLayout(new BoxLayout(nozzle, BoxLayout.Y_AXIS));\r
324                         nozzle.add(new JLabel("Nozzle:"));\r
325                         nozzle.add(new Editor(n));\r
326                         parts.add(nozzle);\r
327 \r
328                         if (n instanceof ChangeListening.Subject) {\r
329                                 ((ChangeListening.Subject) n)\r
330                                                 .addPropertyChangeListener(MotorEditor.this);\r
331                         }\r
332                         if (c instanceof ChangeListening.Subject) {\r
333                                 ((ChangeListening.Subject) c)\r
334                                                 .addPropertyChangeListener(MotorEditor.this);\r
335                         }\r
336                 }\r
337 \r
338                 @Override\r
339                 public void componentHidden(ComponentEvent arg0) {\r
340 \r
341                 }\r
342 \r
343                 @Override\r
344                 public void componentMoved(ComponentEvent arg0) {\r
345 \r
346                 }\r
347 \r
348                 @Override\r
349                 public void componentResized(ComponentEvent arg0) {\r
350                         setResizeWeight(.5);\r
351                         setDividerLocation(.5);\r
352                 }\r
353 \r
354                 @Override\r
355                 public void componentShown(ComponentEvent arg0) {\r
356 \r
357                 }\r
358         }\r
359 \r
360 \r
361         public MotorEditor(Motor m, Collection<Fuel> fuels) {\r
362                 super(JTabbedPane.BOTTOM);\r
363                 for ( Fuel f : fuels )\r
364                         addFuel(f);\r
365                 setMotor(m);\r
366         }\r
367 \r
368         public Motor getMotor() {\r
369                 return motor;\r
370         }\r
371 \r
372 \r
373         private void setMotor(Motor m) {\r
374                 if (motor != null)\r
375                         motor.removePropertyChangeListener(this);\r
376                 motor = m;\r
377                 motor.addPropertyChangeListener(this);\r
378                 if (grainEditor != null)\r
379                         remove(grainEditor);\r
380                 while (getTabCount() > 1)\r
381                         removeTabAt(1);\r
382                 add(new CaseEditor(motor.getNozzle(), motor.getChamber()), CASING_TAB);\r
383                 add(new GrainEditor(motor.getGrain()), GRAIN_TAB);\r
384                 add(bt = new BurnTab(), BURN_TAB);\r
385         }\r
386 \r
387         public static Motor defaultMotor() {\r
388                 Motor m = new Motor();\r
389                 m.setName("Example Motor");\r
390                 m.setFuel(new KNSU());\r
391 \r
392                 CylindricalChamber c = new CylindricalChamber();\r
393                 c.setLength(Amount.valueOf(200, SI.MILLIMETER));\r
394                 c.setID(Amount.valueOf(30, SI.MILLIMETER));\r
395                 m.setChamber(c);\r
396 \r
397                 Schedule40 pvc = new Schedule40();\r
398                 pvc.setLength(Amount.valueOf(200, SI.MILLIMETER));\r
399                 m.setChamber(pvc);\r
400                 \r
401                 CoredCylindricalGrain g = new CoredCylindricalGrain();\r
402                 try {\r
403                         g.setLength(Amount.valueOf(70, SI.MILLIMETER));\r
404                         g.setOD(Amount.valueOf(30, SI.MILLIMETER));\r
405                         g.setID(Amount.valueOf(10, SI.MILLIMETER));\r
406                 } catch (PropertyVetoException v) {\r
407                         throw new Error(v);\r
408                 }\r
409                 \r
410 \r
411                 m.setGrain(new MultiGrain(g, 2));\r
412 \r
413                 ConvergentDivergentNozzle n = new ConvergentDivergentNozzle();\r
414                 n.setThroatDiameter(Amount.valueOf(7.962, SI.MILLIMETER));\r
415                 n.setExitDiameter(Amount.valueOf(13.79, SI.MILLIMETER));\r
416                 n.setEfficiency(.85);\r
417                 m.setNozzle(n);\r
418 \r
419                 return m;\r
420         }\r
421 \r
422         public void focusOnObject(Object o) {\r
423                 if (o instanceof Grain)\r
424                         setSelectedIndex(GRAIN_TAB);\r
425                 if (o instanceof Chamber || o instanceof Nozzle)\r
426                         setSelectedIndex(CASING_TAB);\r
427         }\r
428 \r
429         public void addBurnWatcher(BurnWatcher bw) {\r
430                 burnWatchers.add(bw);\r
431         }\r
432 \r
433         @Deprecated\r
434         public void showAsWindow() {\r
435                 JFrame f = new JFrame();\r
436                 f.setSize(1024, 768);\r
437                 f.setContentPane(this);\r
438                 f.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);\r
439                 f.setVisible(true);\r
440         }\r
441 \r
442         public static void main(String args[]) throws Exception {\r
443                 try {\r
444                         UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());\r
445                 } catch (Exception e1) {\r
446                         e1.printStackTrace();\r
447                 }\r
448                 Vector<Fuel> ff = new Vector<Fuel>();\r
449                 ff.add(new KNSU());\r
450                 //new MotorEditor(defaultMotor(), ff).showAsWindow();\r
451         }\r
452 \r
453         public void propertyChange(PropertyChangeEvent evt) {\r
454                 // Dont re-burn for a name change!\r
455                 if (!evt.getPropertyName().equals("Name")){\r
456                         bt.reBurn();\r
457                 } else {\r
458                         for (BurnWatcher bw : burnWatchers)\r
459                                 bw.replace(burn, burn);\r
460                 }\r
461         }\r
462 \r
463 }\r