altosui: Add configuration UI for beeper tone
[fw/altos] / altoslib / AltosConfigData.java
index b4478da93ffb4aa67455db882f9e0b5fc0dae99e..750faa7142638e8fdf680ad9eeb5c446c5dfac1d 100644 (file)
@@ -15,7 +15,7 @@
  * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
  */
 
-package org.altusmetrum.AltosLib;
+package org.altusmetrum.altoslib_3;
 
 import java.util.*;
 import java.text.*;
@@ -39,7 +39,7 @@ public class AltosConfigData implements Iterable<String> {
        public int      main_deploy;
        public int      apogee_delay;
        public int      apogee_lockout;
-       
+
        /* HAS_RADIO */
        public int      radio_frequency;
        public String   callsign;
@@ -67,6 +67,12 @@ public class AltosConfigData implements Iterable<String> {
        public int              npyro;
        public int              pyro;
 
+       /* HAS_APRS */
+       public int              aprs_interval;
+
+       /* HAS_BEEP */
+       public int              beep;
+
        /* Storage info replies */
        public int      storage_size;
        public int      storage_erase_unit;
@@ -74,7 +80,6 @@ public class AltosConfigData implements Iterable<String> {
        /* Log listing replies */
        public int      stored_flight;
 
-
        public static String get_string(String line, String label) throws  ParseException {
                if (line.startsWith(label)) {
                        String  quoted = line.substring(label.length()).trim();
@@ -114,15 +119,31 @@ public class AltosConfigData implements Iterable<String> {
                default:
                        if (flight_log_max <= 0)
                                return 1;
+                       int     log_max = flight_log_max * 1024;
                        int     log_space = storage_size - storage_erase_unit;
-                       int     log_used = stored_flight * flight_log_max;
+                       int     log_used;
+
+                       if (stored_flight <= 0)
+                               log_used = 0;
+                       else
+                               log_used = stored_flight * log_max;
+                       int     log_avail;
 
                        if (log_used >= log_space)
-                               return 0;
-                       return (log_space - log_used) / flight_log_max;
+                               log_avail = 0;
+                       else
+                               log_avail = (log_space - log_used) / log_max;
+
+                       return log_avail;
                }
        }
 
+       public boolean has_monitor_battery() {
+               if (product.startsWith("TeleBT"))
+                       return true;
+               return false;
+       }
+
        int[] parse_version(String v) {
                String[] parts = v.split("\\.");
                int r[] = new int[parts.length];
@@ -137,7 +158,7 @@ public class AltosConfigData implements Iterable<String> {
 
                return r;
        }
-       
+
        public int compare_version(String other) {
                int[]   me = parse_version(version);
                int[]   them = parse_version(other);
@@ -159,14 +180,46 @@ public class AltosConfigData implements Iterable<String> {
        public void reset() {
                lines = new LinkedList<String>();
 
-               serial = -1;
-               radio_setting = 0;
-               radio_frequency = 0;
-               pyros = null;
-               npyro = 0;
+               manufacturer = "unknown";
+               product = "unknown";
+               serial = 0;
+               flight = 0;
+               log_format = AltosLib.AO_LOG_FORMAT_UNKNOWN;
+               version = "unknown";
+
+               main_deploy = -1;
+               apogee_delay = -1;
+               apogee_lockout = -1;
+
+               radio_frequency = -1;
+               callsign = null;
+               radio_enable = -1;
+               radio_calibration = -1;
+               radio_channel = -1;
+               radio_setting = -1;
+
+               accel_cal_plus = -1;
+               accel_cal_minus = -1;
+               pad_orientation = -1;
+
+               flight_log_max = -1;
+               ignite_mode = -1;
+
+               aes_key = "";
+
                pyro = 0;
+               npyro = 0;
+               pyros = null;
+
+               aprs_interval = -1;
+
+               beep = -1;
+
+               storage_size = -1;
+               storage_erase_unit = -1;
+               stored_flight = 0;
        }
-       
+
        public void parse_line(String line) {
                lines.add(line);
                /* Version replies */
@@ -230,21 +283,27 @@ public class AltosConfigData implements Iterable<String> {
                if (npyro > 0) {
                        try {
                                AltosPyro p = new AltosPyro(pyro, line);
-                               if (pyro < npyro - 1)
+                               if (pyro < npyro)
                                        pyros[pyro++] = p;
                        } catch (Exception e) {}
                }
-               
+
+               /* HAS_APRS */
+               try { aprs_interval = get_int(line, "APRS interval:"); } catch (Exception e) {}
+
+               /* HAS_BEEP */
+               try { beep = get_int(line, "Beeper setting:"); System.out.printf ("beeper now %d\n", beep); } catch (Exception e) {}
+
                /* Storage info replies */
                try { storage_size = get_int(line, "Storage size:"); } catch (Exception e) {}
-               try { storage_erase_unit = get_int(line, "Storage erase unit"); } catch (Exception e) {}
+               try { storage_erase_unit = get_int(line, "Storage erase unit:"); } catch (Exception e) {}
 
                /* Log listing replies */
                try { get_int(line, "flight"); stored_flight++; }  catch (Exception e) {}
        }
 
        public AltosConfigData() {
-               this.reset();
+               reset();
        }
 
        private void read_link(AltosLink link, String finished) throws InterruptedException, TimeoutException {
@@ -262,18 +321,235 @@ public class AltosConfigData implements Iterable<String> {
                }
        }
 
+       public boolean has_frequency() {
+               return radio_frequency >= 0 || radio_setting >= 0 || radio_channel >= 0;
+       }
+
+       public void set_frequency(double freq) {
+               int     frequency = radio_frequency;
+               int     setting = radio_setting;
+
+               if (frequency > 0) {
+                       radio_frequency = (int) Math.floor (freq * 1000 + 0.5);
+                       radio_channel = -1;
+               } else if (setting > 0) {
+                       radio_setting =AltosConvert.radio_frequency_to_setting(freq,
+                                                                                   radio_calibration);
+                       radio_channel = -1;
+               } else {
+                       radio_channel = AltosConvert.radio_frequency_to_channel(freq);
+               }
+       }
+
+       public double frequency() {
+               int     channel = radio_channel;
+               int     setting = radio_setting;
+
+               if (radio_frequency < 0 && channel < 0 && setting < 0)
+                       return -1;
+
+               if (channel < 0)
+                       channel = 0;
+               if (setting < 0)
+                       setting = 0;
+
+               return AltosConvert.radio_to_frequency(radio_frequency,
+                                                      setting,
+                                                      radio_calibration,
+                                                      channel);
+       }
+
+       boolean use_flash_for_config() {
+               if (product.startsWith("TeleMega"))
+                       return false;
+               if (product.startsWith("TeleMetrum-v2"))
+                       return false;
+               return true;
+       }
+
+
+       public int log_limit() {
+               if (storage_size > 0) {
+                       int     log_limit = storage_size;
+
+                       if (storage_erase_unit > 0 && use_flash_for_config())
+                               log_limit -= storage_erase_unit;
+
+                       if (log_limit > 0)
+                               return log_limit / 1024;
+               }
+               return 1024;
+       }
+
+       public void get_values(AltosConfigValues source) {
+
+               /* HAS_FLIGHT */
+               if (main_deploy >= 0)
+                       main_deploy = source.main_deploy();
+               if (apogee_delay >= 0)
+                       apogee_delay = source.apogee_delay();
+               if (apogee_lockout >= 0)
+                       apogee_lockout = source.apogee_lockout();
+
+               /* HAS_RADIO */
+               if (has_frequency())
+                       set_frequency(source.radio_frequency());
+               if (radio_enable >= 0)
+                       radio_enable = source.radio_enable();
+               if (callsign != null)
+                       callsign = source.callsign();
+               if (radio_calibration >= 0)
+                       radio_calibration = source.radio_calibration();
+
+               /* HAS_ACCEL */
+               if (pad_orientation >= 0)
+                       pad_orientation = source.pad_orientation();
+
+               /* HAS_LOG */
+               if (flight_log_max >= 0)
+                       flight_log_max = source.flight_log_max();
+
+               /* HAS_IGNITE */
+               if (ignite_mode >= 0)
+                       ignite_mode = source.ignite_mode();
+
+               /* AO_PYRO_NUM */
+               if (npyro > 0)
+                       pyros = source.pyros();
+
+               /* HAS_APRS */
+               if (aprs_interval >= 0)
+                       aprs_interval = source.aprs_interval();
+
+               /* HAS_BEEP */
+               if (beep >= 0)
+                       beep = source.beep();
+       }
+
+       public void set_values(AltosConfigValues dest) {
+               dest.set_serial(serial);
+               dest.set_product(product);
+               dest.set_version(version);
+               dest.set_main_deploy(main_deploy);
+               dest.set_apogee_delay(apogee_delay);
+               dest.set_apogee_lockout(apogee_lockout);
+               dest.set_radio_calibration(radio_calibration);
+               dest.set_radio_frequency(frequency());
+               boolean max_enabled = true;
+
+               if (log_limit() == 0)
+                       max_enabled = false;
+
+               switch (log_format) {
+               case AltosLib.AO_LOG_FORMAT_TINY:
+                       max_enabled = false;
+                       break;
+               default:
+                       if (stored_flight > 0)
+                               max_enabled = false;
+                       break;
+               }
+
+               dest.set_flight_log_max_enabled(max_enabled);
+               dest.set_radio_enable(radio_enable);
+               dest.set_flight_log_max_limit(log_limit());
+               dest.set_flight_log_max(flight_log_max);
+               dest.set_ignite_mode(ignite_mode);
+               dest.set_pad_orientation(pad_orientation);
+               dest.set_callsign(callsign);
+               if (npyro > 0)
+                       dest.set_pyros(pyros);
+               else
+                       dest.set_pyros(null);
+               dest.set_aprs_interval(aprs_interval);
+               dest.set_beep(beep);
+       }
+
+       public void save(AltosLink link, boolean remote) throws InterruptedException, TimeoutException {
+
+               /* HAS_FLIGHT */
+               if (main_deploy >= 0)
+                       link.printf("c m %d\n", main_deploy);
+               if (apogee_delay >= 0)
+                       link.printf("c d %d\n", apogee_delay);
+               if (apogee_lockout >= 0)
+                       link.printf("c L %d\n", apogee_lockout);
+
+               /* Don't mess with radio calibration when remote */
+               if (radio_calibration > 0 && !remote)
+                       link.printf("c f %d\n", radio_calibration);
+
+               /* HAS_RADIO */
+               if (has_frequency()) {
+                       boolean has_frequency = radio_frequency >= 0;
+                       boolean has_setting = radio_setting > 0;
+                       double frequency = frequency();
+                       link.set_radio_frequency(frequency,
+                                                       has_frequency,
+                                                       has_setting,
+                                                       radio_calibration);
+                       /* When remote, reset the dongle frequency at the same time */
+                       if (remote) {
+                               link.stop_remote();
+                               link.set_radio_frequency(frequency);
+                               link.start_remote();
+                       }
+               }
+
+               if (callsign != null)
+                       link.printf("c c %s\n", callsign);
+               if (radio_enable >= 0)
+                       link.printf("c e %d\n", radio_enable);
+
+               /* HAS_ACCEL */
+               /* UI doesn't support accel cal */
+               if (pad_orientation >= 0)
+                       link.printf("c o %d\n", pad_orientation);
+
+               /* HAS_LOG */
+               if (flight_log_max != 0)
+                       link.printf("c l %d\n", flight_log_max);
+
+               /* HAS_IGNITE */
+               if (ignite_mode >= 0)
+                       link.printf("c i %d\n", ignite_mode);
+
+               /* HAS_AES */
+               /* UI doesn't support AES key config */
+
+               /* AO_PYRO_NUM */
+               if (npyro > 0) {
+                       for (int p = 0; p < pyros.length; p++) {
+                               link.printf("c P %s\n",
+                                                  pyros[p].toString());
+                       }
+               }
+
+               /* HAS_APRS */
+               if (aprs_interval >= 0)
+                       link.printf("c A %d\n", aprs_interval);
+
+               /* HAS_BEEP */
+               if (beep >= 0)
+                       link.printf("c b %d\n", beep);
+
+               link.printf("c w\n");
+               link.flush_output();
+       }
+
        public AltosConfigData(AltosLink link) throws InterruptedException, TimeoutException {
-               this.reset();
+               reset();
                link.printf("c s\nf\nv\n");
                read_link(link, "software-version");
                switch (log_format) {
-               case AltosLib.AO_LOG_FORMAT_TELEMETRY:
-               case AltosLib.AO_LOG_FORMAT_TELESCIENCE:
-                       break;
-               default:
+               case AltosLib.AO_LOG_FORMAT_FULL:
+               case AltosLib.AO_LOG_FORMAT_TINY:
+               case AltosLib.AO_LOG_FORMAT_TELEMEGA:
                        link.printf("l\n");
                        read_link(link, "done");
+               default:
+                       break;
                }
        }
 
-}
\ No newline at end of file
+}