1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy of 6 * 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, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations under 14 * the License. 15 */ 16 17 package com.android.cellbroadcastreceiver; 18 19 import android.annotation.NonNull; 20 import android.content.Context; 21 import android.content.res.Configuration; 22 import android.content.res.Resources; 23 import android.graphics.Typeface; 24 import android.telephony.SmsCbCmasInfo; 25 import android.telephony.SmsCbEtwsInfo; 26 import android.telephony.SmsCbMessage; 27 import android.text.Spannable; 28 import android.text.SpannableStringBuilder; 29 import android.text.TextUtils; 30 import android.text.style.StyleSpan; 31 32 import com.android.cellbroadcastreceiver.CellBroadcastChannelManager.CellBroadcastChannelRange; 33 34 import java.text.DateFormat; 35 import java.util.Locale; 36 37 /** 38 * Returns the string resource ID's for CMAS and ETWS emergency alerts. 39 */ 40 public class CellBroadcastResources { 41 CellBroadcastResources()42 private CellBroadcastResources() { 43 } 44 45 /** 46 * Returns a styled CharSequence containing the message date/time and alert details. 47 * @param context a Context for resource string access 48 * @param showDebugInfo {@code true} if adding more information for debugging purposes. 49 * @param message The cell broadcast message. 50 * @param locationCheckTime The EPOCH time in milliseconds that Device-based Geo-fencing (DBGF) 51 * was last performed. 0 if the message does not have DBGF information. 52 * @param isDisplayed {@code true} if the message is displayed to the user. 53 * @param geometry Geometry string for device-based geo-fencing message. 54 * 55 * @return a CharSequence for display in the broadcast alert dialog 56 */ getMessageDetails(Context context, boolean showDebugInfo, SmsCbMessage message, long locationCheckTime, boolean isDisplayed, String geometry)57 public static CharSequence getMessageDetails(Context context, boolean showDebugInfo, 58 SmsCbMessage message, long locationCheckTime, 59 boolean isDisplayed, String geometry) { 60 SpannableStringBuilder buf = new SpannableStringBuilder(); 61 // Alert date/time 62 appendMessageDetail(context, buf, R.string.delivery_time_heading, 63 DateFormat.getDateTimeInstance().format(message.getReceivedTime())); 64 65 // Message id 66 if (showDebugInfo) { 67 appendMessageDetail(context, buf, R.string.message_identifier, 68 Integer.toString(message.getServiceCategory())); 69 appendMessageDetail(context, buf, R.string.message_serial_number, 70 Integer.toString(message.getSerialNumber())); 71 } 72 73 if (message.isCmasMessage()) { 74 // CMAS category, response type, severity, urgency, certainty 75 appendCmasAlertDetails(context, buf, message.getCmasWarningInfo()); 76 } 77 78 if (showDebugInfo) { 79 appendMessageDetail(context, buf, R.string.data_coding_scheme, 80 Integer.toString(message.getDataCodingScheme())); 81 82 appendMessageDetail(context, buf, R.string.message_content, message.getMessageBody()); 83 84 appendMessageDetail(context, buf, R.string.location_check_time, locationCheckTime == -1 85 ? "N/A" 86 : DateFormat.getDateTimeInstance().format(locationCheckTime)); 87 88 appendMessageDetail(context, buf, R.string.maximum_waiting_time, 89 message.getMaximumWaitingDuration() + " " 90 + context.getString(R.string.seconds)); 91 92 appendMessageDetail(context, buf, R.string.message_displayed, 93 Boolean.toString(isDisplayed)); 94 95 appendMessageDetail(context, buf, R.string.message_coordinates, 96 TextUtils.isEmpty(geometry) ? "N/A" : geometry); 97 } 98 99 return buf; 100 } 101 appendCmasAlertDetails(Context context, SpannableStringBuilder buf, SmsCbCmasInfo cmasInfo)102 private static void appendCmasAlertDetails(Context context, SpannableStringBuilder buf, 103 SmsCbCmasInfo cmasInfo) { 104 // CMAS category 105 int categoryId = getCmasCategoryResId(cmasInfo); 106 if (categoryId != 0) { 107 appendMessageDetail(context, buf, R.string.cmas_category_heading, 108 context.getString(categoryId)); 109 } 110 111 // CMAS response type 112 int responseId = getCmasResponseResId(cmasInfo); 113 if (responseId != 0) { 114 appendMessageDetail(context, buf, R.string.cmas_response_heading, 115 context.getString(responseId)); 116 } 117 118 // CMAS severity 119 int severityId = getCmasSeverityResId(cmasInfo); 120 if (severityId != 0) { 121 appendMessageDetail(context, buf, R.string.cmas_severity_heading, 122 context.getString(severityId)); 123 } 124 125 // CMAS urgency 126 int urgencyId = getCmasUrgencyResId(cmasInfo); 127 if (urgencyId != 0) { 128 appendMessageDetail(context, buf, R.string.cmas_urgency_heading, 129 context.getString(urgencyId)); 130 } 131 132 // CMAS certainty 133 int certaintyId = getCmasCertaintyResId(cmasInfo); 134 if (certaintyId != 0) { 135 appendMessageDetail(context, buf, R.string.cmas_certainty_heading, 136 context.getString(certaintyId)); 137 } 138 } 139 appendMessageDetail(Context context, SpannableStringBuilder buf, int typeId, String value)140 private static void appendMessageDetail(Context context, SpannableStringBuilder buf, 141 int typeId, String value) { 142 if (buf.length() != 0) { 143 buf.append("\n"); 144 } 145 int start = buf.length(); 146 buf.append(context.getString(typeId)); 147 int end = buf.length(); 148 buf.setSpan(new StyleSpan(Typeface.BOLD), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); 149 buf.append(" "); 150 buf.append(value); 151 } 152 153 /** 154 * Returns the string resource ID for the CMAS category. 155 * @return a string resource ID, or 0 if the CMAS category is unknown or not present 156 */ getCmasCategoryResId(SmsCbCmasInfo cmasInfo)157 private static int getCmasCategoryResId(SmsCbCmasInfo cmasInfo) { 158 switch (cmasInfo.getCategory()) { 159 case SmsCbCmasInfo.CMAS_CATEGORY_GEO: 160 return R.string.cmas_category_geo; 161 162 case SmsCbCmasInfo.CMAS_CATEGORY_MET: 163 return R.string.cmas_category_met; 164 165 case SmsCbCmasInfo.CMAS_CATEGORY_SAFETY: 166 return R.string.cmas_category_safety; 167 168 case SmsCbCmasInfo.CMAS_CATEGORY_SECURITY: 169 return R.string.cmas_category_security; 170 171 case SmsCbCmasInfo.CMAS_CATEGORY_RESCUE: 172 return R.string.cmas_category_rescue; 173 174 case SmsCbCmasInfo.CMAS_CATEGORY_FIRE: 175 return R.string.cmas_category_fire; 176 177 case SmsCbCmasInfo.CMAS_CATEGORY_HEALTH: 178 return R.string.cmas_category_health; 179 180 case SmsCbCmasInfo.CMAS_CATEGORY_ENV: 181 return R.string.cmas_category_env; 182 183 case SmsCbCmasInfo.CMAS_CATEGORY_TRANSPORT: 184 return R.string.cmas_category_transport; 185 186 case SmsCbCmasInfo.CMAS_CATEGORY_INFRA: 187 return R.string.cmas_category_infra; 188 189 case SmsCbCmasInfo.CMAS_CATEGORY_CBRNE: 190 return R.string.cmas_category_cbrne; 191 192 case SmsCbCmasInfo.CMAS_CATEGORY_OTHER: 193 return R.string.cmas_category_other; 194 195 default: 196 return 0; 197 } 198 } 199 200 /** 201 * Returns the string resource ID for the CMAS response type. 202 * @return a string resource ID, or 0 if the CMAS response type is unknown or not present 203 */ getCmasResponseResId(SmsCbCmasInfo cmasInfo)204 private static int getCmasResponseResId(SmsCbCmasInfo cmasInfo) { 205 switch (cmasInfo.getResponseType()) { 206 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_SHELTER: 207 return R.string.cmas_response_shelter; 208 209 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_EVACUATE: 210 return R.string.cmas_response_evacuate; 211 212 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_PREPARE: 213 return R.string.cmas_response_prepare; 214 215 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_EXECUTE: 216 return R.string.cmas_response_execute; 217 218 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_MONITOR: 219 return R.string.cmas_response_monitor; 220 221 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_AVOID: 222 return R.string.cmas_response_avoid; 223 224 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_ASSESS: 225 return R.string.cmas_response_assess; 226 227 case SmsCbCmasInfo.CMAS_RESPONSE_TYPE_NONE: 228 return R.string.cmas_response_none; 229 230 default: 231 return 0; 232 } 233 } 234 235 /** 236 * Returns the string resource ID for the CMAS severity. 237 * @return a string resource ID, or 0 if the CMAS severity is unknown or not present 238 */ getCmasSeverityResId(SmsCbCmasInfo cmasInfo)239 private static int getCmasSeverityResId(SmsCbCmasInfo cmasInfo) { 240 switch (cmasInfo.getSeverity()) { 241 case SmsCbCmasInfo.CMAS_SEVERITY_EXTREME: 242 return R.string.cmas_severity_extreme; 243 244 case SmsCbCmasInfo.CMAS_SEVERITY_SEVERE: 245 return R.string.cmas_severity_severe; 246 247 default: 248 return 0; 249 } 250 } 251 252 /** 253 * Returns the string resource ID for the CMAS urgency. 254 * @return a string resource ID, or 0 if the CMAS urgency is unknown or not present 255 */ getCmasUrgencyResId(SmsCbCmasInfo cmasInfo)256 private static int getCmasUrgencyResId(SmsCbCmasInfo cmasInfo) { 257 switch (cmasInfo.getUrgency()) { 258 case SmsCbCmasInfo.CMAS_URGENCY_IMMEDIATE: 259 return R.string.cmas_urgency_immediate; 260 261 case SmsCbCmasInfo.CMAS_URGENCY_EXPECTED: 262 return R.string.cmas_urgency_expected; 263 264 default: 265 return 0; 266 } 267 } 268 269 /** 270 * Returns the string resource ID for the CMAS certainty. 271 * @return a string resource ID, or 0 if the CMAS certainty is unknown or not present 272 */ getCmasCertaintyResId(SmsCbCmasInfo cmasInfo)273 private static int getCmasCertaintyResId(SmsCbCmasInfo cmasInfo) { 274 switch (cmasInfo.getCertainty()) { 275 case SmsCbCmasInfo.CMAS_CERTAINTY_OBSERVED: 276 return R.string.cmas_certainty_observed; 277 278 case SmsCbCmasInfo.CMAS_CERTAINTY_LIKELY: 279 return R.string.cmas_certainty_likely; 280 281 default: 282 return 0; 283 } 284 } 285 286 /** 287 * Return the English string for the SMS sender address. 288 * This exists as a temporary workaround for b/174972822 289 * @param context 290 * @param message 291 * @return 292 */ getSmsSenderAddressResourceEnglishString(@onNull Context context, @NonNull SmsCbMessage message)293 public static String getSmsSenderAddressResourceEnglishString(@NonNull Context context, 294 @NonNull SmsCbMessage message) { 295 296 int resId = getSmsSenderAddressResource(context, message); 297 298 Configuration conf = context.getResources().getConfiguration(); 299 conf = new Configuration(conf); 300 conf.setLocale(Locale.ENGLISH); 301 Context localizedContext = context.createConfigurationContext(conf); 302 return localizedContext.getResources().getText(resId).toString(); 303 } 304 305 /** 306 * @return the string resource ID for the SMS sender address. 307 * As a temporary workaround for b/174972822, prefer getSmsSenderAddressResourceEnglishString, 308 * which ignores all translations for non-English languages for these 4 strings. 309 */ getSmsSenderAddressResource(@onNull Context context, @NonNull SmsCbMessage message)310 public static int getSmsSenderAddressResource(@NonNull Context context, 311 @NonNull SmsCbMessage message) { 312 CellBroadcastChannelManager channelManager = new CellBroadcastChannelManager( 313 context, message.getSubscriptionId()); 314 final int serviceCategory = message.getServiceCategory(); 315 // store to different SMS threads based on channel mappings. 316 int resourcesKey = channelManager.getCellBroadcastChannelResourcesKey(serviceCategory); 317 if (resourcesKey == R.array.cmas_presidential_alerts_channels_range_strings) { 318 return R.string.sms_cb_sender_name_presidential; 319 } else if (resourcesKey == R.array.emergency_alerts_channels_range_strings) { 320 return R.string.sms_cb_sender_name_emergency; 321 } else if (resourcesKey == R.array.public_safety_messages_channels_range_strings) { 322 return R.string.sms_cb_sender_name_public_safety; 323 } 324 325 return R.string.sms_cb_sender_name_default; 326 } 327 getDialogTitleResource(Context context, SmsCbMessage message)328 static int getDialogTitleResource(Context context, SmsCbMessage message) { 329 // ETWS warning types 330 SmsCbEtwsInfo etwsInfo = message.getEtwsWarningInfo(); 331 if (etwsInfo != null) { 332 switch (etwsInfo.getWarningType()) { 333 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_EARTHQUAKE: 334 return R.string.etws_earthquake_warning; 335 336 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_TSUNAMI: 337 return R.string.etws_tsunami_warning; 338 339 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_EARTHQUAKE_AND_TSUNAMI: 340 return R.string.etws_earthquake_and_tsunami_warning; 341 342 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_TEST_MESSAGE: 343 return R.string.etws_test_message; 344 345 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_OTHER_EMERGENCY: 346 default: 347 return R.string.etws_other_emergency_type; 348 } 349 } 350 351 SmsCbCmasInfo cmasInfo = message.getCmasWarningInfo(); 352 int subId = message.getSubscriptionId(); 353 CellBroadcastChannelManager channelManager = new CellBroadcastChannelManager( 354 context, subId); 355 final int serviceCategory = message.getServiceCategory(); 356 int resourcesKey = channelManager.getCellBroadcastChannelResourcesKey(serviceCategory); 357 CellBroadcastChannelRange range = channelManager 358 .getCellBroadcastChannelRange(serviceCategory); 359 360 if (resourcesKey == R.array.emergency_alerts_channels_range_strings) { 361 return R.string.pws_other_message_identifiers; 362 } else if (resourcesKey == R.array.cmas_presidential_alerts_channels_range_strings) { 363 return R.string.cmas_presidential_level_alert; 364 } else if (resourcesKey == R.array.cmas_alert_extreme_channels_range_strings) { 365 return R.string.cmas_extreme_alert; 366 } else if (resourcesKey == R.array.cmas_alerts_severe_range_strings) { 367 return R.string.cmas_severe_alert; 368 } else if (resourcesKey == R.array.cmas_amber_alerts_channels_range_strings) { 369 return R.string.cmas_amber_alert; 370 } else if (resourcesKey == R.array.required_monthly_test_range_strings) { 371 return R.string.cmas_required_monthly_test; 372 } else if (resourcesKey == R.array.exercise_alert_range_strings) { 373 return R.string.cmas_exercise_alert; 374 } else if (resourcesKey == R.array.operator_defined_alert_range_strings) { 375 return R.string.cmas_operator_defined_alert; 376 } else if (resourcesKey == R.array.public_safety_messages_channels_range_strings) { 377 return R.string.public_safety_message; 378 } else if (resourcesKey == R.array.state_local_test_alert_range_strings) { 379 return R.string.state_local_test_alert; 380 } 381 382 if (channelManager.isEmergencyMessage(message)) { 383 if (resourcesKey == R.array.additional_cbs_channels_strings) { 384 switch (range.mAlertType) { 385 case DEFAULT: 386 return R.string.pws_other_message_identifiers; 387 case ETWS_EARTHQUAKE: 388 return R.string.etws_earthquake_warning; 389 case ETWS_TSUNAMI: 390 return R.string.etws_tsunami_warning; 391 case TEST: 392 return R.string.etws_test_message; 393 case ETWS_DEFAULT: 394 case OTHER: 395 return R.string.etws_other_emergency_type; 396 default: 397 break; 398 } 399 } 400 return R.string.pws_other_message_identifiers; 401 } else { 402 return R.string.cb_other_message_identifiers; 403 } 404 } 405 406 /** 407 * Choose pictogram resource according to etws type. 408 * 409 * @param context Application context 410 * @param message Cell broadcast message 411 * 412 * @return The resource of the pictogram, -1 if not available. 413 */ getDialogPictogramResource(Context context, SmsCbMessage message)414 static int getDialogPictogramResource(Context context, SmsCbMessage message) { 415 SmsCbEtwsInfo etwsInfo = message.getEtwsWarningInfo(); 416 if (etwsInfo != null) { 417 switch (etwsInfo.getWarningType()) { 418 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_EARTHQUAKE: 419 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_EARTHQUAKE_AND_TSUNAMI: 420 return R.drawable.pict_icon_earthquake; 421 case SmsCbEtwsInfo.ETWS_WARNING_TYPE_TSUNAMI: 422 return R.drawable.pict_icon_tsunami; 423 } 424 } 425 426 final int serviceCategory = message.getServiceCategory(); 427 int subId = message.getSubscriptionId(); 428 CellBroadcastChannelManager channelManager = new CellBroadcastChannelManager( 429 context, subId); 430 if (channelManager.isEmergencyMessage(message)) { 431 if (channelManager.getCellBroadcastChannelResourcesKey(serviceCategory) 432 == R.array.additional_cbs_channels_strings) { 433 CellBroadcastChannelRange range = channelManager 434 .getCellBroadcastChannelRangeFromMessage(message); 435 // Apply the closest title to the specified tones. 436 switch (range.mAlertType) { 437 case ETWS_EARTHQUAKE: 438 return R.drawable.pict_icon_earthquake; 439 case ETWS_TSUNAMI: 440 return R.drawable.pict_icon_tsunami; 441 default: 442 break; 443 } 444 } 445 return -1; 446 } 447 return -1; 448 } 449 450 /** 451 * If the carrier or country is configured to show the alert dialog title text 452 * and the alert notification title in the language matching the message, this method returns 453 * the string in that language. 454 * Otherwise this method returns the string in the device's current language 455 * 456 * @param resId resource Id 457 * @param res Resources for the subId 458 * @param languageCode the ISO-639-1 language code for this message, or null if unspecified 459 */ overrideTranslation(Context context, int resId, Resources res, String languageCode)460 public static String overrideTranslation(Context context, int resId, Resources res, 461 String languageCode) { 462 if (!TextUtils.isEmpty(languageCode) 463 && res.getBoolean(R.bool.override_alert_title_language_to_match_message_locale)) { 464 // TODO change resources to locale from message 465 Configuration conf = res.getConfiguration(); 466 conf = new Configuration(conf); 467 conf.setLocale(new Locale(languageCode)); 468 Context localizedContext = context.createConfigurationContext(conf); 469 return localizedContext.getResources().getText(resId).toString(); 470 } else { 471 return res.getText(resId).toString(); 472 } 473 } 474 } 475