1 /* 2 * Copyright (C) 2010 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.contacts.editor; 18 19 import android.app.AlertDialog; 20 import android.app.Dialog; 21 import android.content.Context; 22 import android.content.DialogInterface; 23 import android.content.DialogInterface.OnShowListener; 24 import android.os.Bundle; 25 import android.os.Handler; 26 import android.text.Editable; 27 import android.text.TextUtils; 28 import android.text.TextUtils.TruncateAt; 29 import android.text.TextWatcher; 30 import android.util.AttributeSet; 31 import android.view.Gravity; 32 import android.view.LayoutInflater; 33 import android.view.View; 34 import android.view.ViewGroup; 35 import android.view.WindowManager; 36 import android.view.inputmethod.EditorInfo; 37 import android.widget.AdapterView; 38 import android.widget.AdapterView.OnItemSelectedListener; 39 import android.widget.ArrayAdapter; 40 import android.widget.Button; 41 import android.widget.EditText; 42 import android.widget.ImageView; 43 import android.widget.LinearLayout; 44 import android.widget.Spinner; 45 import android.widget.TextView; 46 47 import com.android.contacts.R; 48 import com.android.contacts.common.model.RawContactDelta; 49 import com.android.contacts.common.ContactsUtils; 50 import com.android.contacts.common.model.ValuesDelta; 51 import com.android.contacts.common.model.RawContactModifier; 52 import com.android.contacts.common.model.account.AccountType.EditType; 53 import com.android.contacts.common.model.dataitem.DataKind; 54 import com.android.contacts.util.DialogManager; 55 import com.android.contacts.util.DialogManager.DialogShowingView; 56 57 import java.util.List; 58 59 /** 60 * Base class for editors that handles labels and values. Uses 61 * {@link ValuesDelta} to read any existing {@link RawContact} values, and to 62 * correctly write any changes values. 63 */ 64 public abstract class LabeledEditorView extends LinearLayout implements Editor, DialogShowingView { 65 protected static final String DIALOG_ID_KEY = "dialog_id"; 66 private static final int DIALOG_ID_CUSTOM = 1; 67 68 private static final int INPUT_TYPE_CUSTOM = EditorInfo.TYPE_CLASS_TEXT 69 | EditorInfo.TYPE_TEXT_FLAG_CAP_WORDS; 70 71 private Spinner mLabel; 72 private EditTypeAdapter mEditTypeAdapter; 73 private View mDeleteContainer; 74 private ImageView mDelete; 75 76 private DataKind mKind; 77 private ValuesDelta mEntry; 78 private RawContactDelta mState; 79 private boolean mReadOnly; 80 private boolean mWasEmpty = true; 81 private boolean mIsDeletable = true; 82 private boolean mIsAttachedToWindow; 83 84 private EditType mType; 85 86 private ViewIdGenerator mViewIdGenerator; 87 private DialogManager mDialogManager = null; 88 private EditorListener mListener; 89 protected int mMinLineItemHeight; 90 91 /** 92 * A marker in the spinner adapter of the currently selected custom type. 93 */ 94 public static final EditType CUSTOM_SELECTION = new EditType(0, 0); 95 96 private OnItemSelectedListener mSpinnerListener = new OnItemSelectedListener() { 97 98 @Override 99 public void onItemSelected( 100 AdapterView<?> parent, View view, int position, long id) { 101 onTypeSelectionChange(position); 102 } 103 104 @Override 105 public void onNothingSelected(AdapterView<?> parent) { 106 } 107 }; 108 109 public LabeledEditorView(Context context) { 110 super(context); 111 init(context); 112 } 113 114 public LabeledEditorView(Context context, AttributeSet attrs) { 115 super(context, attrs); 116 init(context); 117 } 118 119 public LabeledEditorView(Context context, AttributeSet attrs, int defStyle) { 120 super(context, attrs, defStyle); 121 init(context); 122 } 123 124 private void init(Context context) { 125 mMinLineItemHeight = context.getResources().getDimensionPixelSize( 126 R.dimen.editor_min_line_item_height); 127 } 128 129 /** {@inheritDoc} */ 130 @Override 131 protected void onFinishInflate() { 132 133 mLabel = (Spinner) findViewById(R.id.spinner); 134 // Turn off the Spinner's own state management. We do this ourselves on rotation 135 mLabel.setId(View.NO_ID); 136 mLabel.setOnItemSelectedListener(mSpinnerListener); 137 138 mDelete = (ImageView) findViewById(R.id.delete_button); 139 mDeleteContainer = findViewById(R.id.delete_button_container); 140 mDeleteContainer.setOnClickListener(new OnClickListener() { 141 @Override 142 public void onClick(View v) { 143 // defer removal of this button so that the pressed state is visible shortly 144 new Handler().post(new Runnable() { 145 @Override 146 public void run() { 147 // Don't do anything if the view is no longer attached to the window 148 // (This check is needed because when this {@link Runnable} is executed, 149 // we can't guarantee the view is still valid. 150 if (!mIsAttachedToWindow) { 151 return; 152 } 153 // Send the delete request to the listener (which will in turn call 154 // deleteEditor() on this view if the deletion is valid - i.e. this is not 155 // the last {@link Editor} in the section). 156 if (mListener != null) { 157 mListener.onDeleteRequested(LabeledEditorView.this); 158 } 159 } 160 }); 161 } 162 }); 163 } 164 165 @Override 166 protected void onAttachedToWindow() { 167 super.onAttachedToWindow(); 168 // Keep track of when the view is attached or detached from the window, so we know it's 169 // safe to remove views (in case the user requests to delete this editor). 170 mIsAttachedToWindow = true; 171 } 172 173 @Override 174 protected void onDetachedFromWindow() { 175 super.onDetachedFromWindow(); 176 mIsAttachedToWindow = false; 177 } 178 179 @Override 180 public void deleteEditor() { 181 // Keep around in model, but mark as deleted 182 mEntry.markDeleted(); 183 184 // Remove the view 185 EditorAnimator.getInstance().removeEditorView(this); 186 } 187 188 public boolean isReadOnly() { 189 return mReadOnly; 190 } 191 192 public int getBaseline(int row) { 193 if (row == 0 && mLabel != null) { 194 return mLabel.getBaseline(); 195 } 196 return -1; 197 } 198 199 /** 200 * Configures the visibility of the type label button and enables or disables it properly. 201 */ 202 private void setupLabelButton(boolean shouldExist) { 203 if (shouldExist) { 204 mLabel.setEnabled(!mReadOnly && isEnabled()); 205 mLabel.setVisibility(View.VISIBLE); 206 } else { 207 mLabel.setVisibility(View.GONE); 208 } 209 } 210 211 /** 212 * Configures the visibility of the "delete" button and enables or disables it properly. 213 */ 214 private void setupDeleteButton() { 215 if (mIsDeletable) { 216 mDeleteContainer.setVisibility(View.VISIBLE); 217 mDelete.setEnabled(!mReadOnly && isEnabled()); 218 } else { 219 mDeleteContainer.setVisibility(View.GONE); 220 } 221 } 222 223 public void setDeleteButtonVisible(boolean visible) { 224 if (mIsDeletable) { 225 mDeleteContainer.setVisibility(visible ? View.VISIBLE : View.INVISIBLE); 226 } 227 } 228 229 protected void onOptionalFieldVisibilityChange() { 230 if (mListener != null) { 231 mListener.onRequest(EditorListener.EDITOR_FORM_CHANGED); 232 } 233 } 234 235 @Override 236 public void setEditorListener(EditorListener listener) { 237 mListener = listener; 238 } 239 240 @Override 241 public void setDeletable(boolean deletable) { 242 mIsDeletable = deletable; 243 setupDeleteButton(); 244 } 245 246 @Override 247 public void setEnabled(boolean enabled) { 248 super.setEnabled(enabled); 249 mLabel.setEnabled(!mReadOnly && enabled); 250 mDelete.setEnabled(!mReadOnly && enabled); 251 } 252 253 public Spinner getLabel() { 254 return mLabel; 255 } 256 257 public ImageView getDelete() { 258 return mDelete; 259 } 260 261 protected DataKind getKind() { 262 return mKind; 263 } 264 265 protected ValuesDelta getEntry() { 266 return mEntry; 267 } 268 269 protected EditType getType() { 270 return mType; 271 } 272 273 /** 274 * Build the current label state based on selected {@link EditType} and 275 * possible custom label string. 276 */ 277 private void rebuildLabel() { 278 mEditTypeAdapter = new EditTypeAdapter(mContext); 279 mLabel.setAdapter(mEditTypeAdapter); 280 if (mEditTypeAdapter.hasCustomSelection()) { 281 mLabel.setSelection(mEditTypeAdapter.getPosition(CUSTOM_SELECTION)); 282 } else { 283 mLabel.setSelection(mEditTypeAdapter.getPosition(mType)); 284 } 285 } 286 287 @Override 288 public void onFieldChanged(String column, String value) { 289 if (!isFieldChanged(column, value)) { 290 return; 291 } 292 293 // Field changes are saved directly 294 saveValue(column, value); 295 296 // Notify listener if applicable 297 notifyEditorListener(); 298 } 299 300 protected void saveValue(String column, String value) { 301 mEntry.put(column, value); 302 } 303 304 protected void notifyEditorListener() { 305 if (mListener != null) { 306 mListener.onRequest(EditorListener.FIELD_CHANGED); 307 } 308 309 boolean isEmpty = isEmpty(); 310 if (mWasEmpty != isEmpty) { 311 if (isEmpty) { 312 if (mListener != null) { 313 mListener.onRequest(EditorListener.FIELD_TURNED_EMPTY); 314 } 315 if (mIsDeletable) mDeleteContainer.setVisibility(View.INVISIBLE); 316 } else { 317 if (mListener != null) { 318 mListener.onRequest(EditorListener.FIELD_TURNED_NON_EMPTY); 319 } 320 if (mIsDeletable) mDeleteContainer.setVisibility(View.VISIBLE); 321 } 322 mWasEmpty = isEmpty; 323 } 324 } 325 326 protected boolean isFieldChanged(String column, String value) { 327 final String dbValue = mEntry.getAsString(column); 328 // nullable fields (e.g. Middle Name) are usually represented as empty columns, 329 // so lets treat null and empty space equivalently here 330 final String dbValueNoNull = dbValue == null ? "" : dbValue; 331 final String valueNoNull = value == null ? "" : value; 332 return !TextUtils.equals(dbValueNoNull, valueNoNull); 333 } 334 335 protected void rebuildValues() { 336 setValues(mKind, mEntry, mState, mReadOnly, mViewIdGenerator); 337 } 338 339 /** 340 * Prepare this editor using the given {@link DataKind} for defining 341 * structure and {@link ValuesDelta} describing the content to edit. 342 */ 343 @Override 344 public void setValues(DataKind kind, ValuesDelta entry, RawContactDelta state, boolean readOnly, 345 ViewIdGenerator vig) { 346 mKind = kind; 347 mEntry = entry; 348 mState = state; 349 mReadOnly = readOnly; 350 mViewIdGenerator = vig; 351 setId(vig.getId(state, kind, entry, ViewIdGenerator.NO_VIEW_INDEX)); 352 353 if (!entry.isVisible()) { 354 // Hide ourselves entirely if deleted 355 setVisibility(View.GONE); 356 return; 357 } 358 setVisibility(View.VISIBLE); 359 360 // Display label selector if multiple types available 361 final boolean hasTypes = RawContactModifier.hasEditTypes(kind); 362 setupLabelButton(hasTypes); 363 mLabel.setEnabled(!readOnly && isEnabled()); 364 if (hasTypes) { 365 mType = RawContactModifier.getCurrentType(entry, kind); 366 rebuildLabel(); 367 } 368 } 369 370 public ValuesDelta getValues() { 371 return mEntry; 372 } 373 374 /** 375 * Prepare dialog for entering a custom label. The input value is trimmed: white spaces before 376 * and after the input text is removed. 377 * <p> 378 * If the final value is empty, this change request is ignored; 379 * no empty text is allowed in any custom label. 380 */ 381 private Dialog createCustomDialog() { 382 final AlertDialog.Builder builder = new AlertDialog.Builder(mContext); 383 final LayoutInflater layoutInflater = LayoutInflater.from(builder.getContext()); 384 builder.setTitle(R.string.customLabelPickerTitle); 385 386 final View view = layoutInflater.inflate(R.layout.contact_editor_label_name_dialog, null); 387 final EditText editText = (EditText) view.findViewById(R.id.custom_dialog_content); 388 editText.setInputType(INPUT_TYPE_CUSTOM); 389 editText.setSaveEnabled(true); 390 391 builder.setView(view); 392 editText.requestFocus(); 393 394 builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { 395 @Override 396 public void onClick(DialogInterface dialog, int which) { 397 final String customText = editText.getText().toString().trim(); 398 if (ContactsUtils.isGraphic(customText)) { 399 final List<EditType> allTypes = 400 RawContactModifier.getValidTypes(mState, mKind, null); 401 mType = null; 402 for (EditType editType : allTypes) { 403 if (editType.customColumn != null) { 404 mType = editType; 405 break; 406 } 407 } 408 if (mType == null) return; 409 410 mEntry.put(mKind.typeColumn, mType.rawValue); 411 mEntry.put(mType.customColumn, customText); 412 rebuildLabel(); 413 requestFocusForFirstEditField(); 414 onLabelRebuilt(); 415 } 416 } 417 }); 418 419 builder.setNegativeButton(android.R.string.cancel, null); 420 421 final AlertDialog dialog = builder.create(); 422 dialog.setOnShowListener(new OnShowListener() { 423 @Override 424 public void onShow(DialogInterface dialogInterface) { 425 updateCustomDialogOkButtonState(dialog, editText); 426 } 427 }); 428 editText.addTextChangedListener(new TextWatcher() { 429 @Override 430 public void onTextChanged(CharSequence s, int start, int before, int count) { 431 } 432 433 @Override 434 public void beforeTextChanged(CharSequence s, int start, int count, int after) { 435 } 436 437 @Override 438 public void afterTextChanged(Editable s) { 439 updateCustomDialogOkButtonState(dialog, editText); 440 } 441 }); 442 dialog.getWindow().setSoftInputMode( 443 WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE); 444 445 return dialog; 446 } 447 448 /* package */ void updateCustomDialogOkButtonState(AlertDialog dialog, EditText editText) { 449 final Button okButton = dialog.getButton(AlertDialog.BUTTON_POSITIVE); 450 okButton.setEnabled(!TextUtils.isEmpty(editText.getText().toString().trim())); 451 } 452 453 /** 454 * Called after the label has changed (either chosen from the list or entered in the Dialog) 455 */ 456 protected void onLabelRebuilt() { 457 } 458 459 protected void onTypeSelectionChange(int position) { 460 EditType selected = mEditTypeAdapter.getItem(position); 461 // See if the selection has in fact changed 462 if (mEditTypeAdapter.hasCustomSelection() && selected == CUSTOM_SELECTION) { 463 return; 464 } 465 466 if (mType == selected && mType.customColumn == null) { 467 return; 468 } 469 470 if (selected.customColumn != null) { 471 showDialog(DIALOG_ID_CUSTOM); 472 } else { 473 // User picked type, and we're sure it's ok to actually write the entry. 474 mType = selected; 475 mEntry.put(mKind.typeColumn, mType.rawValue); 476 rebuildLabel(); 477 requestFocusForFirstEditField(); 478 onLabelRebuilt(); 479 } 480 } 481 482 /* package */ 483 void showDialog(int bundleDialogId) { 484 Bundle bundle = new Bundle(); 485 bundle.putInt(DIALOG_ID_KEY, bundleDialogId); 486 getDialogManager().showDialogInView(this, bundle); 487 } 488 489 private DialogManager getDialogManager() { 490 if (mDialogManager == null) { 491 Context context = getContext(); 492 if (!(context instanceof DialogManager.DialogShowingViewActivity)) { 493 throw new IllegalStateException( 494 "View must be hosted in an Activity that implements " + 495 "DialogManager.DialogShowingViewActivity"); 496 } 497 mDialogManager = ((DialogManager.DialogShowingViewActivity)context).getDialogManager(); 498 } 499 return mDialogManager; 500 } 501 502 @Override 503 public Dialog createDialog(Bundle bundle) { 504 if (bundle == null) throw new IllegalArgumentException("bundle must not be null"); 505 int dialogId = bundle.getInt(DIALOG_ID_KEY); 506 switch (dialogId) { 507 case DIALOG_ID_CUSTOM: 508 return createCustomDialog(); 509 default: 510 throw new IllegalArgumentException("Invalid dialogId: " + dialogId); 511 } 512 } 513 514 protected abstract void requestFocusForFirstEditField(); 515 516 private class EditTypeAdapter extends ArrayAdapter<EditType> { 517 private final LayoutInflater mInflater; 518 private boolean mHasCustomSelection; 519 private int mTextColor; 520 521 public EditTypeAdapter(Context context) { 522 super(context, 0); 523 mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); 524 mTextColor = context.getResources().getColor(R.color.secondary_text_color); 525 526 if (mType != null && mType.customColumn != null) { 527 528 // Use custom label string when present 529 final String customText = mEntry.getAsString(mType.customColumn); 530 if (customText != null) { 531 add(CUSTOM_SELECTION); 532 mHasCustomSelection = true; 533 } 534 } 535 536 addAll(RawContactModifier.getValidTypes(mState, mKind, mType)); 537 } 538 539 public boolean hasCustomSelection() { 540 return mHasCustomSelection; 541 } 542 543 @Override 544 public View getView(int position, View convertView, ViewGroup parent) { 545 return createViewFromResource( 546 position, convertView, parent, android.R.layout.simple_spinner_item); 547 } 548 549 @Override 550 public View getDropDownView(int position, View convertView, ViewGroup parent) { 551 return createViewFromResource( 552 position, convertView, parent, android.R.layout.simple_spinner_dropdown_item); 553 } 554 555 private View createViewFromResource(int position, View convertView, ViewGroup parent, 556 int resource) { 557 TextView textView; 558 559 if (convertView == null) { 560 textView = (TextView) mInflater.inflate(resource, parent, false); 561 textView.setAllCaps(true); 562 textView.setGravity(Gravity.END | Gravity.CENTER_VERTICAL); 563 textView.setTextAppearance(mContext, android.R.style.TextAppearance_Small); 564 textView.setTextColor(mTextColor); 565 textView.setEllipsize(TruncateAt.MIDDLE); 566 } else { 567 textView = (TextView) convertView; 568 } 569 570 EditType type = getItem(position); 571 String text; 572 if (type == CUSTOM_SELECTION) { 573 text = mEntry.getAsString(mType.customColumn); 574 } else { 575 text = getContext().getString(type.labelRes); 576 } 577 textView.setText(text); 578 return textView; 579 } 580 } 581 } 582