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.Dialog; 20 import android.content.Context; 21 import android.content.res.Resources; 22 import android.os.Bundle; 23 import android.provider.ContactsContract; 24 import android.provider.ContactsContract.CommonDataKinds.Event; 25 import android.text.TextUtils; 26 import android.util.AttributeSet; 27 import android.view.View; 28 import android.widget.Button; 29 30 import com.android.contacts.R; 31 import com.android.contacts.datepicker.DatePicker; 32 import com.android.contacts.datepicker.DatePickerDialog; 33 import com.android.contacts.datepicker.DatePickerDialog.OnDateSetListener; 34 import com.android.contacts.common.model.RawContactDelta; 35 import com.android.contacts.common.model.ValuesDelta; 36 import com.android.contacts.common.model.account.AccountType.EditField; 37 import com.android.contacts.common.model.account.AccountType.EventEditType; 38 import com.android.contacts.common.model.dataitem.DataKind; 39 import com.android.contacts.common.util.CommonDateUtils; 40 import com.android.contacts.common.util.DateUtils; 41 42 import java.text.ParsePosition; 43 import java.util.Calendar; 44 import java.util.Date; 45 import java.util.Locale; 46 47 /** 48 * Editor that allows editing Events using a {@link DatePickerDialog} 49 */ 50 public class EventFieldEditorView extends LabeledEditorView { 51 52 /** 53 * Default string to show when there is no date selected yet. 54 */ 55 private String mNoDateString; 56 private int mPrimaryTextColor; 57 private int mHintTextColor; 58 59 private Button mDateView; 60 61 public EventFieldEditorView(Context context) { 62 super(context); 63 } 64 65 public EventFieldEditorView(Context context, AttributeSet attrs) { 66 super(context, attrs); 67 } 68 69 public EventFieldEditorView(Context context, AttributeSet attrs, int defStyle) { 70 super(context, attrs, defStyle); 71 } 72 73 /** {@inheritDoc} */ 74 @Override 75 protected void onFinishInflate() { 76 super.onFinishInflate(); 77 78 Resources resources = getContext().getResources(); 79 mPrimaryTextColor = resources.getColor(R.color.primary_text_color); 80 mHintTextColor = resources.getColor(R.color.editor_disabled_text_color); 81 mNoDateString = getContext().getString(R.string.event_edit_field_hint_text); 82 83 mDateView = (Button) findViewById(R.id.date_view); 84 mDateView.setOnClickListener(new OnClickListener() { 85 @Override 86 public void onClick(View v) { 87 showDialog(R.id.dialog_event_date_picker); 88 } 89 }); 90 } 91 92 @Override 93 public void editNewlyAddedField() { 94 showDialog(R.id.dialog_event_date_picker); 95 } 96 97 @Override 98 protected void requestFocusForFirstEditField() { 99 mDateView.requestFocus(); 100 } 101 102 @Override 103 public void setEnabled(boolean enabled) { 104 super.setEnabled(enabled); 105 106 mDateView.setEnabled(!isReadOnly() && enabled); 107 } 108 109 @Override 110 public void setValues(DataKind kind, ValuesDelta entry, RawContactDelta state, boolean readOnly, 111 ViewIdGenerator vig) { 112 if (kind.fieldList.size() != 1) throw new IllegalStateException("kind must have 1 field"); 113 super.setValues(kind, entry, state, readOnly, vig); 114 115 mDateView.setEnabled(isEnabled() && !readOnly); 116 117 rebuildDateView(); 118 updateEmptiness(); 119 } 120 121 private void rebuildDateView() { 122 final EditField editField = getKind().fieldList.get(0); 123 final String column = editField.column; 124 String data = DateUtils.formatDate(getContext(), getEntry().getAsString(column), 125 false /*Use the short DateFormat to ensure that it fits inside the EditText*/); 126 if (TextUtils.isEmpty(data)) { 127 mDateView.setText(mNoDateString); 128 mDateView.setTextColor(mHintTextColor); 129 setDeleteButtonVisible(false); 130 } else { 131 mDateView.setText(data); 132 mDateView.setTextColor(mPrimaryTextColor); 133 setDeleteButtonVisible(true); 134 } 135 } 136 137 @Override 138 public boolean isEmpty() { 139 final EditField editField = getKind().fieldList.get(0); 140 final String column = editField.column; 141 return TextUtils.isEmpty(getEntry().getAsString(column)); 142 } 143 144 @Override 145 public Dialog createDialog(Bundle bundle) { 146 if (bundle == null) throw new IllegalArgumentException("bundle must not be null"); 147 int dialogId = bundle.getInt(DIALOG_ID_KEY); 148 switch (dialogId) { 149 case R.id.dialog_event_date_picker: 150 return createDatePickerDialog(); 151 default: 152 return super.createDialog(bundle); 153 } 154 } 155 156 @Override 157 protected EventEditType getType() { 158 return (EventEditType) super.getType(); 159 } 160 161 @Override 162 protected void onLabelRebuilt() { 163 // if we changed to a type that requires a year, ensure that it is actually set 164 final String column = getKind().fieldList.get(0).column; 165 final String oldValue = getEntry().getAsString(column); 166 final DataKind kind = getKind(); 167 168 final Calendar calendar = Calendar.getInstance(DateUtils.UTC_TIMEZONE, Locale.US); 169 final int defaultYear = calendar.get(Calendar.YEAR); 170 171 // Check whether the year is optional 172 final boolean isYearOptional = getType() != null && getType().isYearOptional(); 173 174 if (!isYearOptional && !TextUtils.isEmpty(oldValue)) { 175 final ParsePosition position = new ParsePosition(0); 176 final Date date2 = kind.dateFormatWithoutYear.parse(oldValue, position); 177 178 // Don't understand the date, lets not change it 179 if (date2 == null) return; 180 181 // This value is missing the year. Add it now 182 calendar.setTime(date2); 183 calendar.set(defaultYear, calendar.get(Calendar.MONTH), 184 calendar.get(Calendar.DAY_OF_MONTH), CommonDateUtils.DEFAULT_HOUR, 0, 0); 185 186 onFieldChanged(column, kind.dateFormatWithYear.format(calendar.getTime())); 187 rebuildDateView(); 188 } 189 } 190 191 /** 192 * Prepare dialog for entering a date 193 */ 194 private Dialog createDatePickerDialog() { 195 final String column = getKind().fieldList.get(0).column; 196 final String oldValue = getEntry().getAsString(column); 197 final DataKind kind = getKind(); 198 199 final Calendar calendar = Calendar.getInstance(DateUtils.UTC_TIMEZONE, Locale.US); 200 final int defaultYear = calendar.get(Calendar.YEAR); 201 202 // Check whether the year is optional 203 final boolean isYearOptional = getType().isYearOptional(); 204 205 final int oldYear, oldMonth, oldDay; 206 207 if (TextUtils.isEmpty(oldValue)) { 208 // Default to the current date 209 oldYear = defaultYear; 210 oldMonth = calendar.get(Calendar.MONTH); 211 oldDay = calendar.get(Calendar.DAY_OF_MONTH); 212 } else { 213 // Try parsing with year 214 Calendar cal = DateUtils.parseDate(oldValue, false); 215 if (cal != null) { 216 if (DateUtils.isYearSet(cal)) { 217 oldYear = cal.get(Calendar.YEAR); 218 } else { 219 //cal.set(Calendar.YEAR, 0); 220 oldYear = isYearOptional ? DatePickerDialog.NO_YEAR : defaultYear; 221 } 222 oldMonth = cal.get(Calendar.MONTH); 223 oldDay = cal.get(Calendar.DAY_OF_MONTH); 224 } else { 225 return null; 226 } 227 } 228 final OnDateSetListener callBack = new OnDateSetListener() { 229 @Override 230 public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth) { 231 if (year == 0 && !isYearOptional) throw new IllegalStateException(); 232 final Calendar outCalendar = 233 Calendar.getInstance(DateUtils.UTC_TIMEZONE, Locale.US); 234 235 // If no year specified, set it to 2000 (we could pick any leap year here). 236 // The format string will ignore that year. 237 // For formats other than Exchange, the time of the day is ignored 238 outCalendar.clear(); 239 outCalendar.set(year == DatePickerDialog.NO_YEAR ? 2000 : year, monthOfYear, 240 dayOfMonth, CommonDateUtils.DEFAULT_HOUR, 0, 0); 241 242 final String resultString; 243 if (year == 0) { 244 resultString = kind.dateFormatWithoutYear.format(outCalendar.getTime()); 245 } else { 246 resultString = kind.dateFormatWithYear.format(outCalendar.getTime()); 247 } 248 onFieldChanged(column, resultString); 249 rebuildDateView(); 250 } 251 }; 252 final DatePickerDialog resultDialog = new DatePickerDialog(getContext(), callBack, 253 oldYear, oldMonth, oldDay, isYearOptional); 254 return resultDialog; 255 } 256 257 @Override 258 public void clearAllFields() { 259 // Update UI 260 mDateView.setText(mNoDateString); 261 mDateView.setTextColor(mHintTextColor); 262 263 // Update state 264 final String column = getKind().fieldList.get(0).column; 265 onFieldChanged(column, ""); 266 } 267 268 /** 269 * Sets the typeColumn of entry as TYPE_BIRTHDAY and calls rebuildValues() to refresh the view. 270 */ 271 public void restoreBirthday() { 272 saveValue(getKind().typeColumn, Integer.toString(Event.TYPE_BIRTHDAY)); 273 rebuildValues(); 274 } 275 276 /** 277 * EventEditType Birthday: 278 * rawValue=3 labelRes=17039911 secondary=false specificMax=1 customColumn=null 279 * mYearOptional=true 280 */ 281 public boolean isBirthdayType(){ 282 final EventEditType eventType = getType(); 283 return eventType.rawValue == Event.TYPE_BIRTHDAY && !eventType.secondary 284 && eventType.specificMax == 1 && eventType.customColumn == null 285 && eventType.isYearOptional(); 286 } 287 } 288