• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2015 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 
17 package com.android.tv.settings.system;
18 
19 import android.content.ActivityNotFoundException;
20 import android.content.BroadcastReceiver;
21 import android.content.Context;
22 import android.content.Intent;
23 import android.content.IntentFilter;
24 import android.os.Bundle;
25 import android.speech.tts.TextToSpeech;
26 import android.speech.tts.TtsEngines;
27 import android.support.annotation.NonNull;
28 import android.support.v7.preference.ListPreference;
29 import android.support.v7.preference.Preference;
30 import android.support.v7.preference.PreferenceScreen;
31 import android.text.TextUtils;
32 import android.util.Log;
33 import android.util.Pair;
34 
35 import com.android.internal.logging.nano.MetricsProto;
36 import com.android.tv.settings.R;
37 import com.android.tv.settings.SettingsPreferenceFragment;
38 
39 import java.util.ArrayList;
40 import java.util.Locale;
41 
42 /**
43  * The text-to-speech engine settings screen in TV Settings.
44  */
45 public class TtsEngineSettingsFragment extends SettingsPreferenceFragment implements
46         Preference.OnPreferenceClickListener, Preference.OnPreferenceChangeListener {
47     private static final String TAG = "TtsEngineSettings";
48     private static final boolean DBG = false;
49 
50     /**
51      * Key for the name of the TTS engine passed in to the engine
52      * settings fragment {@link TtsEngineSettingsFragment}.
53      */
54     private static final String ARG_ENGINE_NAME = "engineName";
55 
56     /**
57      * Key for the label of the TTS engine passed in to the engine
58      * settings fragment. This is used as the title of the fragment
59      * {@link TtsEngineSettingsFragment}.
60      */
61     private static final String ARG_ENGINE_LABEL = "engineLabel";
62 
63     /**
64      * Key for the voice data data passed in to the engine settings
65      * fragmetn {@link TtsEngineSettingsFragment}.
66      */
67     private static final String ARG_VOICES = "voices";
68 
69 
70     private static final String KEY_ENGINE_LOCALE = "tts_default_lang";
71     private static final String KEY_ENGINE_SETTINGS = "tts_engine_settings";
72     private static final String KEY_INSTALL_DATA = "tts_install_data";
73 
74     private static final String STATE_KEY_LOCALE_ENTRIES = "locale_entries";
75     private static final String STATE_KEY_LOCALE_ENTRY_VALUES= "locale_entry_values";
76     private static final String STATE_KEY_LOCALE_VALUE = "locale_value";
77 
78     private static final int VOICE_DATA_INTEGRITY_CHECK = 1977;
79 
80     private TtsEngines mEnginesHelper;
81     private ListPreference mLocalePreference;
82     private Preference mEngineSettingsPreference;
83     private Preference mInstallVoicesPreference;
84     private Intent mVoiceDataDetails;
85 
86     private TextToSpeech mTts;
87 
88     private int mSelectedLocaleIndex = -1;
89 
90     private final TextToSpeech.OnInitListener mTtsInitListener = new TextToSpeech.OnInitListener() {
91         @Override
92         public void onInit(int status) {
93             if (status != TextToSpeech.SUCCESS) {
94                 getFragmentManager().popBackStack();
95             } else {
96                 getActivity().runOnUiThread(new Runnable() {
97                     @Override
98                     public void run() {
99                         mLocalePreference.setEnabled(true);
100                     }
101                 });
102             }
103         }
104     };
105 
106     private final BroadcastReceiver mLanguagesChangedReceiver = new BroadcastReceiver() {
107         @Override
108         public void onReceive(Context context, Intent intent) {
109             // Installed or uninstalled some data packs
110             if (TextToSpeech.Engine.ACTION_TTS_DATA_INSTALLED.equals(intent.getAction())) {
111                 checkTtsData();
112             }
113         }
114     };
115 
prepareArgs(@onNull Bundle args, String engineName, String engineLabel, Intent voiceCheckData)116     public static void prepareArgs(@NonNull Bundle args, String engineName, String engineLabel,
117             Intent voiceCheckData) {
118         args.clear();
119 
120         args.putString(ARG_ENGINE_NAME, engineName);
121         args.putString(ARG_ENGINE_LABEL, engineLabel);
122         args.putParcelable(ARG_VOICES, voiceCheckData);
123     }
124 
125     @Override
onCreatePreferences(Bundle savedInstanceState, String rootKey)126     public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
127 
128         addPreferencesFromResource(R.xml.tts_engine_settings);
129 
130         final PreferenceScreen screen = getPreferenceScreen();
131         screen.setTitle(getEngineLabel());
132         screen.setKey(getEngineName());
133 
134         mLocalePreference = (ListPreference) findPreference(KEY_ENGINE_LOCALE);
135         mLocalePreference.setOnPreferenceChangeListener(this);
136         mEngineSettingsPreference = findPreference(KEY_ENGINE_SETTINGS);
137         mEngineSettingsPreference.setOnPreferenceClickListener(this);
138         mInstallVoicesPreference = findPreference(KEY_INSTALL_DATA);
139         mInstallVoicesPreference.setOnPreferenceClickListener(this);
140 
141         mEngineSettingsPreference.setTitle(getResources().getString(
142                 R.string.tts_engine_settings_title, getEngineLabel()));
143         final Intent settingsIntent = mEnginesHelper.getSettingsIntent(getEngineName());
144         mEngineSettingsPreference.setIntent(settingsIntent);
145         if (settingsIntent == null) {
146             mEngineSettingsPreference.setEnabled(false);
147         }
148         mInstallVoicesPreference.setEnabled(false);
149 
150         if (savedInstanceState == null) {
151             mLocalePreference.setEnabled(false);
152             mLocalePreference.setEntries(new CharSequence[0]);
153             mLocalePreference.setEntryValues(new CharSequence[0]);
154         } else {
155             // Repopulate mLocalePreference with saved state. Will be updated later with
156             // up-to-date values when checkTtsData() calls back with results.
157             final CharSequence[] entries =
158                     savedInstanceState.getCharSequenceArray(STATE_KEY_LOCALE_ENTRIES);
159             final CharSequence[] entryValues =
160                     savedInstanceState.getCharSequenceArray(STATE_KEY_LOCALE_ENTRY_VALUES);
161             final CharSequence value =
162                     savedInstanceState.getCharSequence(STATE_KEY_LOCALE_VALUE);
163 
164             mLocalePreference.setEntries(entries);
165             mLocalePreference.setEntryValues(entryValues);
166             mLocalePreference.setValue(value != null ? value.toString() : null);
167             mLocalePreference.setEnabled(entries.length > 0);
168         }
169 
170     }
171 
172     @Override
onCreate(Bundle savedInstanceState)173     public void onCreate(Bundle savedInstanceState) {
174         mEnginesHelper = new TtsEngines(getActivity());
175 
176         super.onCreate(savedInstanceState);
177 
178         mVoiceDataDetails = getArguments().getParcelable(ARG_VOICES);
179 
180         mTts = new TextToSpeech(getActivity().getApplicationContext(), mTtsInitListener,
181                 getEngineName());
182 
183         // Check if data packs changed
184         checkTtsData();
185 
186         getActivity().registerReceiver(mLanguagesChangedReceiver,
187                 new IntentFilter(TextToSpeech.Engine.ACTION_TTS_DATA_INSTALLED));
188     }
189 
190     @Override
onDestroy()191     public void onDestroy() {
192         getActivity().unregisterReceiver(mLanguagesChangedReceiver);
193         mTts.shutdown();
194         super.onDestroy();
195     }
196 
197     @Override
onSaveInstanceState(Bundle outState)198     public void onSaveInstanceState(Bundle outState) {
199         super.onSaveInstanceState(outState);
200 
201         // Save the mLocalePreference values, so we can repopulate it with entries.
202         outState.putCharSequenceArray(STATE_KEY_LOCALE_ENTRIES,
203                 mLocalePreference.getEntries());
204         outState.putCharSequenceArray(STATE_KEY_LOCALE_ENTRY_VALUES,
205                 mLocalePreference.getEntryValues());
206         outState.putCharSequence(STATE_KEY_LOCALE_VALUE,
207                 mLocalePreference.getValue());
208     }
209 
checkTtsData()210     private void checkTtsData() {
211         Intent intent = new Intent(TextToSpeech.Engine.ACTION_CHECK_TTS_DATA);
212         intent.setPackage(getEngineName());
213         try {
214             if (DBG) Log.d(TAG, "Updating engine: Checking voice data: " + intent.toUri(0));
215             startActivityForResult(intent, VOICE_DATA_INTEGRITY_CHECK);
216         } catch (ActivityNotFoundException ex) {
217             Log.e(TAG, "Failed to check TTS data, no activity found for " + intent + ")");
218         }
219     }
220 
221     @Override
onActivityResult(int requestCode, int resultCode, Intent data)222     public void onActivityResult(int requestCode, int resultCode, Intent data) {
223         if (requestCode == VOICE_DATA_INTEGRITY_CHECK) {
224             if (resultCode != TextToSpeech.Engine.CHECK_VOICE_DATA_FAIL) {
225                 updateVoiceDetails(data);
226             } else {
227                 Log.e(TAG, "CheckVoiceData activity failed");
228             }
229         }
230     }
231 
updateVoiceDetails(Intent data)232     private void updateVoiceDetails(Intent data) {
233         if (data == null){
234             Log.e(TAG, "Engine failed voice data integrity check (null return)" +
235                     mTts.getCurrentEngine());
236             return;
237         }
238         mVoiceDataDetails = data;
239 
240         if (DBG) Log.d(TAG, "Parsing voice data details, data: " + mVoiceDataDetails.toUri(0));
241 
242         final ArrayList<String> available = mVoiceDataDetails.getStringArrayListExtra(
243                 TextToSpeech.Engine.EXTRA_AVAILABLE_VOICES);
244         final ArrayList<String> unavailable = mVoiceDataDetails.getStringArrayListExtra(
245                 TextToSpeech.Engine.EXTRA_UNAVAILABLE_VOICES);
246 
247         if (unavailable != null && unavailable.size() > 0) {
248             mInstallVoicesPreference.setEnabled(true);
249         } else {
250             mInstallVoicesPreference.setEnabled(false);
251         }
252 
253         if (available == null){
254             Log.e(TAG, "TTS data check failed (available == null).");
255             mLocalePreference.setEnabled(false);
256         } else {
257             updateDefaultLocalePref(available);
258         }
259     }
260 
updateDefaultLocalePref(ArrayList<String> availableLangs)261     private void updateDefaultLocalePref(ArrayList<String> availableLangs) {
262         if (availableLangs == null || availableLangs.size() == 0) {
263             mLocalePreference.setEnabled(false);
264             return;
265         }
266         Locale currentLocale = null;
267         if (!mEnginesHelper.isLocaleSetToDefaultForEngine(getEngineName())) {
268             currentLocale = mEnginesHelper.getLocalePrefForEngine(getEngineName());
269         }
270 
271         ArrayList<Pair<String, Locale>> entryPairs =
272                 new ArrayList<>(availableLangs.size());
273         for (int i = 0; i < availableLangs.size(); i++) {
274             Locale locale = mEnginesHelper.parseLocaleString(availableLangs.get(i));
275             if (locale != null){
276                 entryPairs.add(new Pair<>(locale.getDisplayName(), locale));
277             }
278         }
279 
280         // Sort it
281         entryPairs.sort((lhs, rhs) -> lhs.first.compareToIgnoreCase(rhs.first));
282 
283         // Get two arrays out of one of pairs
284         mSelectedLocaleIndex = 0; // Will point to the R.string.tts_lang_use_system value
285         CharSequence[] entries = new CharSequence[availableLangs.size()+1];
286         CharSequence[] entryValues = new CharSequence[availableLangs.size()+1];
287 
288         entries[0] = getString(R.string.tts_lang_use_system);
289         entryValues[0] = "";
290 
291         int i = 1;
292         for (Pair<String, Locale> entry : entryPairs) {
293             if (entry.second.equals(currentLocale)) {
294                 mSelectedLocaleIndex = i;
295             }
296             entries[i] = entry.first;
297             entryValues[i++] = entry.second.toString();
298         }
299 
300         mLocalePreference.setEntries(entries);
301         mLocalePreference.setEntryValues(entryValues);
302         mLocalePreference.setEnabled(true);
303         setLocalePreference(mSelectedLocaleIndex);
304     }
305 
306     /** Set entry from entry table in mLocalePreference */
setLocalePreference(int index)307     private void setLocalePreference(int index) {
308         if (index < 0) {
309             mLocalePreference.setValue("");
310             mLocalePreference.setSummary(R.string.tts_lang_not_selected);
311         } else {
312             mLocalePreference.setValueIndex(index);
313             mLocalePreference.setSummary(mLocalePreference.getEntries()[index]);
314         }
315     }
316 
317     /**
318      * Ask the current default engine to launch the matching INSTALL_TTS_DATA activity
319      * so the required TTS files are properly installed.
320      */
installVoiceData()321     private void installVoiceData() {
322         if (TextUtils.isEmpty(getEngineName())) return;
323         Intent intent = new Intent(TextToSpeech.Engine.ACTION_INSTALL_TTS_DATA);
324         intent.setPackage(getEngineName());
325         try {
326             startActivity(intent);
327         } catch (ActivityNotFoundException ex) {
328             Log.e(TAG, "Failed to install TTS data, no activity found for " + intent + ")");
329         }
330     }
331 
332     @Override
onPreferenceClick(Preference preference)333     public boolean onPreferenceClick(Preference preference) {
334         if (preference == mInstallVoicesPreference) {
335             installVoiceData();
336             return true;
337         }
338 
339         return false;
340     }
341 
342     @Override
onPreferenceChange(Preference preference, Object newValue)343     public boolean onPreferenceChange(Preference preference, Object newValue) {
344         if (preference == mLocalePreference) {
345             String localeString = (String) newValue;
346             updateLanguageTo((!TextUtils.isEmpty(localeString) ?
347                     mEnginesHelper.parseLocaleString(localeString) : null));
348             return true;
349         }
350         return false;
351     }
352 
updateLanguageTo(Locale locale)353     private void updateLanguageTo(Locale locale) {
354         int selectedLocaleIndex = -1;
355         String localeString = (locale != null) ? locale.toString() : "";
356         for (int i=0; i < mLocalePreference.getEntryValues().length; i++) {
357             if (localeString.equalsIgnoreCase(mLocalePreference.getEntryValues()[i].toString())) {
358                 selectedLocaleIndex = i;
359                 break;
360             }
361         }
362 
363         if (selectedLocaleIndex == -1) {
364             Log.w(TAG, "updateLanguageTo called with unknown locale argument");
365             return;
366         }
367         mLocalePreference.setSummary(mLocalePreference.getEntries()[selectedLocaleIndex]);
368         mSelectedLocaleIndex = selectedLocaleIndex;
369 
370         mEnginesHelper.updateLocalePrefForEngine(getEngineName(), locale);
371 
372         if (getEngineName().equals(mTts.getCurrentEngine())) {
373             // Null locale means "use system default"
374             mTts.setLanguage((locale != null) ? locale : Locale.getDefault());
375         }
376     }
377 
getEngineName()378     private String getEngineName() {
379         return getArguments().getString(ARG_ENGINE_NAME);
380     }
381 
getEngineLabel()382     private String getEngineLabel() {
383         return getArguments().getString(ARG_ENGINE_LABEL);
384     }
385 
386     @Override
getMetricsCategory()387     public int getMetricsCategory() {
388         return MetricsProto.MetricsEvent.TTS_ENGINE_SETTINGS;
389     }
390 }
391