2 * Copyright © 2011 Keith Packard <keithp@keithp.com>
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.
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.
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.
19 package org.altusmetrum.altosuilib_12;
24 import org.altusmetrum.altoslib_12.*;
26 public class AltosFlightStatsTable extends JComponent implements AltosFontListener {
29 LinkedList<FlightStat> flight_stats = new LinkedList<FlightStat>();
31 class FlightStat implements AltosFontListener {
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);
41 public void set(String ... values) {
42 for (int j = 0; j < values.length; j++)
43 value[j].setText(values[j]);
46 public FlightStat(GridBagLayout layout, int y, String label_text, String ... values) {
47 GridBagConstraints c = new GridBagConstraints();
48 c.insets = new Insets(AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad, AltosUILib.tab_elt_pad);
51 label = new JLabel(label_text);
52 label.setFont(AltosUILib.label_font);
53 label.setHorizontalAlignment(SwingConstants.LEFT);
54 c.gridx = 0; c.gridy = y;
55 c.anchor = GridBagConstraints.WEST;
56 c.fill = GridBagConstraints.VERTICAL;
58 layout.setConstraints(label, c);
61 value = new JTextField[values.length];
62 for (int j = 0; j < values.length; j++) {
63 value[j] = new JTextField(values[j]);
64 value[j].setEditable(false);
65 value[j].setFont(AltosUILib.value_font);
66 value[j].setHorizontalAlignment(SwingConstants.RIGHT);
67 c.gridx = j+1; c.gridy = y;
68 c.anchor = GridBagConstraints.EAST;
69 c.fill = GridBagConstraints.BOTH;
71 layout.setConstraints(value[j], c);
74 flight_stats.add(this);
79 public void font_size_changed(int font_size) {
80 for (FlightStat f : flight_stats)
81 f.font_size_changed(font_size);
84 static String pos(double p, String pos, String neg) {
90 int deg = (int) Math.floor(p);
91 double min = (p - Math.floor(p)) * 60.0;
92 return String.format("%s %4d° %9.6f'", h, deg, min);
95 private FlightStat max_height_stat;
96 private FlightStat max_speed_stat;
97 private FlightStat max_accel_stat;
98 private FlightStat boost_accel_stat;
99 private FlightStat drogue_descent_stat;
100 private FlightStat main_descent_stat;
102 public void set_values(AltosFlightStats stats) {
103 if (max_height_stat != null && stats.max_height != AltosLib.MISSING) {
104 max_height_stat.set(String.format("%6.1f m", stats.max_height),
105 String.format("%5.0f ft", AltosConvert.meters_to_feet(stats.max_height)));
107 if (max_speed_stat != null && stats.max_speed != AltosLib.MISSING) {
108 max_speed_stat.set(String.format("%6.1f m/s", stats.max_speed),
109 String.format("%5.0f fps", AltosConvert.mps_to_fps(stats.max_speed)),
110 String.format("Mach %4.1f", AltosConvert.meters_to_mach(stats.max_speed)));
112 if (max_accel_stat != null && stats.max_acceleration != AltosLib.MISSING) {
113 max_accel_stat.set(String.format("%6.1f m/s²", stats.max_acceleration),
114 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.max_acceleration)),
115 String.format("%6.2f G", AltosConvert.meters_to_g(stats.max_acceleration)));
117 if (boost_accel_stat != null && stats.state_accel[AltosLib.ao_flight_boost] != AltosLib.MISSING) {
118 boost_accel_stat.set(String.format("%6.1f m/s²", stats.state_accel[AltosLib.ao_flight_boost]),
119 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.state_accel[AltosLib.ao_flight_boost])),
120 String.format("%6.2f G", AltosConvert.meters_to_g(stats.state_accel[AltosLib.ao_flight_boost])));
122 if (drogue_descent_stat != null && stats.state_speed[AltosLib.ao_flight_drogue] != AltosLib.MISSING) {
123 drogue_descent_stat.set(String.format("%6.1f m/s", -stats.state_speed[AltosLib.ao_flight_drogue]),
124 String.format("%5.0f ft/s", -AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_drogue])));
126 if (main_descent_stat != null && stats.state_speed[AltosLib.ao_flight_main] != AltosLib.MISSING) {
127 main_descent_stat.set(String.format("%6.1f m/s", -stats.state_speed[AltosLib.ao_flight_main]),
128 String.format("%5.0f ft/s", -AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_main])));
132 public void set_stats(AltosFlightStats stats) {
134 if (stats.serial != AltosLib.MISSING) {
135 if (stats.product != null && stats.firmware_version != null)
136 new FlightStat(layout, y++, "Device",
138 String.format("version %s", stats.firmware_version),
139 String.format("serial %d", stats.serial));
141 new FlightStat(layout, y++, "Serial", String.format("%d", stats.serial));
143 if (stats.flight != AltosLib.MISSING)
144 new FlightStat(layout, y++, "Flight", String.format("%d", stats.flight));
145 if (stats.year != AltosLib.MISSING && stats.hour != AltosLib.MISSING)
146 new FlightStat(layout, y++, "Date/Time",
147 String.format("%04d-%02d-%02d", stats.year, stats.month, stats.day),
148 String.format("%02d:%02d:%02d UTC", stats.hour, stats.minute, stats.second));
150 if (stats.year != AltosLib.MISSING)
151 new FlightStat(layout, y++, "Date",
152 String.format("%04d-%02d-%02d", stats.year, stats.month, stats.day));
153 if (stats.hour != AltosLib.MISSING)
154 new FlightStat(layout, y++, "Time",
155 String.format("%02d:%02d:%02d UTC", stats.hour, stats.minute, stats.second));
157 if (stats.max_height != AltosLib.MISSING) {
158 max_height_stat = new FlightStat(layout, y++, "Maximum height",
159 String.format("%6.1f m", stats.max_height),
160 String.format("%5.0f ft", AltosConvert.meters_to_feet(stats.max_height)));
162 if (stats.max_gps_height != AltosLib.MISSING) {
163 new FlightStat(layout, y++, "Maximum GPS height",
164 String.format("%6.1f m", stats.max_gps_height),
165 String.format("%5.0f ft", AltosConvert.meters_to_feet(stats.max_gps_height)));
167 if (stats.max_speed != AltosLib.MISSING) {
168 max_speed_stat = new FlightStat(layout, y++, "Maximum speed",
169 String.format("%6.1f m/s", stats.max_speed),
170 String.format("%5.0f fps", AltosConvert.mps_to_fps(stats.max_speed)),
171 String.format("Mach %4.1f", AltosConvert.meters_to_mach(stats.max_speed)));
173 if (stats.max_acceleration != AltosLib.MISSING)
174 max_accel_stat = new FlightStat(layout, y++, "Maximum boost acceleration",
175 String.format("%6.1f m/s²", stats.max_acceleration),
176 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.max_acceleration)),
177 String.format("%6.2f G", AltosConvert.meters_to_g(stats.max_acceleration)));
178 if (stats.state_accel[AltosLib.ao_flight_boost] != AltosLib.MISSING)
179 boost_accel_stat = new FlightStat(layout, y++, "Average boost acceleration",
180 String.format("%6.1f m/s²", stats.state_accel[AltosLib.ao_flight_boost]),
181 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.state_accel[AltosLib.ao_flight_boost])),
182 String.format("%6.2f G", AltosConvert.meters_to_g(stats.state_accel[AltosLib.ao_flight_boost])));
183 if (stats.state_time[AltosLib.ao_flight_boost] != 0 || stats.state_time[AltosLib.ao_flight_fast] != 0 || stats.state_time[AltosLib.ao_flight_coast] != 0) {
185 double boost_time = stats.state_time[AltosLib.ao_flight_boost];
186 double fast_time = stats.state_time[AltosLib.ao_flight_fast];
187 double coast_time = stats.state_time[AltosLib.ao_flight_coast];
190 new FlightStat(layout, y++, "Ascent time",
191 String.format("%6.1f s %s", boost_time,
192 AltosLib.state_name(AltosLib.ao_flight_boost)),
193 String.format("%6.1f s %s", fast_time,
194 AltosLib.state_name(AltosLib.ao_flight_fast)),
195 String.format("%6.1f s %s", coast_time,
196 AltosLib.state_name(AltosLib.ao_flight_coast)));
198 new FlightStat(layout, y++, "Ascent time",
199 String.format("%6.1f s %s", boost_time,
200 AltosLib.state_name(AltosLib.ao_flight_boost)),
201 String.format("%6.1f s %s", coast_time,
202 AltosLib.state_name(AltosLib.ao_flight_coast)));
205 if (stats.state_speed[AltosLib.ao_flight_drogue] != AltosLib.MISSING) {
208 if (stats.state_speed[AltosLib.ao_flight_main] == AltosLib.MISSING)
209 label = "Descent rate";
211 label = "Drogue descent rate";
212 drogue_descent_stat = new FlightStat(layout, y++, label,
213 String.format("%6.1f m/s", -stats.state_speed[AltosLib.ao_flight_drogue]),
214 String.format("%5.0f ft/s", -AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_drogue])));
216 if (stats.state_speed[AltosLib.ao_flight_main] != AltosLib.MISSING)
217 main_descent_stat = new FlightStat(layout, y++, "Main descent rate",
218 String.format("%6.1f m/s", -stats.state_speed[AltosLib.ao_flight_main]),
219 String.format("%5.0f ft/s", -AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_main])));
220 if (stats.state_time[AltosLib.ao_flight_drogue] != 0 || stats.state_time[AltosLib.ao_flight_main] != 0) {
221 double drogue_duration = stats.state_time[AltosLib.ao_flight_drogue];
222 double main_duration = stats.state_time[AltosLib.ao_flight_main];
223 double duration = drogue_duration + main_duration;
225 if (drogue_duration > 0 && main_duration > 0) {
226 new FlightStat(layout, y++, "Descent time",
227 String.format("%6.1f s %s", drogue_duration,
228 AltosLib.state_name(AltosLib.ao_flight_drogue)),
229 String.format("%6.1f s %s", main_duration,
230 AltosLib.state_name(AltosLib.ao_flight_main)));
231 } else if (duration > 0) {
232 new FlightStat(layout, y++, "Descent time",
233 String.format("%6.1f s", duration));
236 if (stats.landed_time > stats.boost_time)
237 new FlightStat(layout, y++, "Flight time",
238 String.format("%6.1f s", stats.landed_time - stats.boost_time));
239 if (stats.has_gps && stats.pad_lat != AltosLib.MISSING) {
240 new FlightStat(layout, y++, "Pad location",
241 pos(stats.pad_lat,"N","S"),
242 pos(stats.pad_lon,"E","W"));
244 if (stats.has_gps && stats.lat != AltosLib.MISSING) {
245 new FlightStat(layout, y++, "Last reported location",
246 pos(stats.lat,"N","S"),
247 pos(stats.lon,"E","W"));
251 public void tell_closing() {
252 AltosUIPreferences.unregister_font_listener(this);
255 public void filter_changed(AltosFlightStats stats) {
259 public AltosFlightStatsTable() {
260 layout = new GridBagLayout();
264 AltosUIPreferences.register_font_listener(this);
267 public AltosFlightStatsTable(AltosFlightStats stats) {