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.text.TextUtils; 24 import android.util.AttributeSet; 25 import android.view.View; 26 import android.widget.Button; 27 28 import com.android.contacts.R; 29 import com.android.contacts.datepicker.DatePicker; 30 import com.android.contacts.datepicker.DatePickerDialog; 31 import com.android.contacts.datepicker.DatePickerDialog.OnDateSetListener; 32 import com.android.contacts.model.RawContactDelta; 33 import com.android.contacts.common.model.ValuesDelta; 34 import com.android.contacts.common.model.account.AccountType.EditField; 35 import com.android.contacts.common.model.account.AccountType.EventEditType; 36 import com.android.contacts.common.model.dataitem.DataKind; 37 import com.android.contacts.util.DateUtils; 38 39 import java.text.ParsePosition; 40 import java.util.Calendar; 41 import java.util.Date; 42 import java.util.Locale; 43 44 /** 45 * Editor that allows editing Events using a {@link DatePickerDialog} 46 */ 47 public class EventFieldEditorView extends LabeledEditorView { 48 /** 49 * Exchange requires 8:00 for birthdays 50 */ 51 private final static int DEFAULT_HOUR = 8; 52 53 /** 54 * Default string to show when there is no date selected yet. 55 */ 56 private String mNoDateString; 57 private int mPrimaryTextColor; 58 private int mSecondaryTextColor; 59 60 private Button mDateView; 61 62 public EventFieldEditorView(Context context) { 63 super(context); 64 } 65 66 public EventFieldEditorView(Context context, AttributeSet attrs) { 67 super(context, attrs); 68 } 69 70 public EventFieldEditorView(Context context, AttributeSet attrs, int defStyle) { 71 super(context, attrs, defStyle); 72 } 73 74 /** {@inheritDoc} */ 75 @Override 76 protected void onFinishInflate() { 77 super.onFinishInflate(); 78 79 Resources resources = mContext.getResources(); 80 mPrimaryTextColor = resources.getColor(R.color.primary_text_color); 81 mSecondaryTextColor = resources.getColor(R.color.secondary_text_color); 82 mNoDateString = mContext.getString(R.string.event_edit_field_hint_text); 83 84 mDateView = (Button) findViewById(R.id.date_view); 85 mDateView.setOnClickListener(new OnClickListener() { 86 @Override 87 public void onClick(View v) { 88 showDialog(R.id.dialog_event_date_picker); 89 } 90 }); 91 } 92 93 @Override 94 public void editNewlyAddedField() { 95 showDialog(R.id.dialog_event_date_picker); 96 } 97 98 @Override 99 protected void requestFocusForFirstEditField() { 100 mDateView.requestFocus(); 101 } 102 103 @Override 104 public void setEnabled(boolean enabled) { 105 super.setEnabled(enabled); 106 107 mDateView.setEnabled(!isReadOnly() && enabled); 108 } 109 110 @Override 111 public void setValues(DataKind kind, ValuesDelta entry, RawContactDelta state, boolean readOnly, 112 ViewIdGenerator vig) { 113 if (kind.fieldList.size() != 1) throw new IllegalStateException("kind must have 1 field"); 114 super.setValues(kind, entry, state, readOnly, vig); 115 116 mDateView.setEnabled(isEnabled() && !readOnly); 117 118 rebuildDateView(); 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(mSecondaryTextColor); 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().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), 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, 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 /** 258 * @return Default hour which should be used for birthday field. 259 */ 260 public static int getDefaultHourForBirthday() { 261 return DEFAULT_HOUR; 262 } 263 264 @Override 265 public void clearAllFields() { 266 // Update UI 267 mDateView.setText(mNoDateString); 268 mDateView.setTextColor(mSecondaryTextColor); 269 270 // Update state 271 final String column = getKind().fieldList.get(0).column; 272 onFieldChanged(column, ""); 273 } 274 } 275