blob: 89a46419819af794fdf71e9a13aa78ec9e7690bb [file] [log] [blame]
Jean-Michel Trivied29a652009-06-05 18:37:29 -07001/*
2 * Copyright (C) 2009 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.settings;
18
19import static android.provider.Settings.Secure.TTS_USE_DEFAULTS;
20import static android.provider.Settings.Secure.TTS_DEFAULT_RATE;
Jean-Michel Trivi80368622009-06-09 19:29:27 -070021import static android.provider.Settings.Secure.TTS_DEFAULT_LANG;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070022import static android.provider.Settings.Secure.TTS_DEFAULT_COUNTRY;
23import static android.provider.Settings.Secure.TTS_DEFAULT_VARIANT;
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -070024import static android.provider.Settings.Secure.TTS_DEFAULT_SYNTH;
Charles Chenf47cce02010-03-17 17:33:23 -070025import static android.provider.Settings.Secure.TTS_ENABLED_PLUGINS;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070026
Charles Chen0a0eb5f2010-03-16 20:09:17 -070027import android.app.AlertDialog;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070028import android.content.ContentResolver;
Charles Chen0a0eb5f2010-03-16 20:09:17 -070029import android.content.DialogInterface;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070030import android.content.Intent;
31import android.content.pm.ActivityInfo;
32import android.content.pm.PackageManager;
33import android.content.pm.ResolveInfo;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070034import android.os.Bundle;
35import android.preference.ListPreference;
36import android.preference.Preference;
Charles Chen0a0eb5f2010-03-16 20:09:17 -070037import android.preference.Preference.OnPreferenceClickListener;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070038import android.preference.PreferenceActivity;
Charles Chen0a0eb5f2010-03-16 20:09:17 -070039import android.preference.PreferenceGroup;
40import android.preference.PreferenceScreen;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070041import android.preference.CheckBoxPreference;
42import android.provider.Settings;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070043import android.provider.Settings.SettingNotFoundException;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070044import android.speech.tts.TextToSpeech;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070045import android.util.Log;
46
Charles Chenc8298712010-02-10 13:58:23 -080047import java.util.ArrayList;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070048import java.util.List;
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -070049import java.util.Locale;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070050import java.util.StringTokenizer;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070051
Jean-Michel Trivied29a652009-06-05 18:37:29 -070052public class TextToSpeechSettings extends PreferenceActivity implements
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070053 Preference.OnPreferenceChangeListener, Preference.OnPreferenceClickListener,
54 TextToSpeech.OnInitListener {
Jean-Michel Trivied29a652009-06-05 18:37:29 -070055
56 private static final String TAG = "TextToSpeechSettings";
Jean-Michel Trivied29a652009-06-05 18:37:29 -070057
Charles Chen0a0eb5f2010-03-16 20:09:17 -070058 private static final String SYSTEM_TTS = "com.svox.pico";
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070059 private static final String KEY_TTS_PLAY_EXAMPLE = "tts_play_example";
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070060 private static final String KEY_TTS_INSTALL_DATA = "tts_install_data";
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070061 private static final String KEY_TTS_USE_DEFAULT = "toggle_use_default_tts_settings";
Jean-Michel Trivied29a652009-06-05 18:37:29 -070062 private static final String KEY_TTS_DEFAULT_RATE = "tts_default_rate";
Jean-Michel Trivi80368622009-06-09 19:29:27 -070063 private static final String KEY_TTS_DEFAULT_LANG = "tts_default_lang";
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070064 private static final String KEY_TTS_DEFAULT_COUNTRY = "tts_default_country";
65 private static final String KEY_TTS_DEFAULT_VARIANT = "tts_default_variant";
Charles Chen5dbc74a2009-12-07 12:08:13 -080066 private static final String KEY_TTS_DEFAULT_SYNTH = "tts_default_synth";
Charles Chen0a0eb5f2010-03-16 20:09:17 -070067
68 private static final String KEY_PLUGIN_ENABLED_PREFIX = "ENABLED_";
69 private static final String KEY_PLUGIN_SETTINGS_PREFIX = "SETTINGS_";
70
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -070071 // TODO move default Locale values to TextToSpeech.Engine
72 private static final String DEFAULT_LANG_VAL = "eng";
73 private static final String DEFAULT_COUNTRY_VAL = "USA";
74 private static final String DEFAULT_VARIANT_VAL = "";
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070075
76 private static final String LOCALE_DELIMITER = "-";
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070077
Jean-Michel Trivi628431d2009-07-17 16:52:54 -070078 private static final String FALLBACK_TTS_DEFAULT_SYNTH =
Jean-Michel Trivi387dc0c2009-07-28 15:13:35 -070079 TextToSpeech.Engine.DEFAULT_SYNTH;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070080
81 private Preference mPlayExample = null;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070082 private Preference mInstallData = null;
Jean-Michel Trivied29a652009-06-05 18:37:29 -070083 private CheckBoxPreference mUseDefaultPref = null;
84 private ListPreference mDefaultRatePref = null;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070085 private ListPreference mDefaultLocPref = null;
Charles Chen5dbc74a2009-12-07 12:08:13 -080086 private ListPreference mDefaultSynthPref = null;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -070087 private String mDefaultLanguage = null;
88 private String mDefaultCountry = null;
89 private String mDefaultLocVariant = null;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070090 private String mDefaultEng = "";
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -070091 private int mDefaultRate = TextToSpeech.Engine.DEFAULT_RATE;
92
93 // Array of strings used to demonstrate TTS in the different languages.
94 private String[] mDemoStrings;
95 // Index of the current string to use for the demo.
96 private int mDemoStringIndex = 0;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -070097
98 private boolean mEnableDemo = false;
Charles Chenc8298712010-02-10 13:58:23 -080099 private boolean mVoicesMissing = false;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700100
101 private TextToSpeech mTts = null;
Charles Chencf31e652010-04-07 14:26:31 -0700102 private boolean mTtsStarted = false;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700103
104 /**
105 * Request code (arbitrary value) for voice data check through
106 * startActivityForResult.
107 */
108 private static final int VOICE_DATA_INTEGRITY_CHECK = 1977;
Charles Chen4df6c792010-01-22 11:17:40 -0800109 private static final int GET_SAMPLE_TEXT = 1983;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700110
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700111 @Override
112 protected void onCreate(Bundle savedInstanceState) {
113 super.onCreate(savedInstanceState);
114
115 addPreferencesFromResource(R.xml.tts_settings);
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700116
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700117 addEngineSpecificSettings();
118
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700119 mDemoStrings = getResources().getStringArray(R.array.tts_demo_strings);
120
Jean-Michel Trivi6dde8962009-08-11 09:06:58 -0700121 setVolumeControlStream(TextToSpeech.Engine.DEFAULT_STREAM);
122
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700123 mEnableDemo = false;
Charles Chencf31e652010-04-07 14:26:31 -0700124 mTtsStarted = false;
Charles Chenbe6e8272010-03-22 16:06:05 -0700125
Charles Chen8c8185b2010-04-08 16:51:35 -0700126 Locale currentLocale = Locale.getDefault();
127 mDefaultLanguage = currentLocale.getISO3Language();
128 mDefaultCountry = currentLocale.getISO3Country();
129 mDefaultLocVariant = currentLocale.getVariant();
130
Charles Chenbe6e8272010-03-22 16:06:05 -0700131 mTts = new TextToSpeech(this, this);
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700132 }
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700133
134
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700135 @Override
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700136 protected void onStart() {
137 super.onStart();
Charles Chencf31e652010-04-07 14:26:31 -0700138 if (mTtsStarted){
139 // whenever we return to this screen, we don't know the state of the
140 // system, so we have to recheck that we can play the demo, or it must be disabled.
141 // TODO make the TTS service listen to "changes in the system", i.e. sd card un/mount
142 initClickers();
143 updateWidgetState();
144 checkVoiceData();
145 }
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700146 }
147
148
149 @Override
150 protected void onDestroy() {
151 super.onDestroy();
152 if (mTts != null) {
153 mTts.shutdown();
154 }
155 }
156
Charles Chen8c8185b2010-04-08 16:51:35 -0700157 @Override
158 protected void onPause() {
159 super.onPause();
160 if ((mDefaultRatePref != null) && (mDefaultRatePref.getDialog() != null)) {
161 mDefaultRatePref.getDialog().dismiss();
162 }
163 if ((mDefaultLocPref != null) && (mDefaultLocPref.getDialog() != null)) {
164 mDefaultLocPref.getDialog().dismiss();
165 }
166 if ((mDefaultSynthPref != null) && (mDefaultSynthPref.getDialog() != null)) {
167 mDefaultSynthPref.getDialog().dismiss();
168 }
169 }
170
171
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700172
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700173 private void addEngineSpecificSettings() {
174 PreferenceGroup enginesCategory = (PreferenceGroup) findPreference("tts_engines_section");
175 Intent intent = new Intent("android.intent.action.START_TTS_ENGINE");
176 ResolveInfo[] enginesArray = new ResolveInfo[0];
177 PackageManager pm = getPackageManager();
178 enginesArray = pm.queryIntentActivities(intent, 0).toArray(enginesArray);
179 for (int i = 0; i < enginesArray.length; i++) {
180 String prefKey = "";
181 final String pluginPackageName = enginesArray[i].activityInfo.packageName;
182 if (!enginesArray[i].activityInfo.packageName.equals(SYSTEM_TTS)) {
183 CheckBoxPreference chkbxPref = new CheckBoxPreference(this);
184 prefKey = KEY_PLUGIN_ENABLED_PREFIX + pluginPackageName;
185 chkbxPref.setKey(prefKey);
186 chkbxPref.setTitle(enginesArray[i].loadLabel(pm));
187 enginesCategory.addPreference(chkbxPref);
188 }
189 if (pluginHasSettings(pluginPackageName)) {
190 Preference pref = new Preference(this);
191 prefKey = KEY_PLUGIN_SETTINGS_PREFIX + pluginPackageName;
192 pref.setKey(prefKey);
193 pref.setTitle(enginesArray[i].loadLabel(pm));
194 CharSequence settingsLabel = getResources().getString(
195 R.string.tts_engine_name_settings, enginesArray[i].loadLabel(pm));
196 pref.setSummary(settingsLabel);
197 pref.setOnPreferenceClickListener(new OnPreferenceClickListener(){
198 public boolean onPreferenceClick(Preference preference){
199 Intent i = new Intent();
200 i.setClassName(pluginPackageName,
201 pluginPackageName + ".EngineSettings");
202 startActivity(i);
203 return true;
204 }
205 });
206 enginesCategory.addPreference(pref);
207 }
208 }
209 }
210
211 private boolean pluginHasSettings(String pluginPackageName) {
212 PackageManager pm = getPackageManager();
213 Intent i = new Intent();
214 i.setClassName(pluginPackageName, pluginPackageName + ".EngineSettings");
215 if (pm.resolveActivity(i, PackageManager.MATCH_DEFAULT_ONLY) != null){
216 return true;
217 }
218 return false;
219 }
220
221
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700222 private void initClickers() {
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700223 mPlayExample = findPreference(KEY_TTS_PLAY_EXAMPLE);
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700224 mPlayExample.setOnPreferenceClickListener(this);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700225
226 mInstallData = findPreference(KEY_TTS_INSTALL_DATA);
227 mInstallData.setOnPreferenceClickListener(this);
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700228 }
229
230
231 private void initDefaultSettings() {
232 ContentResolver resolver = getContentResolver();
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700233
234 // Find the default TTS values in the settings, initialize and store the
235 // settings if they are not found.
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700236
237 // "Use Defaults"
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700238 int useDefault = 0;
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700239 mUseDefaultPref = (CheckBoxPreference) findPreference(KEY_TTS_USE_DEFAULT);
240 try {
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700241 useDefault = Settings.Secure.getInt(resolver, TTS_USE_DEFAULTS);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700242 } catch (SettingNotFoundException e) {
243 // "use default" setting not found, initialize it
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700244 useDefault = TextToSpeech.Engine.USE_DEFAULTS;
245 Settings.Secure.putInt(resolver, TTS_USE_DEFAULTS, useDefault);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700246 }
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700247 mUseDefaultPref.setChecked(useDefault == 1);
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700248 mUseDefaultPref.setOnPreferenceChangeListener(this);
249
Charles Chen5dbc74a2009-12-07 12:08:13 -0800250 // Default synthesis engine
251 mDefaultSynthPref = (ListPreference) findPreference(KEY_TTS_DEFAULT_SYNTH);
252 loadEngines();
253 mDefaultSynthPref.setOnPreferenceChangeListener(this);
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700254 String engine = Settings.Secure.getString(resolver, TTS_DEFAULT_SYNTH);
255 if (engine == null) {
256 // TODO move FALLBACK_TTS_DEFAULT_SYNTH to TextToSpeech
257 engine = FALLBACK_TTS_DEFAULT_SYNTH;
258 Settings.Secure.putString(resolver, TTS_DEFAULT_SYNTH, engine);
259 }
260 mDefaultEng = engine;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700261
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700262 // Default rate
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700263 mDefaultRatePref = (ListPreference) findPreference(KEY_TTS_DEFAULT_RATE);
264 try {
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700265 mDefaultRate = Settings.Secure.getInt(resolver, TTS_DEFAULT_RATE);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700266 } catch (SettingNotFoundException e) {
267 // default rate setting not found, initialize it
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700268 mDefaultRate = TextToSpeech.Engine.DEFAULT_RATE;
269 Settings.Secure.putInt(resolver, TTS_DEFAULT_RATE, mDefaultRate);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700270 }
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700271 mDefaultRatePref.setValue(String.valueOf(mDefaultRate));
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700272 mDefaultRatePref.setOnPreferenceChangeListener(this);
273
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700274 // Default language / country / variant : these three values map to a single ListPref
275 // representing the matching Locale
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700276 mDefaultLocPref = (ListPreference) findPreference(KEY_TTS_DEFAULT_LANG);
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700277 initDefaultLang();
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700278 mDefaultLocPref.setOnPreferenceChangeListener(this);
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700279 }
280
281
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700282 /**
283 * Ask the current default engine to launch the matching CHECK_TTS_DATA activity
284 * to check the required TTS files are properly installed.
285 */
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700286 private void checkVoiceData() {
287 PackageManager pm = getPackageManager();
288 Intent intent = new Intent();
Jean-Michel Trivi387dc0c2009-07-28 15:13:35 -0700289 intent.setAction(TextToSpeech.Engine.ACTION_CHECK_TTS_DATA);
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700290 List<ResolveInfo> resolveInfos = pm.queryIntentActivities(intent, 0);
291 // query only the package that matches that of the default engine
292 for (int i = 0; i < resolveInfos.size(); i++) {
293 ActivityInfo currentActivityInfo = resolveInfos.get(i).activityInfo;
294 if (mDefaultEng.equals(currentActivityInfo.packageName)) {
295 intent.setClassName(mDefaultEng, currentActivityInfo.name);
296 this.startActivityForResult(intent, VOICE_DATA_INTEGRITY_CHECK);
297 }
298 }
299 }
300
301
302 /**
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700303 * Ask the current default engine to launch the matching INSTALL_TTS_DATA activity
304 * so the required TTS files are properly installed.
305 */
306 private void installVoiceData() {
307 PackageManager pm = getPackageManager();
308 Intent intent = new Intent();
Jean-Michel Trivi387dc0c2009-07-28 15:13:35 -0700309 intent.setAction(TextToSpeech.Engine.ACTION_INSTALL_TTS_DATA);
Jean-Michel Trivi58ea43a2009-09-09 15:13:38 -0700310 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700311 List<ResolveInfo> resolveInfos = pm.queryIntentActivities(intent, 0);
312 // query only the package that matches that of the default engine
313 for (int i = 0; i < resolveInfos.size(); i++) {
314 ActivityInfo currentActivityInfo = resolveInfos.get(i).activityInfo;
315 if (mDefaultEng.equals(currentActivityInfo.packageName)) {
316 intent.setClassName(mDefaultEng, currentActivityInfo.name);
Jean-Michel Trivi58ea43a2009-09-09 15:13:38 -0700317 this.startActivity(intent);
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700318 }
319 }
320 }
321
Charles Chen4df6c792010-01-22 11:17:40 -0800322 /**
323 * Ask the current default engine to return a string of sample text to be
324 * spoken to the user.
325 */
326 private void getSampleText() {
327 PackageManager pm = getPackageManager();
328 Intent intent = new Intent();
329 // TODO (clchen): Replace Intent string with the actual
330 // Intent defined in the list of platform Intents.
331 intent.setAction("android.speech.tts.engine.GET_SAMPLE_TEXT");
332 intent.putExtra("language", mDefaultLanguage);
333 intent.putExtra("country", mDefaultCountry);
334 intent.putExtra("variant", mDefaultLocVariant);
335 List<ResolveInfo> resolveInfos = pm.queryIntentActivities(intent, 0);
336 // query only the package that matches that of the default engine
337 for (int i = 0; i < resolveInfos.size(); i++) {
338 ActivityInfo currentActivityInfo = resolveInfos.get(i).activityInfo;
339 if (mDefaultEng.equals(currentActivityInfo.packageName)) {
340 intent.setClassName(mDefaultEng, currentActivityInfo.name);
341 this.startActivityForResult(intent, GET_SAMPLE_TEXT);
342 }
343 }
344 }
345
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700346
347 /**
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700348 * Called when the TTS engine is initialized.
349 */
350 public void onInit(int status) {
Jean-Michel Trivi387dc0c2009-07-28 15:13:35 -0700351 if (status == TextToSpeech.SUCCESS) {
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700352 mEnableDemo = true;
Jean-Michel Trivi7330a882010-02-17 12:50:44 -0800353 if (mDefaultLanguage == null) {
Charles Chencf3998b2010-02-11 18:13:42 -0800354 mDefaultLanguage = Locale.getDefault().getISO3Language();
355 }
Jean-Michel Trivi7330a882010-02-17 12:50:44 -0800356 if (mDefaultCountry == null) {
357 mDefaultCountry = Locale.getDefault().getISO3Country();
358 }
359 if (mDefaultLocVariant == null) {
360 mDefaultLocVariant = new String();
361 }
Charles Chencf3998b2010-02-11 18:13:42 -0800362 mTts.setLanguage(new Locale(mDefaultLanguage, mDefaultCountry, mDefaultLocVariant));
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700363 mTts.setSpeechRate((float)(mDefaultRate/100.0f));
Charles Chencf31e652010-04-07 14:26:31 -0700364 initDefaultSettings();
365 initClickers();
366 updateWidgetState();
367 checkVoiceData();
368 mTtsStarted = true;
369 Log.v(TAG, "TTS engine for settings screen initialized.");
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700370 } else {
371 Log.v(TAG, "TTS engine for settings screen failed to initialize successfully.");
372 mEnableDemo = false;
373 }
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700374 updateWidgetState();
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700375 }
376
377
378 /**
379 * Called when voice data integrity check returns
380 */
381 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
382 if (requestCode == VOICE_DATA_INTEGRITY_CHECK) {
Charles Chend5f013a2010-02-18 10:11:25 -0800383 if (data == null){
384 // The CHECK_TTS_DATA activity for the plugin did not run properly;
385 // disable the preview and install controls and return.
386 mEnableDemo = false;
387 mVoicesMissing = false;
388 updateWidgetState();
389 return;
390 }
Charles Chenc8298712010-02-10 13:58:23 -0800391 ArrayList<String> available =
Charles Chen510bc4a2010-04-07 17:41:07 -0700392 data.getStringArrayListExtra(TextToSpeech.Engine.EXTRA_AVAILABLE_VOICES);
Charles Chenc8298712010-02-10 13:58:23 -0800393 ArrayList<String> unavailable =
Charles Chen510bc4a2010-04-07 17:41:07 -0700394 data.getStringArrayListExtra(TextToSpeech.Engine.EXTRA_UNAVAILABLE_VOICES);
Charles Chend5f013a2010-02-18 10:11:25 -0800395 if ((available == null) || (unavailable == null)){
396 // The CHECK_TTS_DATA activity for the plugin did not run properly;
397 // disable the preview and install controls and return.
398 mEnableDemo = false;
399 mVoicesMissing = false;
400 updateWidgetState();
401 return;
402 }
Charles Chenc8298712010-02-10 13:58:23 -0800403 if (available.size() > 0){
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700404 if (mTts == null) {
405 mTts = new TextToSpeech(this, this);
406 }
Charles Chenc8298712010-02-10 13:58:23 -0800407 ListPreference ttsLanguagePref =
408 (ListPreference) findPreference("tts_default_lang");
409 CharSequence[] entries = new CharSequence[available.size()];
410 CharSequence[] entryValues = new CharSequence[available.size()];
Jean-Baptiste Queru6e61b212010-04-14 10:40:48 -0700411 int selectedLanguageIndex = -1;
412 String selectedLanguagePref = mDefaultLanguage;
413 if (mDefaultCountry.length() > 0) {
414 selectedLanguagePref = selectedLanguagePref + LOCALE_DELIMITER +
415 mDefaultCountry;
416 }
417 if (mDefaultLocVariant.length() > 0) {
418 selectedLanguagePref = selectedLanguagePref + LOCALE_DELIMITER +
419 mDefaultLocVariant;
420 }
421 for (int i = 0; i < available.size(); i++) {
Charles Chenc8298712010-02-10 13:58:23 -0800422 String[] langCountryVariant = available.get(i).split("-");
423 Locale loc = null;
424 if (langCountryVariant.length == 1){
425 loc = new Locale(langCountryVariant[0]);
426 } else if (langCountryVariant.length == 2){
427 loc = new Locale(langCountryVariant[0], langCountryVariant[1]);
428 } else if (langCountryVariant.length == 3){
429 loc = new Locale(langCountryVariant[0], langCountryVariant[1],
430 langCountryVariant[2]);
431 }
432 if (loc != null){
433 entries[i] = loc.getDisplayName();
434 entryValues[i] = available.get(i);
Jean-Baptiste Queru6e61b212010-04-14 10:40:48 -0700435 if (entryValues[i].equals(selectedLanguagePref)) {
436 selectedLanguageIndex = i;
437 }
Charles Chenc8298712010-02-10 13:58:23 -0800438 }
439 }
440 ttsLanguagePref.setEntries(entries);
441 ttsLanguagePref.setEntryValues(entryValues);
Jean-Baptiste Queru6e61b212010-04-14 10:40:48 -0700442 if (selectedLanguageIndex > -1) {
443 ttsLanguagePref.setValueIndex(selectedLanguageIndex);
444 }
Charles Chenc8298712010-02-10 13:58:23 -0800445 mEnableDemo = true;
Charles Chen473111b2010-04-06 15:50:42 -0700446 // Make sure that the default language can be used.
447 int languageResult = mTts.setLanguage(
448 new Locale(mDefaultLanguage, mDefaultCountry, mDefaultLocVariant));
449 if (languageResult < TextToSpeech.LANG_AVAILABLE){
450 Locale currentLocale = Locale.getDefault();
451 mDefaultLanguage = currentLocale.getISO3Language();
452 mDefaultCountry = currentLocale.getISO3Country();
453 mDefaultLocVariant = currentLocale.getVariant();
454 languageResult = mTts.setLanguage(
455 new Locale(mDefaultLanguage, mDefaultCountry, mDefaultLocVariant));
456 // If the default Locale isn't supported, just choose the first available
457 // language so that there is at least something.
458 if (languageResult < TextToSpeech.LANG_AVAILABLE){
459 parseLocaleInfo(ttsLanguagePref.getEntryValues()[0].toString());
460 mTts.setLanguage(
461 new Locale(mDefaultLanguage, mDefaultCountry, mDefaultLocVariant));
462 }
463 ContentResolver resolver = getContentResolver();
464 Settings.Secure.putString(resolver, TTS_DEFAULT_LANG, mDefaultLanguage);
465 Settings.Secure.putString(resolver, TTS_DEFAULT_COUNTRY, mDefaultCountry);
466 Settings.Secure.putString(resolver, TTS_DEFAULT_VARIANT, mDefaultLocVariant);
467 }
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700468 } else {
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700469 mEnableDemo = false;
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700470 }
Charles Chenc8298712010-02-10 13:58:23 -0800471
472 if (unavailable.size() > 0){
473 mVoicesMissing = true;
474 } else {
475 mVoicesMissing = false;
476 }
477
478 updateWidgetState();
Charles Chen4df6c792010-01-22 11:17:40 -0800479 } else if (requestCode == GET_SAMPLE_TEXT) {
480 if (resultCode == TextToSpeech.LANG_AVAILABLE) {
Charles Chenec05e712010-02-18 15:06:26 -0800481 String sample = getString(R.string.tts_demo);
482 if ((data != null) && (data.getStringExtra("sampleText") != null)) {
483 sample = data.getStringExtra("sampleText");
Charles Chend5f013a2010-02-18 10:11:25 -0800484 }
Charles Chen4df6c792010-01-22 11:17:40 -0800485 if (mTts != null) {
Charles Chen4df6c792010-01-22 11:17:40 -0800486 mTts.speak(sample, TextToSpeech.QUEUE_FLUSH, null);
487 }
488 } else {
489 // TODO: Display an error here to the user.
490 Log.e(TAG, "Did not have a sample string for the requested language");
491 }
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700492 }
493 }
494
495
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700496 public boolean onPreferenceChange(Preference preference, Object objValue) {
497 if (KEY_TTS_USE_DEFAULT.equals(preference.getKey())) {
498 // "Use Defaults"
499 int value = (Boolean)objValue ? 1 : 0;
Jean-Michel Trivi80368622009-06-09 19:29:27 -0700500 Settings.Secure.putInt(getContentResolver(), TTS_USE_DEFAULTS,
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700501 value);
502 Log.i(TAG, "TTS use default settings is "+objValue.toString());
503 } else if (KEY_TTS_DEFAULT_RATE.equals(preference.getKey())) {
504 // Default rate
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700505 mDefaultRate = Integer.parseInt((String) objValue);
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700506 try {
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700507 Settings.Secure.putInt(getContentResolver(),
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700508 TTS_DEFAULT_RATE, mDefaultRate);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700509 if (mTts != null) {
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700510 mTts.setSpeechRate((float)(mDefaultRate/100.0f));
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700511 }
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700512 Log.i(TAG, "TTS default rate is " + mDefaultRate);
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700513 } catch (NumberFormatException e) {
514 Log.e(TAG, "could not persist default TTS rate setting", e);
515 }
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700516 } else if (KEY_TTS_DEFAULT_LANG.equals(preference.getKey())) {
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700517 // Default locale
518 ContentResolver resolver = getContentResolver();
519 parseLocaleInfo((String) objValue);
520 Settings.Secure.putString(resolver, TTS_DEFAULT_LANG, mDefaultLanguage);
521 Settings.Secure.putString(resolver, TTS_DEFAULT_COUNTRY, mDefaultCountry);
522 Settings.Secure.putString(resolver, TTS_DEFAULT_VARIANT, mDefaultLocVariant);
523 Log.v(TAG, "TTS default lang/country/variant set to "
524 + mDefaultLanguage + "/" + mDefaultCountry + "/" + mDefaultLocVariant);
Jean-Michel Trivi628431d2009-07-17 16:52:54 -0700525 if (mTts != null) {
Charles Chencf3998b2010-02-11 18:13:42 -0800526 mTts.setLanguage(new Locale(mDefaultLanguage, mDefaultCountry, mDefaultLocVariant));
Jean-Michel Trivi628431d2009-07-17 16:52:54 -0700527 }
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700528 int newIndex = mDefaultLocPref.findIndexOfValue((String)objValue);
529 Log.v("Settings", " selected is " + newIndex);
530 mDemoStringIndex = newIndex > -1 ? newIndex : 0;
Charles Chen5dbc74a2009-12-07 12:08:13 -0800531 } else if (KEY_TTS_DEFAULT_SYNTH.equals(preference.getKey())) {
Charles Chen5dbc74a2009-12-07 12:08:13 -0800532 mDefaultEng = objValue.toString();
533 Settings.Secure.putString(getContentResolver(), TTS_DEFAULT_SYNTH, mDefaultEng);
534 if (mTts != null) {
535 mTts.setEngineByPackageName(mDefaultEng);
Charles Chencf3998b2010-02-11 18:13:42 -0800536 mEnableDemo = false;
537 mVoicesMissing = false;
538 updateWidgetState();
539 checkVoiceData();
Charles Chen5dbc74a2009-12-07 12:08:13 -0800540 }
541 Log.v("Settings", "The default synth is: " + objValue.toString());
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700542 }
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700543
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700544 return true;
545 }
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700546
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700547
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700548 /**
549 * Called when mPlayExample or mInstallData is clicked
550 */
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700551 public boolean onPreferenceClick(Preference preference) {
552 if (preference == mPlayExample) {
Charles Chen4df6c792010-01-22 11:17:40 -0800553 // Get the sample text from the TTS engine; onActivityResult will do
554 // the actual speaking
555 getSampleText();
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700556 return true;
557 }
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700558 if (preference == mInstallData) {
Jean-Michel Trivi2acc02e2009-06-25 10:03:43 -0700559 installVoiceData();
560 // quit this activity so it needs to be restarted after installation of the voice data
561 finish();
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700562 return true;
563 }
Jean-Michel Trivi74e565d2009-06-18 18:44:52 -0700564 return false;
565 }
566
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700567 @Override
568 public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference) {
569 if (Utils.isMonkeyRunning()) {
570 return false;
571 }
572
573 if (preference instanceof CheckBoxPreference) {
574 final CheckBoxPreference chkPref = (CheckBoxPreference) preference;
575 if (!chkPref.getKey().equals(KEY_TTS_USE_DEFAULT)){
576 if (chkPref.isChecked()) {
577 chkPref.setChecked(false);
578 AlertDialog d = (new AlertDialog.Builder(this))
579 .setTitle(android.R.string.dialog_alert_title)
580 .setIcon(android.R.drawable.ic_dialog_alert)
581 .setMessage(getString(R.string.tts_engine_security_warning,
582 chkPref.getTitle()))
583 .setCancelable(true)
584 .setPositiveButton(android.R.string.ok,
585 new DialogInterface.OnClickListener() {
586 public void onClick(DialogInterface dialog, int which) {
587 chkPref.setChecked(true);
588 loadEngines();
589 }
590 })
591 .setNegativeButton(android.R.string.cancel,
592 new DialogInterface.OnClickListener() {
593 public void onClick(DialogInterface dialog, int which) {
594 }
595 })
596 .create();
597 d.show();
598 } else {
599 loadEngines();
600 }
601 return true;
602 }
603 }
604 return false;
605 }
606
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700607
608 private void updateWidgetState() {
609 mPlayExample.setEnabled(mEnableDemo);
610 mUseDefaultPref.setEnabled(mEnableDemo);
611 mDefaultRatePref.setEnabled(mEnableDemo);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700612 mDefaultLocPref.setEnabled(mEnableDemo);
613
Charles Chenc8298712010-02-10 13:58:23 -0800614 mInstallData.setEnabled(mVoicesMissing);
Jean-Michel Trivi1e6a45a2009-06-22 16:03:40 -0700615 }
616
617
618 private void parseLocaleInfo(String locale) {
619 StringTokenizer tokenizer = new StringTokenizer(locale, LOCALE_DELIMITER);
620 mDefaultLanguage = "";
621 mDefaultCountry = "";
622 mDefaultLocVariant = "";
623 if (tokenizer.hasMoreTokens()) {
624 mDefaultLanguage = tokenizer.nextToken().trim();
625 }
626 if (tokenizer.hasMoreTokens()) {
627 mDefaultCountry = tokenizer.nextToken().trim();
628 }
629 if (tokenizer.hasMoreTokens()) {
630 mDefaultLocVariant = tokenizer.nextToken().trim();
631 }
632 }
633
634
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700635 /**
636 * Initialize the default language in the UI and in the preferences.
637 * After this method has been invoked, the default language is a supported Locale.
638 */
639 private void initDefaultLang() {
640 // if there isn't already a default language preference
641 if (!hasLangPref()) {
642 // if the current Locale is supported
643 if (isCurrentLocSupported()) {
644 // then use the current Locale as the default language
645 useCurrentLocAsDefault();
646 } else {
647 // otherwise use a default supported Locale as the default language
648 useSupportedLocAsDefault();
649 }
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700650 }
651
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700652 // Update the language preference list with the default language and the matching
653 // demo string (at this stage there is a default language pref)
654 ContentResolver resolver = getContentResolver();
655 mDefaultLanguage = Settings.Secure.getString(resolver, TTS_DEFAULT_LANG);
Charles Chen681d0b82010-04-12 12:06:30 -0700656 mDefaultCountry = Settings.Secure.getString(resolver, TTS_DEFAULT_COUNTRY);
657 mDefaultLocVariant = Settings.Secure.getString(resolver, TTS_DEFAULT_VARIANT);
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700658
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700659 // update the demo string
660 mDemoStringIndex = mDefaultLocPref.findIndexOfValue(mDefaultLanguage + LOCALE_DELIMITER
661 + mDefaultCountry);
Charles Chen8a37e612010-02-04 15:52:30 -0800662 if (mDemoStringIndex > -1){
663 mDefaultLocPref.setValueIndex(mDemoStringIndex);
664 }
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700665 }
666
667 /**
668 * (helper function for initDefaultLang() )
669 * Returns whether there is a default language in the TTS settings.
670 */
671 private boolean hasLangPref() {
Jean-Baptiste Queru6e61b212010-04-14 10:40:48 -0700672 ContentResolver resolver = getContentResolver();
673 String language = Settings.Secure.getString(resolver, TTS_DEFAULT_LANG);
674 if ((language == null) || (language.length() < 1)) {
675 return false;
676 }
677 String country = Settings.Secure.getString(resolver, TTS_DEFAULT_COUNTRY);
678 if (country == null) {
679 return false;
680 }
681 String variant = Settings.Secure.getString(resolver, TTS_DEFAULT_VARIANT);
682 if (variant == null) {
683 return false;
684 }
685 return true;
Jean-Michel Trivie8e23db2009-09-02 15:15:33 -0700686 }
687
688 /**
689 * (helper function for initDefaultLang() )
690 * Returns whether the current Locale is supported by this Settings screen
691 */
692 private boolean isCurrentLocSupported() {
693 String currentLocID = Locale.getDefault().getISO3Language() + LOCALE_DELIMITER
694 + Locale.getDefault().getISO3Country();
695 return (mDefaultLocPref.findIndexOfValue(currentLocID) > -1);
696 }
697
698 /**
699 * (helper function for initDefaultLang() )
700 * Sets the default language in TTS settings to be the current Locale.
701 * This should only be used after checking that the current Locale is supported.
702 */
703 private void useCurrentLocAsDefault() {
704 Locale currentLocale = Locale.getDefault();
705 ContentResolver resolver = getContentResolver();
706 Settings.Secure.putString(resolver, TTS_DEFAULT_LANG, currentLocale.getISO3Language());
707 Settings.Secure.putString(resolver, TTS_DEFAULT_COUNTRY, currentLocale.getISO3Country());
708 Settings.Secure.putString(resolver, TTS_DEFAULT_VARIANT, currentLocale.getVariant());
709 }
710
711 /**
712 * (helper function for initDefaultLang() )
713 * Sets the default language in TTS settings to be one known to be supported
714 */
715 private void useSupportedLocAsDefault() {
716 ContentResolver resolver = getContentResolver();
717 Settings.Secure.putString(resolver, TTS_DEFAULT_LANG, DEFAULT_LANG_VAL);
718 Settings.Secure.putString(resolver, TTS_DEFAULT_COUNTRY, DEFAULT_COUNTRY_VAL);
719 Settings.Secure.putString(resolver, TTS_DEFAULT_VARIANT, DEFAULT_VARIANT_VAL);
Jean-Michel Trivi44fbbea2009-07-06 14:04:54 -0700720 }
721
Charles Chen5dbc74a2009-12-07 12:08:13 -0800722
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700723 private void loadEngines() {
Charles Chen8c8185b2010-04-08 16:51:35 -0700724 mDefaultSynthPref = (ListPreference) findPreference(KEY_TTS_DEFAULT_SYNTH);
Charles Chen5dbc74a2009-12-07 12:08:13 -0800725
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700726 // TODO (clchen): Try to see if it is possible to be more efficient here
727 // and not search for plugins again.
728 Intent intent = new Intent("android.intent.action.START_TTS_ENGINE");
729 ResolveInfo[] enginesArray = new ResolveInfo[0];
730 PackageManager pm = getPackageManager();
731 enginesArray = pm.queryIntentActivities(intent, 0).toArray(enginesArray);
732 ArrayList<CharSequence> entries = new ArrayList<CharSequence>();
733 ArrayList<CharSequence> values = new ArrayList<CharSequence>();
Charles Chenf47cce02010-03-17 17:33:23 -0700734 String enabledEngines = "";
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700735 for (int i = 0; i < enginesArray.length; i++) {
Charles Chenf47cce02010-03-17 17:33:23 -0700736 String pluginPackageName = enginesArray[i].activityInfo.packageName;
737 if (pluginPackageName.equals(SYSTEM_TTS)) {
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700738 entries.add(enginesArray[i].loadLabel(pm));
Charles Chenf47cce02010-03-17 17:33:23 -0700739 values.add(pluginPackageName);
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700740 } else {
741 CheckBoxPreference pref = (CheckBoxPreference) findPreference(
Charles Chenf47cce02010-03-17 17:33:23 -0700742 KEY_PLUGIN_ENABLED_PREFIX + pluginPackageName);
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700743 if ((pref != null) && pref.isChecked()){
744 entries.add(enginesArray[i].loadLabel(pm));
Charles Chenf47cce02010-03-17 17:33:23 -0700745 values.add(pluginPackageName);
746 enabledEngines = enabledEngines + pluginPackageName + " ";
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700747 }
748 }
749 }
Charles Chenf47cce02010-03-17 17:33:23 -0700750 ContentResolver resolver = getContentResolver();
751 Settings.Secure.putString(resolver, TTS_ENABLED_PLUGINS, enabledEngines);
Charles Chen5dbc74a2009-12-07 12:08:13 -0800752
Charles Chen0a0eb5f2010-03-16 20:09:17 -0700753 CharSequence entriesArray[] = new CharSequence[entries.size()];
754 CharSequence valuesArray[] = new CharSequence[values.size()];
755
Charles Chen8c8185b2010-04-08 16:51:35 -0700756 mDefaultSynthPref.setEntries(entries.toArray(entriesArray));
757 mDefaultSynthPref.setEntryValues(values.toArray(valuesArray));
Charles Chenbe6e8272010-03-22 16:06:05 -0700758
759 // Set the selected engine based on the saved preference
760 String selectedEngine = Settings.Secure.getString(getContentResolver(), TTS_DEFAULT_SYNTH);
Charles Chen8c8185b2010-04-08 16:51:35 -0700761 int selectedEngineIndex = mDefaultSynthPref.findIndexOfValue(selectedEngine);
Charles Chenbe6e8272010-03-22 16:06:05 -0700762 if (selectedEngineIndex == -1){
Charles Chen8c8185b2010-04-08 16:51:35 -0700763 selectedEngineIndex = mDefaultSynthPref.findIndexOfValue(SYSTEM_TTS);
Charles Chenbe6e8272010-03-22 16:06:05 -0700764 }
Charles Chen8c8185b2010-04-08 16:51:35 -0700765 mDefaultSynthPref.setValueIndex(selectedEngineIndex);
Charles Chen5dbc74a2009-12-07 12:08:13 -0800766 }
Charles Chen5dbc74a2009-12-07 12:08:13 -0800767
Jean-Michel Trivied29a652009-06-05 18:37:29 -0700768}