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_11;
24 import org.altusmetrum.altoslib_11.*;
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 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);
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;
53 layout.setConstraints(label, c);
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;
66 layout.setConstraints(value[j], c);
69 flight_stats.add(this);
74 public void font_size_changed(int font_size) {
75 for (FlightStat f : flight_stats)
76 f.font_size_changed(font_size);
79 static String pos(double p, String pos, String neg) {
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);
90 public AltosFlightStatsTable(AltosFlightStats stats) {
91 layout = new GridBagLayout();
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));
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));
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)));
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)));
119 new FlightStat(layout, y++, "Maximum speed",
120 String.format("%5.0f m/s", stats.max_speed),
121 String.format("%5.0f fps", AltosConvert.mps_to_fps(stats.max_speed)),
122 String.format("Mach %4.1f", AltosConvert.meters_to_mach(stats.max_speed)));
123 if (stats.max_acceleration != AltosLib.MISSING)
124 new FlightStat(layout, y++, "Maximum boost acceleration",
125 String.format("%5.0f m/s²", stats.max_acceleration),
126 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.max_acceleration)),
127 String.format("%5.0f G", AltosConvert.meters_to_g(stats.max_acceleration)));
128 if (stats.state_accel[AltosLib.ao_flight_boost] != AltosLib.MISSING)
129 new FlightStat(layout, y++, "Average boost acceleration",
130 String.format("%5.0f m/s²", stats.state_accel[AltosLib.ao_flight_boost]),
131 String.format("%5.0f ft/s²", AltosConvert.meters_to_feet(stats.state_accel[AltosLib.ao_flight_boost])),
132 String.format("%5.0f G", AltosConvert.meters_to_g(stats.state_accel[AltosLib.ao_flight_boost])));
133 if (stats.state_speed[AltosLib.ao_flight_drogue] != AltosLib.MISSING)
134 new FlightStat(layout, y++, "Drogue descent rate",
135 String.format("%5.0f m/s", stats.state_speed[AltosLib.ao_flight_drogue]),
136 String.format("%5.0f ft/s", AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_drogue])));
137 if (stats.state_speed[AltosLib.ao_flight_main] != AltosLib.MISSING)
138 new FlightStat(layout, y++, "Main descent rate",
139 String.format("%5.0f m/s", stats.state_speed[AltosLib.ao_flight_main]),
140 String.format("%5.0f ft/s", AltosConvert.meters_to_feet(stats.state_speed[AltosLib.ao_flight_main])));
141 if (stats.state_start[AltosLib.ao_flight_boost] < stats.state_end[AltosLib.ao_flight_coast])
142 new FlightStat(layout, y++, "Ascent time",
143 String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_boost] - stats.state_start[AltosLib.ao_flight_boost],
144 AltosLib.state_name(AltosLib.ao_flight_boost)),
145 String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_fast] - stats.state_start[AltosLib.ao_flight_fast],
146 AltosLib.state_name(AltosLib.ao_flight_fast)),
147 String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_coast] - stats.state_start[AltosLib.ao_flight_coast],
148 AltosLib.state_name(AltosLib.ao_flight_coast)));
149 if (stats.state_start[AltosLib.ao_flight_drogue] < stats.state_end[AltosLib.ao_flight_main])
150 new FlightStat(layout, y++, "Descent time",
151 String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_drogue] - stats.state_start[AltosLib.ao_flight_drogue],
152 AltosLib.state_name(AltosLib.ao_flight_drogue)),
153 String.format("%6.1f s %s", stats.state_end[AltosLib.ao_flight_main] - stats.state_start[AltosLib.ao_flight_main],
154 AltosLib.state_name(AltosLib.ao_flight_main)));
155 if (stats.state_start[AltosLib.ao_flight_boost] < stats.state_end[AltosLib.ao_flight_main])
156 new FlightStat(layout, y++, "Flight time",
157 String.format("%6.1f s", stats.state_end[AltosLib.ao_flight_main] -
158 stats.state_start[AltosLib.ao_flight_boost]));
160 new FlightStat(layout, y++, "Pad location",
161 pos(stats.pad_lat,"N","S"),
162 pos(stats.pad_lon,"E","W"));
163 new FlightStat(layout, y++, "Last reported location",
164 pos(stats.lat,"N","S"),
165 pos(stats.lon,"E","W"));