altosuilib: Adapt to AltosFlightSeries data processing plan
[fw/altos] / altosuilib / AltosFlightStatsTable.java
1 /*
2  * Copyright © 2011 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 org.altusmetrum.altosuilib_11;
20
21 import java.awt.*;
22 import javax.swing.*;
23 import java.util.*;
24 import org.altusmetrum.altoslib_11.*;
25
26 public class AltosFlightStatsTable extends JComponent implements AltosFontListener {
27         GridBagLayout   layout;
28
29         LinkedList<FlightStat> flight_stats = new LinkedList<FlightStat>();
30
31         class FlightStat implements AltosFontListener {
32                 JLabel          label;
33                 JTextField[]    value;
34
35                 public void font_size_changed(int font_size) {
36                         label.setFont(AltosUILib.label_font);
37                         for (int i = 0; i < value.length; i++)
38                                 value[i].setFont(AltosUILib.value_font);
39                 }
40
41                 public FlightStat(GridBagLayout layout, int y, String label_text, String ... values) {
42                         GridBagConstraints      c = new GridBagConstraints();
43                         c.insets = new Insets(AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad);
44                         c.weighty = 1;
45
46                         label = new JLabel(label_text);
47                         label.setFont(AltosUILib.label_font);
48                         label.setHorizontalAlignment(SwingConstants.LEFT);
49                         c.gridx = 0; c.gridy = y;
50                         c.anchor = GridBagConstraints.WEST;
51                         c.fill = GridBagConstraints.VERTICAL;
52                         c.weightx = 0;
53                         layout.setConstraints(label, c);
54                         add(label);
55
56                         value = new JTextField[values.length];
57                         for (int j = 0; j < values.length; j++) {
58                                 value[j] = new JTextField(values[j]);
59                                 value[j].setEditable(false);
60                                 value[j].setFont(AltosUILib.value_font);
61                                 value[j].setHorizontalAlignment(SwingConstants.RIGHT);
62                                 c.gridx = j+1; c.gridy = y;
63                                 c.anchor = GridBagConstraints.EAST;
64                                 c.fill = GridBagConstraints.BOTH;
65                                 c.weightx = 1;
66                                 layout.setConstraints(value[j], c);
67                                 add(value[j]);
68                         }
69                         flight_stats.add(this);
70                 }
71
72         }
73
74         public void font_size_changed(int font_size) {
75                 for (FlightStat f : flight_stats)
76                         f.font_size_changed(font_size);
77         }
78
79         static String pos(double p, String pos, String neg) {
80                 String  h = pos;
81                 if (p < 0) {
82                         h = neg;
83                         p = -p;
84                 }
85                 int deg = (int) Math.floor(p);
86                 double min = (p - Math.floor(p)) * 60.0;
87                 return String.format("%s %4d° %9.6f'", h, deg, min);
88         }
89
90         public AltosFlightStatsTable(AltosFlightStats stats) {
91                 layout = new GridBagLayout();
92
93                 setLayout(layout);
94                 int y = 0;
95                 new FlightStat(layout, y++, "Serial", String.format("%d", stats.serial));
96                 new FlightStat(layout, y++, "Flight", String.format("%d", stats.flight));
97                 if (stats.year != AltosLib.MISSING && stats.hour != AltosLib.MISSING)
98                         new FlightStat(layout, y++, "Date/Time",
99                                        String.format("%04d-%02d-%02d", stats.year, stats.month, stats.day),
100                                        String.format("%02d:%02d:%02d UTC", stats.hour, stats.minute, stats.second));
101                 else {
102                         if (stats.year != AltosLib.MISSING)
103                                 new FlightStat(layout, y++, "Date",
104                                                String.format("%04d-%02d-%02d", stats.year, stats.month, stats.day));
105                         if (stats.hour != AltosLib.MISSING)
106                                 new FlightStat(layout, y++, "Time",
107                                                String.format("%02d:%02d:%02d UTC", stats.hour, stats.minute, stats.second));
108                 }
109                 if (stats.max_height != AltosLib.MISSING) {
110                         new FlightStat(layout, y++, "Maximum height",
111                                        String.format("%5.0f m", stats.max_height),
112                                        String.format("%5.0f ft", AltosConvert.meters_to_feet(stats.max_height)));
113                 }
114                 if (stats.max_gps_height != AltosLib.MISSING) {
115                         new FlightStat(layout, y++, "Maximum GPS height",
116                                        String.format("%5.0f m", stats.max_gps_height),
117                                        String.format("%5.0f ft", AltosConvert.meters_to_feet(stats.max_gps_height)));
118                 }
119                 if (stats.max_speed != AltosLib.MISSING) {
120                         new FlightStat(layout, y++, "Maximum speed",
121                                        String.format("%5.0f m/s", stats.max_speed),
122                                        String.format("%5.0f fps", AltosConvert.mps_to_fps(stats.max_speed)),
123                                        String.format("Mach %4.1f", AltosConvert.meters_to_mach(stats.max_speed)));
124                 }
125                 if (stats.max_acceleration != AltosLib.MISSING)
126                         new FlightStat(layout, y++, "Maximum boost acceleration",
127                                        String.format("%5.0f m/s²", stats.max_acceleration),
128                                        String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.max_acceleration)),
129                                        String.format("%5.0f G", AltosConvert.meters_to_g(stats.max_acceleration)));
130                 if (stats.state_accel[AltosLib.ao_flight_boost] != AltosLib.MISSING)
131                         new FlightStat(layout, y++, "Average boost acceleration",
132                                        String.format("%5.0f m/s²", stats.state_accel[AltosLib.ao_flight_boost]),
133                                        String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.state_accel[AltosLib.ao_flight_boost])),
134                                        String.format("%5.0f G", AltosConvert.meters_to_g(stats.state_accel[AltosLib.ao_flight_boost])));
135                 if (stats.state_speed[AltosLib.ao_flight_drogue] != AltosLib.MISSING)
136                         new FlightStat(layout, y++, "Drogue descent rate",
137                                        String.format("%5.0f m/s", stats.state_speed[AltosLib.ao_flight_drogue]),
138                                        String.format("%5.0f ft/s", AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_drogue])));
139                 if (stats.state_speed[AltosLib.ao_flight_main] != AltosLib.MISSING)
140                         new FlightStat(layout, y++, "Main descent rate",
141                                        String.format("%5.0f m/s", stats.state_speed[AltosLib.ao_flight_main]),
142                                        String.format("%5.0f ft/s", AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_main])));
143                 if (stats.state_start[AltosLib.ao_flight_boost] < stats.state_end[AltosLib.ao_flight_coast])
144                         new FlightStat(layout, y++, "Ascent time",
145                                        String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_boost] - stats.state_start[AltosLib.ao_flight_boost],
146                                                      AltosLib.state_name(AltosLib.ao_flight_boost)),
147                                        String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_fast] - stats.state_start[AltosLib.ao_flight_fast],
148                                                      AltosLib.state_name(AltosLib.ao_flight_fast)),
149                                        String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_coast] - stats.state_start[AltosLib.ao_flight_coast],
150                                                      AltosLib.state_name(AltosLib.ao_flight_coast)));
151                 if (stats.state_start[AltosLib.ao_flight_drogue] < stats.state_end[AltosLib.ao_flight_main])
152                         new FlightStat(layout, y++, "Descent time",
153                                        String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_drogue] - stats.state_start[AltosLib.ao_flight_drogue],
154                                                      AltosLib.state_name(AltosLib.ao_flight_drogue)),
155                                        String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_main] - stats.state_start[AltosLib.ao_flight_main],
156                                                      AltosLib.state_name(AltosLib.ao_flight_main)));
157                 if (stats.state_start[AltosLib.ao_flight_boost] < stats.state_end[AltosLib.ao_flight_main])
158                         new FlightStat(layout, y++, "Flight time",
159                                        String.format("%6.1f s", stats.state_end[AltosLib.ao_flight_main] -
160                                                      stats.state_start[AltosLib.ao_flight_boost]));
161                 if (stats.has_gps) {
162                         new FlightStat(layout, y++, "Pad location",
163                                        pos(stats.pad_lat,"N","S"),
164                                        pos(stats.pad_lon,"E","W"));
165                         new FlightStat(layout, y++, "Last reported location",
166                                        pos(stats.lat,"N","S"),
167                                        pos(stats.lon,"E","W"));
168                 }
169         }
170 }