]> git.gag.com Git - fw/altos/blob - altosui/AltosUI.java
altos: Support normalized log
[fw/altos] / altosui / AltosUI.java
1 /*
2  * Copyright © 2010 Keith Packard <keithp@keithp.com>
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License along
15  * with this program; if not, write to the Free Software Foundation, Inc.,
16  * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
17  */
18
19 package altosui;
20
21 import java.awt.*;
22 import java.awt.event.*;
23 import javax.swing.*;
24 import java.io.*;
25 import java.util.concurrent.*;
26 import org.altusmetrum.altoslib_14.*;
27 import org.altusmetrum.altosuilib_14.*;
28
29 public class AltosUI extends AltosUIFrame implements AltosEepromGrapher {
30         public AltosVoice voice = new AltosVoice();
31
32         public static boolean load_library(Frame frame) {
33                 if (!Altos.load_library()) {
34                         JOptionPane.showMessageDialog(frame,
35                                                       String.format("No AltOS library in \"%s\"",
36                                                                     System.getProperty("java.library.path","<undefined>")),
37                                                       "Cannot load device access library",
38                                                       JOptionPane.ERROR_MESSAGE);
39                         return false;
40                 }
41                 return true;
42         }
43
44         void telemetry_window(AltosDevice device) {
45                 try {
46                         AltosFlightReader reader = new AltosTelemetryReader(new AltosSerial(device));
47                         if (reader != null)
48                                 new AltosFlightUI(voice, reader, device.getSerial());
49                 } catch (FileNotFoundException ee) {
50                         JOptionPane.showMessageDialog(AltosUI.this,
51                                                       ee.getMessage(),
52                                                       String.format ("Cannot open %s", device.toShortString()),
53                                                       JOptionPane.ERROR_MESSAGE);
54                 } catch (AltosSerialInUseException si) {
55                         JOptionPane.showMessageDialog(AltosUI.this,
56                                                       String.format("Device \"%s\" already in use",
57                                                                     device.toShortString()),
58                                                       "Device in use",
59                                                       JOptionPane.ERROR_MESSAGE);
60                 } catch (IOException ee) {
61                         JOptionPane.showMessageDialog(AltosUI.this,
62                                                       String.format ("Unknown I/O error on %s", device.toShortString()),
63                                                       "Unknown I/O error",
64                                                       JOptionPane.ERROR_MESSAGE);
65                 } catch (TimeoutException te) {
66                         JOptionPane.showMessageDialog(this,
67                                                       String.format ("Timeout on %s", device.toShortString()),
68                                                       "Timeout error",
69                                                       JOptionPane.ERROR_MESSAGE);
70                 } catch (InterruptedException ie) {
71                         JOptionPane.showMessageDialog(this,
72                                                       String.format("Interrupted %s", device.toShortString()),
73                                                       "Interrupted exception",
74                                                       JOptionPane.ERROR_MESSAGE);
75                 }
76         }
77
78         public void scan_device_selected(AltosDevice device) {
79                 telemetry_window(device);
80         }
81
82         Container       pane;
83         GridBagLayout   gridbag;
84
85         JButton addButton(int x, int y, String label) {
86                 GridBagConstraints      c;
87                 JButton                 b;
88
89                 c = new GridBagConstraints();
90                 c.gridx = x; c.gridy = y;
91                 c.fill = GridBagConstraints.BOTH;
92                 c.weightx = 1;
93                 c.weighty = 1;
94                 b = new JButton(label);
95
96                 //Dimension ps = b.getPreferredSize();
97
98                 gridbag.setConstraints(b, c);
99                 add(b, c);
100                 return b;
101         }
102
103         /* OSXAdapter interfaces */
104         public void macosx_file_handler(String path) {
105                 process_graph(null, new File(path));
106         }
107
108         public void macosx_quit_handler() {
109                 System.exit(0);
110         }
111
112         public void macosx_preferences_handler() {
113                 ConfigureAltosUI();
114         }
115
116         public AltosUI() {
117
118                 load_library(null);
119
120                 register_for_macosx_events();
121
122                 AltosUIPreferences.set_component(this);
123
124                 pane = getContentPane();
125                 gridbag = new GridBagLayout();
126                 pane.setLayout(gridbag);
127
128                 JButton b;
129
130                 b = addButton(0, 0, "Monitor Flight");
131                 b.addActionListener(new ActionListener() {
132                                         public void actionPerformed(ActionEvent e) {
133                                                 ConnectToDevice();
134                                         }
135                                 });
136                 b.setToolTipText("Connect to TeleDongle and monitor telemetry");
137                 b = addButton(1, 0, "Save Flight Data");
138                 b.addActionListener(new ActionListener() {
139                                         public void actionPerformed(ActionEvent e) {
140                                                 SaveFlightData();
141                                         }
142                                 });
143                 b.setToolTipText("Download and/or delete flight data from an altimeter");
144                 b = addButton(2, 0, "Replay Flight");
145                 b.addActionListener(new ActionListener() {
146                                         public void actionPerformed(ActionEvent e) {
147                                                 Replay();
148                                         }
149                                 });
150                 b.setToolTipText("Watch an old flight in real-time");
151                 b = addButton(3, 0, "Graph Data");
152                 b.addActionListener(new ActionListener() {
153                                         public void actionPerformed(ActionEvent e) {
154                                                 GraphData();
155                                         }
156                                 });
157                 b.setToolTipText("Present flight data in a graph and table of statistics");
158                 b = addButton(4, 0, "Export Data");
159                 b.addActionListener(new ActionListener() {
160                                         public void actionPerformed(ActionEvent e) {
161                                                 ExportData();
162                                         }
163                                 });
164                 b.setToolTipText("Convert flight data for a spreadsheet or GoogleEarth");
165                 b = addButton(0, 1, "Configure Altimeter");
166                 b.addActionListener(new ActionListener() {
167                                         public void actionPerformed(ActionEvent e) {
168                                                 ConfigureTeleMetrum();
169                                         }
170                                 });
171                 b.setToolTipText("Set flight, storage and communication parameters");
172                 b = addButton(1, 1, "Configure AltosUI");
173                 b.addActionListener(new ActionListener() {
174                                 public void actionPerformed(ActionEvent e) {
175                                         ConfigureAltosUI();
176                                 }
177                         });
178                 b.setToolTipText("Global AltosUI settings");
179
180                 b = addButton(2, 1, "Configure Ground Station");
181                 b.addActionListener(new ActionListener() {
182                                 public void actionPerformed(ActionEvent e) {
183                                         ConfigureTeleDongle();
184                                 }
185                         });
186
187                 b = addButton(3, 1, "Flash Image");
188                 b.addActionListener(new ActionListener() {
189                                 public void actionPerformed(ActionEvent e) {
190                                         FlashImage();
191                                 }
192                         });
193                 b.setToolTipText("Replace the firmware in any AltusMetrum product");
194
195                 b = addButton(4, 1, "Fire Igniter");
196                 b.addActionListener(new ActionListener() {
197                                 public void actionPerformed(ActionEvent e) {
198                                         FireIgniter();
199                                 }
200                         });
201                 b.setToolTipText("Remote control of igniters for deployment testing");
202                 b = addButton(0, 2, "Scan Channels");
203                 b.addActionListener(new ActionListener() {
204                                 public void actionPerformed(ActionEvent e) {
205                                         ScanChannels();
206                                 }
207                         });
208                 b.setToolTipText("Find what channel an altimeter is sending telemetry on");
209                 b = addButton(1, 2, "Load Maps");
210                 b.addActionListener(new ActionListener() {
211                                 public void actionPerformed(ActionEvent e) {
212                                         LoadMaps();
213                                 }
214                         });
215                 b.setToolTipText("Download satellite images for off-line flight monitoring");
216                 b = addButton(2, 2, "Monitor Idle");
217                 b.addActionListener(new ActionListener() {
218                                 public void actionPerformed(ActionEvent e) {
219                                         IdleMonitor();
220                                 }
221                         });
222                 b.setToolTipText("Check flight readiness of altimeter in idle mode");
223
224 //              b = addButton(3, 2, "Launch Controller");
225 //              b.addActionListener(new ActionListener() {
226 //                              public void actionPerformed(ActionEvent e) {
227 //                                      LaunchController();
228 //                              }
229 //                      });
230
231                 b = addButton(4, 2, "Quit");
232                 b.addActionListener(new ActionListener() {
233                                 public void actionPerformed(ActionEvent e) {
234                                         System.exit(0);
235                                 }
236                         });
237                 b.setToolTipText("Close all active windows and terminate AltosUI");
238
239                 setTitle("AltOS");
240
241                 pane.doLayout();
242                 pane.validate();
243
244                 doLayout();
245                 validate();
246
247                 setDefaultCloseOperation(JFrame.DO_NOTHING_ON_CLOSE);
248                 addWindowListener(new WindowAdapter() {
249                         @Override
250                         public void windowClosing(WindowEvent e) {
251                                 System.exit(0);
252                         }
253                 });
254
255                 setLocationByPlatform(false);
256
257                 /* Insets aren't set before the window is visible */
258                 setVisible(true);
259         }
260
261         private void ConnectToDevice() {
262                 AltosDevice     device = AltosDeviceUIDialog.show(AltosUI.this,
263                                                                 Altos.product_basestation);
264
265                 if (device != null)
266                         telemetry_window(device);
267         }
268
269         void ConfigureCallsign() {
270                 String  result;
271                 result = JOptionPane.showInputDialog(AltosUI.this,
272                                                      "Configure Callsign",
273                                                      AltosUIPreferences.callsign());
274                 if (result != null)
275                         AltosUIPreferences.set_callsign(result);
276         }
277
278         void ConfigureTeleMetrum() {
279                 new AltosConfigFC(AltosUI.this);
280         }
281
282         void ConfigureTeleDongle() {
283                 new AltosConfigTD(AltosUI.this);
284         }
285
286         void FlashImage() {
287                 AltosFlashUI.show(AltosUI.this);
288         }
289
290         void FireIgniter() {
291                 new AltosIgniteUI(AltosUI.this);
292         }
293
294         void ScanChannels() {
295                 new AltosScanUI(AltosUI.this, true);
296         }
297
298         void LoadMaps() {
299                 new AltosUIMapPreload(AltosUI.this);
300         }
301
302         void LaunchController() {
303                 new AltosLaunchUI(AltosUI.this);
304         }
305
306         /*
307          * Replay a flight from telemetry data
308          */
309         private void Replay() {
310                 AltosDataChooser chooser = new AltosDataChooser(
311                         AltosUI.this);
312
313                 AltosRecordSet set = chooser.runDialog();
314                 if (set != null) {
315                         AltosReplayReader reader = new AltosReplayReader(set, chooser.file());
316                         new AltosFlightUI(voice, reader);
317                 }
318         }
319
320         /* Connect to TeleMetrum, either directly or through
321          * a TeleDongle over the packet link
322          */
323
324         public void graph_flights(AltosEepromList flights) {
325                 for (AltosEepromLog flight : flights) {
326                         if (flight.graph_selected && flight.file != null) {
327                                 process_graph(this, flight.file);
328                         }
329                 }
330         }
331
332         private void SaveFlightData() {
333                 new AltosEepromManage(this, this, AltosLib.product_any);
334         }
335
336         private static AltosFlightSeries make_series(AltosRecordSet set) {
337                 AltosFlightSeries series = new AltosFlightSeries(set.cal_data());
338                 set.capture_series(series);
339                 series.finish();
340                 return series;
341         }
342
343         /* Load a flight log file and write out a CSV file containing
344          * all of the data in standard units
345          */
346
347         private void ExportData() {
348                 AltosDataChooser chooser;
349                 chooser = new AltosDataChooser(this);
350                 AltosRecordSet set = chooser.runDialog();
351                 if (set == null)
352                         return;
353                 AltosFlightSeries series = make_series(set);
354                 new AltosCSVUI(AltosUI.this, series, chooser.file());
355         }
356
357         private static boolean graph_file(AltosUI altosui, AltosRecordSet set, File file) {
358                 if (set == null)
359                         return false;
360                 if (!set.valid()) {
361                         JOptionPane.showMessageDialog(altosui,
362                                                       String.format("Failed to parse file %s", file),
363                                                       "Graph Failed",
364                                                       JOptionPane.ERROR_MESSAGE);
365                         return false;
366                 }
367                 try {
368                         new AltosGraphUI(set, file);
369                         return true;
370                 } catch (InterruptedException ie) {
371                 } catch (IOException ie) {
372                 }
373                 return false;
374         }
375
376         /* Load a flight log CSV file and display a pretty graph.
377          */
378
379         private void GraphData() {
380                 AltosDataChooser chooser;
381                 chooser = new AltosDataChooser(this);
382                 AltosRecordSet set = chooser.runDialog();
383                 graph_file(this, set, chooser.file());
384         }
385
386         private void ConfigureAltosUI() {
387                 new AltosConfigureUI(AltosUI.this, voice);
388         }
389
390         private void IdleMonitor() {
391                 try {
392                         new AltosIdleMonitorUI(this);
393                 } catch (Exception e) {
394                 }
395         }
396
397         static AltosWriter open_csv(File file) {
398                 try {
399                         return new AltosCSV(file);
400                 } catch (FileNotFoundException fe) {
401                         System.out.printf("%s\n", fe.getMessage());
402                         return null;
403                 }
404         }
405
406         static AltosWriter open_kml(File file) {
407                 try {
408                         return new AltosKML(file);
409                 } catch (FileNotFoundException fe) {
410                         System.out.printf("%s\n", fe.getMessage());
411                         return null;
412                 }
413         }
414
415         static AltosRecordSet record_set(File input) {
416                 try {
417                         return AltosLib.record_set(input);
418                 } catch (IOException ie) {
419                         String message = ie.getMessage();
420                         if (message == null)
421                                 message = String.format("%s (I/O error)", input.toString());
422                         System.err.printf("%s: %s\n", input.toString(), message);
423                 }
424                 return null;
425         }
426
427         static final int process_none = 0;
428         static final int process_csv = 1;
429         static final int process_kml = 2;
430         static final int process_graph = 3;
431         static final int process_replay = 4;
432         static final int process_summary = 5;
433         static final int process_oneline = 6;
434
435         static boolean process_csv(File input) {
436                 AltosRecordSet set = record_set(input);
437                 if (set == null)
438                         return false;
439
440                 File output = Altos.replace_extension(input,".csv");
441                 System.out.printf("Processing \"%s\" to \"%s\"\n", input, output);
442                 if (input.equals(output)) {
443                         System.out.printf("Not processing '%s'\n", input);
444                         return false;
445                 } else {
446                         AltosWriter writer = open_csv(output);
447                         if (writer == null)
448                                 return false;
449                         AltosFlightSeries series = make_series(set);
450                         writer.write(series);
451                         writer.close();
452                 }
453                 return true;
454         }
455
456         static boolean process_kml(File input) {
457                 AltosRecordSet set = record_set(input);
458                 if (set == null)
459                         return false;
460
461                 File output = Altos.replace_extension(input,".kml");
462                 System.out.printf("Processing \"%s\" to \"%s\"\n", input, output);
463                 if (input.equals(output)) {
464                         System.out.printf("Not processing '%s'\n", input);
465                         return false;
466                 } else {
467                         AltosWriter writer = open_kml(output);
468                         if (writer == null)
469                                 return false;
470                         AltosFlightSeries series = make_series(set);
471                         series.finish();
472                         writer.write(series);
473                         writer.close();
474                         return true;
475                 }
476         }
477
478         static AltosReplayReader replay_file(File file) {
479                 AltosRecordSet set = record_set(file);
480                 if (set == null)
481                         return null;
482                 return new AltosReplayReader(set, file);
483         }
484
485         static boolean process_replay(File file) {
486                 AltosReplayReader reader = replay_file(file);
487                 if (reader == null)
488                         return false;
489                 AltosFlightUI flight_ui = new AltosFlightUI(new AltosVoice(), reader);
490                 return true;
491         }
492
493         static boolean process_graph(AltosUI altosui, File file) {
494                 AltosRecordSet set = record_set(file);
495                 return graph_file(altosui, set, file);
496         }
497
498         static boolean process_summary(File file) {
499                 AltosRecordSet set = record_set(file);
500                 if (set == null)
501                         return false;
502                 System.out.printf("%s:\n", file.toString());
503                 AltosFlightSeries series = make_series(set);
504                 AltosFlightStats stats = new AltosFlightStats(series);
505                 if (stats.serial != AltosLib.MISSING)
506                         System.out.printf("Serial:       %5d\n", stats.serial);
507                 if (stats.flight != AltosLib.MISSING)
508                         System.out.printf("Flight:       %5d\n", stats.flight);
509                 if (stats.year != AltosLib.MISSING)
510                         System.out.printf("Date:    %04d-%02d-%02d\n",
511                                           stats.year, stats.month, stats.day);
512                 if (stats.hour != AltosLib.MISSING)
513                         System.out.printf("Time:      %02d:%02d:%02d UTC\n",
514                                           stats.hour, stats.minute, stats.second);
515                 if (stats.max_height != AltosLib.MISSING)
516                         System.out.printf("Max height:  %6.0f m    %6.0f ft\n",
517                                           stats.max_height,
518                                           AltosConvert.meters_to_feet(stats.max_height));
519                 if (stats.max_speed != AltosLib.MISSING)
520                         System.out.printf("Max speed:   %6.0f m/s  %6.0f ft/s  %6.4f Mach\n",
521                                           stats.max_speed,
522                                           AltosConvert.meters_to_feet(stats.max_speed),
523                                           AltosConvert.meters_to_mach(stats.max_speed));
524                 if (stats.max_acceleration != AltosLib.MISSING) {
525                         System.out.printf("Max accel:   %6.0f m/s² %6.0f ft/s² %6.2f g\n",
526                                           stats.max_acceleration,
527                                           AltosConvert.meters_to_feet(stats.max_acceleration),
528                                           AltosConvert.meters_to_g(stats.max_acceleration));
529                 }
530                 if (stats.state_speed[Altos.ao_flight_drogue] != AltosLib.MISSING)
531                         System.out.printf("Drogue rate: %6.0f m/s  %6.0f ft/s\n",
532                                           stats.state_speed[Altos.ao_flight_drogue],
533                                           AltosConvert.meters_to_feet(stats.state_speed[Altos.ao_flight_drogue]));
534                 if (stats.state_speed[Altos.ao_flight_main] != AltosLib.MISSING)
535                         System.out.printf("Main rate:   %6.0f m/s  %6.0f ft/s\n",
536                                           stats.state_speed[Altos.ao_flight_main],
537                                           AltosConvert.meters_to_feet(stats.state_speed[Altos.ao_flight_main]));
538                 if (stats.landed_time != AltosLib.MISSING &&
539                     stats.boost_time != AltosLib.MISSING &&
540                     stats.landed_time > stats.boost_time)
541                         System.out.printf("Flight time: %6.0f s\n",
542                                           stats.landed_time -
543                                           stats.boost_time);
544                 System.out.printf("\n");
545                 return true;
546         }
547
548         static boolean process_oneline(File file) {
549                 AltosRecordSet set = record_set(file);
550                 if (set == null)
551                         return false;
552                 System.out.printf("%s", file.toString());
553                 AltosFlightSeries series = make_series(set);
554                 AltosFlightStats stats = new AltosFlightStats(series);
555                 if (stats.max_height != AltosLib.MISSING)
556                         System.out.printf(" height  %6.0f m", stats.max_height);
557                 if (stats.max_speed != AltosLib.MISSING)
558                         System.out.printf(" speed   %6.0f m/s", stats.max_speed);
559                 if (stats.state_enter_speed[AltosLib.ao_flight_drogue] != AltosLib.MISSING)
560                         System.out.printf(" drogue-deploy   %6.0f m/s", stats.state_enter_speed[AltosLib.ao_flight_drogue]);
561                 if (stats.max_acceleration != AltosLib.MISSING)
562                         System.out.printf(" accel   %6.0f m/s²", stats.max_acceleration);
563                 System.out.printf("\n");
564                 return true;
565         }
566
567         public static void help(int code) {
568                 System.out.printf("Usage: altosui [OPTION]... [FILE]...\n");
569                 System.out.printf("  Options:\n");
570                 System.out.printf("    --replay <filename>\t\trelive the glory of past flights \n");
571                 System.out.printf("    --graph <filename>\t\tgraph a flight\n");
572                 System.out.printf("    --summary <filename>\t\tText summary of a flight\n");
573                 System.out.printf("    --oneline <filename>\t\tOne line summary of a flight\n");
574                 System.out.printf("    --csv\tgenerate comma separated output for spreadsheets, etc\n");
575                 System.out.printf("    --kml\tgenerate KML output for use with Google Earth\n");
576                 System.exit(code);
577         }
578
579         public static void main(final String[] args) {
580                 int     errors = 0;
581                 load_library(null);
582                 try {
583                         UIManager.setLookAndFeel(AltosUIPreferences.look_and_feel());
584                 } catch (Exception e) {
585                 }
586                 AltosUI altosui = null;
587
588                 /* Handle batch-mode */
589                 if (args.length == 0) {
590                         altosui = new AltosUI();
591                         java.util.List<AltosDevice> devices = AltosUSBDevice.list(Altos.product_basestation);
592                         if (devices != null)
593                                 for (AltosDevice device : devices)
594                                         altosui.telemetry_window(device);
595                 } else {
596                         int process = process_none;
597                         for (int i = 0; i < args.length; i++) {
598                                 if (args[i].equals("--help"))
599                                         help(0);
600                                 else if (args[i].equals("--replay"))
601                                         process = process_replay;
602                                 else if (args[i].equals("--kml"))
603                                         process = process_kml;
604                                 else if (args[i].equals("--csv"))
605                                         process = process_csv;
606                                 else if (args[i].equals("--graph"))
607                                         process = process_graph;
608                                 else if (args[i].equals("--summary"))
609                                         process = process_summary;
610                                 else if (args[i].equals("--oneline"))
611                                         process = process_oneline;
612                                 else if (args[i].startsWith("--"))
613                                         help(1);
614                                 else {
615                                         File file = new File(args[i]);
616                                         switch (process) {
617                                         case process_none:
618                                                 if (altosui == null)
619                                                         altosui = new AltosUI();
620                                         case process_graph:
621                                                 if (!process_graph(null, file))
622                                                         ++errors;
623                                                 break;
624                                         case process_replay:
625                                                 if (!process_replay(file))
626                                                         ++errors;
627                                                 break;
628                                         case process_kml:
629                                                 if (!process_kml(file))
630                                                         ++errors;
631                                                 break;
632                                         case process_csv:
633                                                 if (!process_csv(file))
634                                                         ++errors;
635                                                 break;
636                                         case process_summary:
637                                                 if (!process_summary(file))
638                                                         ++errors;
639                                                 break;
640                                         case process_oneline:
641                                                 if (!process_oneline(file))
642                                                         ++errors;
643                                                 break;
644                                         }
645                                 }
646                         }
647                 }
648                 if (errors != 0)
649                         System.exit(errors);
650         }
651 }