Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2016 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 | */ |
mariagpuyol | 5874835 | 2016-03-09 15:36:28 -0800 | [diff] [blame] | 16 | package com.android.emergency.preferences; |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 17 | |
Akshay Kannan | 44ef39d | 2016-01-26 09:08:41 -0800 | [diff] [blame] | 18 | import android.app.AlertDialog; |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 19 | import android.content.Context; |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 20 | import android.content.DialogInterface; |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 21 | import android.content.Intent; |
mariagpuyol | 95dc040 | 2016-02-17 11:12:46 -0800 | [diff] [blame] | 22 | import android.graphics.drawable.Drawable; |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 23 | import android.net.Uri; |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 24 | import android.os.Bundle; |
| 25 | import android.os.Parcel; |
| 26 | import android.os.Parcelable; |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 27 | import android.preference.Preference; |
mariagpuyol | aee2378 | 2016-02-16 13:29:49 -0800 | [diff] [blame] | 28 | import android.support.annotation.NonNull; |
mariagpuyol | 16472ca | 2016-02-16 17:49:26 -0800 | [diff] [blame] | 29 | import android.support.annotation.Nullable; |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 30 | import android.view.View; |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 31 | |
mariagpuyol | 5874835 | 2016-03-09 15:36:28 -0800 | [diff] [blame] | 32 | import com.android.emergency.EmergencyContactManager; |
mariagpuyol | 5874835 | 2016-03-09 15:36:28 -0800 | [diff] [blame] | 33 | import com.android.emergency.R; |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 34 | import com.android.emergency.ReloadablePreferenceInterface; |
Akshay Kannan | 44ef39d | 2016-01-26 09:08:41 -0800 | [diff] [blame] | 35 | import com.android.internal.logging.MetricsLogger; |
| 36 | import com.android.internal.logging.MetricsProto.MetricsEvent; |
mariagpuyol | 9be3265 | 2016-03-16 12:34:53 -0700 | [diff] [blame] | 37 | import com.android.settingslib.drawable.CircleFramedDrawable; |
Akshay Kannan | 44ef39d | 2016-01-26 09:08:41 -0800 | [diff] [blame] | 38 | |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 39 | |
| 40 | /** |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 41 | * A {@link Preference} to display or call a contact using the specified URI string. |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 42 | */ |
| 43 | public class ContactPreference extends Preference { |
| 44 | |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 45 | private EmergencyContactManager.Contact mContact; |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 46 | @Nullable private RemoveContactPreferenceListener mRemoveContactPreferenceListener; |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 47 | @Nullable private AlertDialog mRemoveContactDialog; |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 48 | |
| 49 | /** |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 50 | * Listener for removing a contact. |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 51 | */ |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 52 | public interface RemoveContactPreferenceListener { |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 53 | /** |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 54 | * Callback to remove a contact preference. |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 55 | */ |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 56 | void onRemoveContactPreference(ContactPreference preference); |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 57 | } |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 58 | |
| 59 | /** |
| 60 | * Instantiates a ContactPreference that displays an emergency contact, taking in a Context and |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 61 | * the Uri. |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 62 | */ |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 63 | public ContactPreference(Context context, @NonNull Uri contactUri) { |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 64 | super(context); |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 65 | setOrder(DEFAULT_ORDER); |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 66 | |
| 67 | setUri(contactUri); |
| 68 | |
| 69 | setWidgetLayoutResource(R.layout.preference_user_delete_widget); |
| 70 | setPersistent(false); |
| 71 | } |
| 72 | |
| 73 | public void setUri(@NonNull Uri contactUri) { |
| 74 | if (mContact != null && !contactUri.equals(mContact.getContactUri()) && |
| 75 | mRemoveContactDialog != null) { |
| 76 | mRemoveContactDialog.dismiss(); |
| 77 | } |
| 78 | |
| 79 | mContact = EmergencyContactManager.getContact(getContext(), contactUri); |
| 80 | |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 81 | setTitle(mContact.getName()); |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 82 | setKey(mContact.getContactUri().toString()); |
mariagpuyol | f9ab89a | 2016-03-10 09:19:07 -0800 | [diff] [blame] | 83 | String summary = mContact.getPhoneType() == null ? |
| 84 | mContact.getPhoneNumber() : |
| 85 | String.format( |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 86 | getContext().getResources().getString(R.string.phone_type_and_phone_number), |
mariagpuyol | f9ab89a | 2016-03-10 09:19:07 -0800 | [diff] [blame] | 87 | mContact.getPhoneType(), |
| 88 | mContact.getPhoneNumber()); |
| 89 | setSummary(summary); |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 90 | |
| 91 | // Update the message to show the correct name. |
| 92 | if (mRemoveContactDialog != null) { |
| 93 | mRemoveContactDialog.setMessage( |
| 94 | String.format(getContext().getString(R.string.remove_contact), |
| 95 | mContact.getName())); |
| 96 | } |
mariagpuyol | 95dc040 | 2016-02-17 11:12:46 -0800 | [diff] [blame] | 97 | |
| 98 | //TODO: Consider doing the following in a non-UI thread. |
mariagpuyol | 9be3265 | 2016-03-16 12:34:53 -0700 | [diff] [blame] | 99 | Drawable icon; |
| 100 | if (mContact.getPhoto() != null) { |
| 101 | icon = new CircleFramedDrawable(mContact.getPhoto(), |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 102 | (int) getContext().getResources().getDimension(R.dimen.circle_avatar_size)); |
mariagpuyol | 9be3265 | 2016-03-16 12:34:53 -0700 | [diff] [blame] | 103 | } else { |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 104 | icon = getContext().getResources().getDrawable(R.drawable.ic_person_black_24dp); |
mariagpuyol | 9be3265 | 2016-03-16 12:34:53 -0700 | [diff] [blame] | 105 | } |
mariagpuyol | 95dc040 | 2016-02-17 11:12:46 -0800 | [diff] [blame] | 106 | setIcon(icon); |
| 107 | } |
| 108 | |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 109 | /** Listener to be informed when a contact preference should be deleted. */ |
| 110 | public void setRemoveContactPreferenceListener( |
| 111 | RemoveContactPreferenceListener removeContactListener) { |
| 112 | mRemoveContactPreferenceListener = removeContactListener; |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 113 | if (mRemoveContactPreferenceListener == null) { |
| 114 | mRemoveContactDialog = null; |
| 115 | return; |
| 116 | } |
| 117 | if (mRemoveContactDialog != null) { |
| 118 | return; |
| 119 | } |
| 120 | // Create the remove contact dialog |
| 121 | AlertDialog.Builder builder = new AlertDialog.Builder(getContext()); |
| 122 | builder.setNegativeButton(getContext().getString(R.string.cancel), null); |
| 123 | builder.setPositiveButton(getContext().getString(R.string.remove), |
| 124 | new DialogInterface.OnClickListener() { |
| 125 | @Override |
| 126 | public void onClick(DialogInterface dialogInterface, |
| 127 | int which) { |
| 128 | if (mRemoveContactPreferenceListener != null) { |
| 129 | mRemoveContactPreferenceListener |
| 130 | .onRemoveContactPreference(ContactPreference.this); |
| 131 | } |
| 132 | } |
| 133 | }); |
| 134 | builder.setMessage(String.format(getContext().getString(R.string.remove_contact), |
| 135 | mContact.getName())); |
| 136 | mRemoveContactDialog = builder.create(); |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 137 | } |
| 138 | |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 139 | @Override |
| 140 | protected void onBindView(View view) { |
| 141 | super.onBindView(view); |
| 142 | View deleteContactIcon = view.findViewById(R.id.delete_contact); |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 143 | if (mRemoveContactPreferenceListener == null) { |
mariagpuyol | 16472ca | 2016-02-16 17:49:26 -0800 | [diff] [blame] | 144 | deleteContactIcon.setVisibility(View.GONE); |
| 145 | } else { |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 146 | deleteContactIcon.setOnClickListener(new View.OnClickListener() { |
| 147 | @Override |
| 148 | public void onClick(View view) { |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 149 | showRemoveContactDialog(null); |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 150 | } |
| 151 | }); |
| 152 | |
mariagpuyol | d19ace5 | 2016-02-12 10:37:26 -0800 | [diff] [blame] | 153 | } |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 154 | } |
| 155 | |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 156 | public Uri getContactUri() { |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 157 | return mContact.getContactUri(); |
mariagpuyol | 609f68a | 2016-02-22 17:45:47 -0800 | [diff] [blame] | 158 | } |
| 159 | |
Akshay Kannan | cdb6c14 | 2016-01-20 18:25:44 -0800 | [diff] [blame] | 160 | /** |
Akshay Kannan | 44ef39d | 2016-01-26 09:08:41 -0800 | [diff] [blame] | 161 | * Calls the contact. |
| 162 | */ |
mariagpuyol | 5cf7f73 | 2016-03-08 09:48:15 -0800 | [diff] [blame] | 163 | public void callContact() { |
| 164 | Intent callIntent = |
| 165 | new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + mContact.getPhoneNumber())); |
| 166 | MetricsLogger.action(getContext(), MetricsEvent.ACTION_CALL_EMERGENCY_CONTACT); |
| 167 | getContext().startActivity(callIntent); |
mariagpuyol | 0504a44 | 2016-02-17 16:22:57 -0800 | [diff] [blame] | 168 | } |
mariagpuyol | 36b3f9d | 2016-03-11 16:02:31 -0800 | [diff] [blame] | 169 | |
| 170 | /** |
| 171 | * Displays a contact card for the contact. |
| 172 | */ |
| 173 | public void displayContact() { |
| 174 | Intent contactIntent = new Intent(Intent.ACTION_VIEW); |
| 175 | contactIntent.setData(mContact.getContactLookupUri()); |
| 176 | getContext().startActivity(contactIntent); |
| 177 | } |
mariagpuyol | 53745bf | 2016-03-23 14:43:55 -0700 | [diff] [blame^] | 178 | |
| 179 | /** Shows the dialog to remove the contact, restoring it from {@code state} if it's not null. */ |
| 180 | private void showRemoveContactDialog(Bundle state) { |
| 181 | if (mRemoveContactDialog == null) { |
| 182 | return; |
| 183 | } |
| 184 | if (state != null) { |
| 185 | mRemoveContactDialog.onRestoreInstanceState(state); |
| 186 | } |
| 187 | mRemoveContactDialog.show(); |
| 188 | } |
| 189 | |
| 190 | @Override |
| 191 | protected Parcelable onSaveInstanceState() { |
| 192 | final Parcelable superState = super.onSaveInstanceState(); |
| 193 | if (mRemoveContactDialog == null || !mRemoveContactDialog.isShowing()) { |
| 194 | return superState; |
| 195 | } |
| 196 | final SavedState myState = new SavedState(superState); |
| 197 | myState.isDialogShowing = true; |
| 198 | myState.dialogBundle = mRemoveContactDialog.onSaveInstanceState(); |
| 199 | return myState; |
| 200 | } |
| 201 | |
| 202 | @Override |
| 203 | protected void onRestoreInstanceState(Parcelable state) { |
| 204 | if (state == null || !state.getClass().equals(SavedState.class)) { |
| 205 | // Didn't save state for us in onSaveInstanceState |
| 206 | super.onRestoreInstanceState(state); |
| 207 | return; |
| 208 | } |
| 209 | SavedState myState = (SavedState) state; |
| 210 | super.onRestoreInstanceState(myState.getSuperState()); |
| 211 | if (myState.isDialogShowing) { |
| 212 | showRemoveContactDialog(myState.dialogBundle); |
| 213 | } |
| 214 | } |
| 215 | |
| 216 | private static class SavedState extends BaseSavedState { |
| 217 | boolean isDialogShowing; |
| 218 | Bundle dialogBundle; |
| 219 | |
| 220 | public SavedState(Parcel source) { |
| 221 | super(source); |
| 222 | isDialogShowing = source.readInt() == 1; |
| 223 | dialogBundle = source.readBundle(); |
| 224 | } |
| 225 | |
| 226 | @Override |
| 227 | public void writeToParcel(Parcel dest, int flags) { |
| 228 | super.writeToParcel(dest, flags); |
| 229 | dest.writeInt(isDialogShowing ? 1 : 0); |
| 230 | dest.writeBundle(dialogBundle); |
| 231 | } |
| 232 | |
| 233 | public SavedState(Parcelable superState) { |
| 234 | super(superState); |
| 235 | } |
| 236 | |
| 237 | public static final Parcelable.Creator<SavedState> CREATOR = |
| 238 | new Parcelable.Creator<SavedState>() { |
| 239 | public SavedState createFromParcel(Parcel in) { |
| 240 | return new SavedState(in); |
| 241 | } |
| 242 | |
| 243 | public SavedState[] newArray(int size) { |
| 244 | return new SavedState[size]; |
| 245 | } |
| 246 | }; |
| 247 | } |
mariagpuyol | bc6555b | 2016-02-26 15:38:19 -0800 | [diff] [blame] | 248 | } |