altosdroid: Skip updating hidden UI elements
[fw/altos] / altosdroid / src / org / altusmetrum / AltosDroid / AltosVoice.java
1 /*
2  * Copyright © 2011 Keith Packard <keithp@keithp.com>
3  * Copyright © 2012 Mike Beattie <mike@ethernal.org>
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; version 2 of the License.
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.AltosDroid;
20
21 import android.speech.tts.TextToSpeech;
22 import android.speech.tts.TextToSpeech.OnInitListener;
23
24 import org.altusmetrum.altoslib_5.*;
25
26 public class AltosVoice {
27
28         private TextToSpeech tts         = null;
29         private boolean      tts_enabled = false;
30
31         private IdleThread   idle_thread = null;
32
33         private AltosState   old_state   = null;
34
35         public AltosVoice(AltosDroid a) {
36
37                 tts = new TextToSpeech(a, new OnInitListener() {
38                         public void onInit(int status) {
39                                 if (status == TextToSpeech.SUCCESS) tts_enabled = true;
40                                 if (tts_enabled) {
41                                         idle_thread = new IdleThread();
42                                 }
43                         }
44                 });
45
46         }
47
48         public synchronized void speak(String s) {
49                 if (!tts_enabled) return;
50                 tts.speak(s, TextToSpeech.QUEUE_ADD, null);
51         }
52
53         public void stop() {
54                 if (tts != null) tts.shutdown();
55                 if (idle_thread != null) {
56                         idle_thread.interrupt();
57                         idle_thread = null;
58                 }
59         }
60
61         public void tell(AltosState state) {
62                 if (!tts_enabled) return;
63
64                 boolean spoke = false;
65                 if (old_state == null || old_state.state != state.state) {
66                         if (state.state != AltosLib.ao_flight_stateless)
67                                 speak(state.state_name());
68                         if ((old_state == null || old_state.state <= AltosLib.ao_flight_boost) &&
69                             state.state > AltosLib.ao_flight_boost) {
70                                 if (state.max_speed() != AltosLib.MISSING)
71                                         speak(String.format("max speed: %d meters per second.", (int) (state.max_speed() + 0.5)));
72                                 spoke = true;
73                         } else if ((old_state == null || old_state.state < AltosLib.ao_flight_drogue) &&
74                                    state.state >= AltosLib.ao_flight_drogue) {
75                                 if (state.max_height() != AltosLib.MISSING)
76                                         speak(String.format("max height: %d meters.", (int) (state.max_height() + 0.5)));
77                                 spoke = true;
78                         }
79                 }
80                 if (old_state == null || old_state.gps_ready != state.gps_ready) {
81                         if (state.gps_ready) {
82                                 speak("GPS ready");
83                                 spoke = true;
84                         } else if (old_state != null) {
85                                 speak("GPS lost");
86                                 spoke = true;
87                         }
88                 }
89                 old_state = state;
90                 if (idle_thread != null)
91                         idle_thread.notice(state, spoke);
92         }
93
94
95         class IdleThread extends Thread {
96                 boolean            started;
97                 private AltosState state;
98                 int                reported_landing;
99                 int                report_interval;
100                 long               report_time;
101
102                 public synchronized void report(boolean last) {
103                         if (state == null)
104                                 return;
105
106                         /* reset the landing count once we hear about a new flight */
107                         if (state.state < AltosLib.ao_flight_drogue)
108                                 reported_landing = 0;
109
110                         /* Shut up once the rocket is on the ground */
111                         if (reported_landing > 2) {
112                                 return;
113                         }
114
115                         /* If the rocket isn't on the pad, then report height */
116                         if (((AltosLib.ao_flight_drogue <= state.state &&
117                               state.state < AltosLib.ao_flight_landed) ||
118                              state.state == AltosLib.ao_flight_stateless) &&
119                             state.range >= 0)
120                         {
121                                 if (state.from_pad != null) {
122                                         speak(String.format("Height %d, bearing %s %d, elevation %d, range %d.\n",
123                                                             (int) (state.height() + 0.5),
124                                                             state.from_pad.bearing_words(
125                                                                     AltosGreatCircle.BEARING_VOICE),
126                                                             (int) (state.from_pad.bearing + 0.5),
127                                                             (int) (state.elevation + 0.5),
128                                                             (int) (state.range + 0.5)));
129                                 } else {
130                                         speak(String.format("Height %d, elevation %d, range %d.\n",
131                                                             (int) (state.height() + 0.5),
132                                                             (int) (state.elevation + 0.5),
133                                                             (int) (state.range + 0.5)));
134                                 }
135                         } else if (state.state > AltosLib.ao_flight_pad) {
136                                 if (state.height() != AltosLib.MISSING)
137                                         speak(String.format("%d meters", (int) (state.height() + 0.5)));
138                         } else {
139                                 reported_landing = 0;
140                         }
141
142                         /* If the rocket is coming down, check to see if it has landed;
143                          * either we've got a landed report or we haven't heard from it in
144                          * a long time
145                          */
146                         if (state.state >= AltosLib.ao_flight_drogue &&
147                             (last ||
148                              System.currentTimeMillis() - state.received_time >= 15000 ||
149                              state.state == AltosLib.ao_flight_landed))
150                         {
151                                 if (Math.abs(state.speed()) < 20 && state.height() < 100)
152                                         speak("rocket landed safely");
153                                 else
154                                         speak("rocket may have crashed");
155                                 if (state.from_pad != null)
156                                         speak(String.format("Bearing %d degrees, range %d meters.",
157                                                             (int) (state.from_pad.bearing + 0.5),
158                                                             (int) (state.from_pad.distance + 0.5)));
159                                 ++reported_landing;
160                         }
161                 }
162
163                 long now () {
164                         return System.currentTimeMillis();
165                 }
166
167                 void set_report_time() {
168                         report_time = now() + report_interval;
169                 }
170
171                 public void run () {
172                         try {
173                                 for (;;) {
174                                         set_report_time();
175                                         for (;;) {
176                                                 synchronized (this) {
177                                                         long sleep_time = report_time - now();
178                                                         if (sleep_time <= 0)
179                                                                 break;
180                                                         wait(sleep_time);
181                                                 }
182                                         }
183                                         report(false);
184                                 }
185                         } catch (InterruptedException ie) {
186                         }
187                 }
188
189                 public synchronized void notice(AltosState new_state, boolean spoken) {
190                         AltosState old_state = state;
191                         state = new_state;
192                         if (!started && state.state > AltosLib.ao_flight_pad) {
193                                 started = true;
194                                 start();
195                         }
196
197                         if (state.state < AltosLib.ao_flight_drogue)
198                                 report_interval = 10000;
199                         else
200                                 report_interval = 20000;
201                         if (old_state != null && old_state.state != state.state) {
202                                 report_time = now();
203                                 this.notify();
204                         } else if (spoken)
205                                 set_report_time();
206                 }
207
208                 public IdleThread() {
209                         state = null;
210                         reported_landing = 0;
211                         report_interval = 10000;
212                 }
213         }
214
215 }