Split out summary panel
[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                                                                 new Burn.BurnProgressListener() {\r
161                                                                         @Override\r
162                                                                         public void setProgress(float f) {\r
163                                                                                 int pct = (int)(f*100);\r
164                                                                                 bar.setValue(pct);\r
165                                                                                 Amount<Length> web = motor.getGrain().webThickness();\r
166                                                                                 Amount<Length> remaining = web.times(1.0 - f);\r
167                                                                                 \r
168                                                                                 progress.setText("Progress: " + pct + "% (" + RocketScience.ammountToRoundedString(remaining) + " web thickness remaining)");\r
169                                                                                 if ( currentThread != me ){\r
170                                                                                         throw new BurnCanceled();\r
171                                                                                 }\r
172                                                                         }\r
173                                                                 });\r
174 \r
175                                                 final BurnPanel bp = new BurnPanel(b);\r
176                                                 SwingUtilities.invokeLater(new Thread() {\r
177                                                         public void run() {\r
178                                                                 remove(bar);\r
179                                                                 remove(progress);\r
180                                                                 add(bp, BorderLayout.CENTER);\r
181 \r
182                                                                 for (BurnWatcher bw : burnWatchers)\r
183                                                                         bw.replace(burn, b);\r
184                                                                 burn = b;\r
185 \r
186                                                                 revalidate();\r
187                                                         }\r
188                                                 });\r
189                                         } catch (BurnCanceled c){\r
190                                                 log.info("Burn Canceled!");\r
191                                         } catch (Exception e) {\r
192                                                 remove(bar);\r
193                                                 JTextArea t = new JTextArea(e.getMessage());\r
194                                                 t.setEditable(false);\r
195                                                 add(t);\r
196                                         }\r
197                                 }\r
198                         };\r
199                         currentThread.start();\r
200                 }\r
201         }\r
202 \r
203         private class GrainEditor extends JSplitPane {\r
204                 private static final long serialVersionUID = 1L;\r
205 \r
206                 public GrainEditor(final Grain g) {\r
207                         super(JSplitPane.HORIZONTAL_SPLIT);\r
208                         setName("Grain Geometry");\r
209                         setRightComponent(new GrainPanel(g));\r
210                         if (g instanceof Grain.Composite) {\r
211                                 final JPanel p = new JPanel();\r
212                                 p.setLayout(new BoxLayout(p, BoxLayout.Y_AXIS));\r
213                                 p.add(new Editor(g));\r
214                                 for (Grain gg : ((Grain.Composite) g).getGrains()) {\r
215                                         final int grainEditorIndex = p.getComponentCount() + 1;\r
216                                         p.add(new Chooser<Grain>(gg, grainTypes) {\r
217                                                 private static final long serialVersionUID = 1L;\r
218 \r
219                                                 @Override\r
220                                                 protected void choiceMade(Grain ng) {\r
221                                                         if (g instanceof MultiGrain) {\r
222                                                                 ((MultiGrain) g).setGrain(ng);\r
223                                                                 p.remove(grainEditorIndex);\r
224                                                                 p.add(new Editor(ng), grainEditorIndex);\r
225                                                                 p.remove(0);\r
226                                                                 p.add(new Editor(g), 0);\r
227                                                         }\r
228                                                 }\r
229                                         });\r
230                                         p.add(new Editor(gg));\r
231                                         if (gg instanceof ChangeListening.Subject) {\r
232                                                 ((ChangeListening.Subject) gg)\r
233                                                                 .addPropertyChangeListener(MotorEditor.this);\r
234                                         }\r
235                                 }\r
236                                 setLeftComponent(p);\r
237                         } else {\r
238                                 setLeftComponent(new Editor(g));\r
239                         }\r
240                         // setDividerLocation(.25);\r
241                         // setResizeWeight(.25);\r
242                         if (g instanceof ChangeListening.Subject) {\r
243                                 ((ChangeListening.Subject) g)\r
244                                                 .addPropertyChangeListener(MotorEditor.this);\r
245                         }\r
246                 }\r
247         }\r
248 \r
249         private class CaseEditor extends JSplitPane implements ComponentListener {\r
250                 private static final long serialVersionUID = 1L;\r
251 \r
252                 public CaseEditor(Nozzle n, Chamber c) {\r
253                         super(JSplitPane.VERTICAL_SPLIT);\r
254                         setName("General Parameters");\r
255                         this.addComponentListener(this);\r
256                         \r
257                         JPanel parts = new JPanel();\r
258                         parts.setLayout(new BoxLayout(parts, BoxLayout.X_AXIS));\r
259                         setTopComponent(parts);\r
260                         setBottomComponent(new HardwarePanel(n, c));\r
261                         \r
262                         JPanel nameAndFuel = new JPanel();\r
263                         nameAndFuel.setLayout(new BoxLayout(nameAndFuel, BoxLayout.Y_AXIS));\r
264 \r
265                         nameAndFuel.add(new JLabel("Name:"));\r
266                         nameAndFuel.add(new JTextField(motor.getName()) {\r
267                                 private static final long serialVersionUID = 1L;\r
268                                 {\r
269                                         setMinimumSize(new Dimension(200, 20));\r
270                                         setMaximumSize(new Dimension(Short.MAX_VALUE, 20));\r
271                                         final JTextField t = this;\r
272                                         addFocusListener(new FocusListener() {\r
273 \r
274                                                 @Override\r
275                                                 public void focusLost(FocusEvent e) {\r
276                                                         String n = t.getText();\r
277                                                         if (!"".equals(n) && !n.equals(motor.getName())) {\r
278                                                                 motor.setName(n);\r
279                                                         } else {\r
280                                                                 t.setText(motor.getName());\r
281                                                         }\r
282                                                 }\r
283 \r
284                                                 @Override\r
285                                                 public void focusGained(FocusEvent e) {\r
286 \r
287                                                 }\r
288                                         });\r
289 \r
290                                 }\r
291                         });\r
292                         nameAndFuel.add(new JLabel("Fuel:"));\r
293                         nameAndFuel.add( new JComboBox(availableFuels){\r
294                                 {\r
295                                         this.setSelectedItem(motor.getFuel());\r
296                                 }\r
297                                 private static final long serialVersionUID = 1L;\r
298                                 {\r
299                                         setMinimumSize(new Dimension(200, 20));\r
300                                         setMaximumSize(new Dimension(Short.MAX_VALUE, 20));\r
301                                         addActionListener(new ActionListener(){\r
302                                                 @Override\r
303                                                 public void actionPerformed(ActionEvent e) {\r
304                                                         motor.setFuel((Fuel)getSelectedItem());\r
305                                                         System.out.println("FUEL CHANGED");\r
306                                                 }});\r
307                                 }\r
308                         });\r
309                         nameAndFuel.add(Box.createVerticalGlue());\r
310                         parts.add(nameAndFuel);\r
311                         \r
312                         JPanel casing = new JPanel();\r
313                         casing.setLayout(new BoxLayout(casing, BoxLayout.Y_AXIS));\r
314                         casing.add(new JLabel("Casing:"));\r
315                         casing.add(new Editor(c));\r
316                         parts.add(casing);\r
317                         \r
318                         JPanel nozzle = new JPanel();\r
319                         nozzle.setLayout(new BoxLayout(nozzle, BoxLayout.Y_AXIS));\r
320                         nozzle.add(new JLabel("Nozzle:"));\r
321                         nozzle.add(new Editor(n));\r
322                         parts.add(nozzle);\r
323 \r
324                         if (n instanceof ChangeListening.Subject) {\r
325                                 ((ChangeListening.Subject) n)\r
326                                                 .addPropertyChangeListener(MotorEditor.this);\r
327                         }\r
328                         if (c instanceof ChangeListening.Subject) {\r
329                                 ((ChangeListening.Subject) c)\r
330                                                 .addPropertyChangeListener(MotorEditor.this);\r
331                         }\r
332                 }\r
333 \r
334                 @Override\r
335                 public void componentHidden(ComponentEvent arg0) {\r
336 \r
337                 }\r
338 \r
339                 @Override\r
340                 public void componentMoved(ComponentEvent arg0) {\r
341 \r
342                 }\r
343 \r
344                 @Override\r
345                 public void componentResized(ComponentEvent arg0) {\r
346                         setResizeWeight(.5);\r
347                         setDividerLocation(.5);\r
348                 }\r
349 \r
350                 @Override\r
351                 public void componentShown(ComponentEvent arg0) {\r
352 \r
353                 }\r
354         }\r
355 \r
356 \r
357         public MotorEditor(Motor m, Collection<Fuel> fuels) {\r
358                 super(JTabbedPane.BOTTOM);\r
359                 for ( Fuel f : fuels )\r
360                         addFuel(f);\r
361                 setMotor(m);\r
362         }\r
363 \r
364         public Motor getMotor() {\r
365                 return motor;\r
366         }\r
367 \r
368 \r
369         private void setMotor(Motor m) {\r
370                 if (motor != null)\r
371                         motor.removePropertyChangeListener(this);\r
372                 motor = m;\r
373                 motor.addPropertyChangeListener(this);\r
374                 if (grainEditor != null)\r
375                         remove(grainEditor);\r
376                 while (getTabCount() > 1)\r
377                         removeTabAt(1);\r
378                 add(new CaseEditor(motor.getNozzle(), motor.getChamber()), CASING_TAB);\r
379                 add(new GrainEditor(motor.getGrain()), GRAIN_TAB);\r
380                 add(bt = new BurnTab(), BURN_TAB);\r
381         }\r
382 \r
383         public static Motor defaultMotor() {\r
384                 Motor m = new Motor();\r
385                 m.setName("Example Motor");\r
386                 m.setFuel(new KNSU());\r
387 \r
388                 CylindricalChamber c = new CylindricalChamber();\r
389                 c.setLength(Amount.valueOf(200, SI.MILLIMETER));\r
390                 c.setID(Amount.valueOf(30, SI.MILLIMETER));\r
391                 m.setChamber(c);\r
392 \r
393                 Schedule40 pvc = new Schedule40();\r
394                 pvc.setLength(Amount.valueOf(200, SI.MILLIMETER));\r
395                 m.setChamber(pvc);\r
396                 \r
397                 CoredCylindricalGrain g = new CoredCylindricalGrain();\r
398                 try {\r
399                         g.setLength(Amount.valueOf(70, SI.MILLIMETER));\r
400                         g.setOD(Amount.valueOf(30, SI.MILLIMETER));\r
401                         g.setID(Amount.valueOf(10, SI.MILLIMETER));\r
402                 } catch (PropertyVetoException v) {\r
403                         throw new Error(v);\r
404                 }\r
405                 \r
406 \r
407                 m.setGrain(new MultiGrain(g, 2));\r
408 \r
409                 ConvergentDivergentNozzle n = new ConvergentDivergentNozzle();\r
410                 n.setThroatDiameter(Amount.valueOf(7.962, SI.MILLIMETER));\r
411                 n.setExitDiameter(Amount.valueOf(13.79, SI.MILLIMETER));\r
412                 n.setEfficiency(.85);\r
413                 m.setNozzle(n);\r
414 \r
415                 return m;\r
416         }\r
417 \r
418         public void focusOnObject(Object o) {\r
419                 if (o instanceof Grain)\r
420                         setSelectedIndex(GRAIN_TAB);\r
421                 if (o instanceof Chamber || o instanceof Nozzle)\r
422                         setSelectedIndex(CASING_TAB);\r
423         }\r
424 \r
425         public void addBurnWatcher(BurnWatcher bw) {\r
426                 burnWatchers.add(bw);\r
427         }\r
428 \r
429         @Deprecated\r
430         public void showAsWindow() {\r
431                 JFrame f = new JFrame();\r
432                 f.setSize(1024, 768);\r
433                 f.setContentPane(this);\r
434                 f.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);\r
435                 f.setVisible(true);\r
436         }\r
437 \r
438         public static void main(String args[]) throws Exception {\r
439                 try {\r
440                         UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());\r
441                 } catch (Exception e1) {\r
442                         e1.printStackTrace();\r
443                 }\r
444                 Vector<Fuel> ff = new Vector<Fuel>();\r
445                 ff.add(new KNSU());\r
446                 //new MotorEditor(defaultMotor(), ff).showAsWindow();\r
447         }\r
448 \r
449         public void propertyChange(PropertyChangeEvent evt) {\r
450                 // Dont re-burn for a name change!\r
451                 if (!evt.getPropertyName().equals("Name")){\r
452                         bt.reBurn();\r
453                 } else {\r
454                         for (BurnWatcher bw : burnWatchers)\r
455                                 bw.replace(burn, burn);\r
456                 }\r
457         }\r
458 \r
459 }\r