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