• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2007 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 android.widget;
18 
19 import android.annotation.Widget;
20 import android.content.Context;
21 import android.content.res.Configuration;
22 import android.content.res.TypedArray;
23 import android.os.Parcel;
24 import android.os.Parcelable;
25 import android.text.format.DateUtils;
26 import android.util.AttributeSet;
27 import android.view.LayoutInflater;
28 import android.view.View;
29 import android.view.accessibility.AccessibilityEvent;
30 import android.view.accessibility.AccessibilityNodeInfo;
31 import android.view.inputmethod.EditorInfo;
32 import android.view.inputmethod.InputMethodManager;
33 import android.widget.NumberPicker.OnValueChangeListener;
34 
35 import com.android.internal.R;
36 
37 import java.text.DateFormatSymbols;
38 import java.util.Calendar;
39 import java.util.Locale;
40 
41 /**
42  * A view for selecting the time of day, in either 24 hour or AM/PM mode. The
43  * hour, each minute digit, and AM/PM (if applicable) can be conrolled by
44  * vertical spinners. The hour can be entered by keyboard input. Entering in two
45  * digit hours can be accomplished by hitting two digits within a timeout of
46  * about a second (e.g. '1' then '2' to select 12). The minutes can be entered
47  * by entering single digits. Under AM/PM mode, the user can hit 'a', 'A", 'p'
48  * or 'P' to pick. For a dialog using this view, see
49  * {@link android.app.TimePickerDialog}.
50  *<p>
51  * See the <a href="{@docRoot}guide/topics/ui/controls/pickers.html">Pickers</a>
52  * guide.
53  * </p>
54  */
55 @Widget
56 public class TimePicker extends FrameLayout {
57 
58     private static final boolean DEFAULT_ENABLED_STATE = true;
59 
60     private static final int HOURS_IN_HALF_DAY = 12;
61 
62     /**
63      * A no-op callback used in the constructor to avoid null checks later in
64      * the code.
65      */
66     private static final OnTimeChangedListener NO_OP_CHANGE_LISTENER = new OnTimeChangedListener() {
67         public void onTimeChanged(TimePicker view, int hourOfDay, int minute) {
68         }
69     };
70 
71     // state
72     private boolean mIs24HourView;
73 
74     private boolean mIsAm;
75 
76     // ui components
77     private final NumberPicker mHourSpinner;
78 
79     private final NumberPicker mMinuteSpinner;
80 
81     private final NumberPicker mAmPmSpinner;
82 
83     private final EditText mHourSpinnerInput;
84 
85     private final EditText mMinuteSpinnerInput;
86 
87     private final EditText mAmPmSpinnerInput;
88 
89     private final TextView mDivider;
90 
91     // Note that the legacy implementation of the TimePicker is
92     // using a button for toggling between AM/PM while the new
93     // version uses a NumberPicker spinner. Therefore the code
94     // accommodates these two cases to be backwards compatible.
95     private final Button mAmPmButton;
96 
97     private final String[] mAmPmStrings;
98 
99     private boolean mIsEnabled = DEFAULT_ENABLED_STATE;
100 
101     // callbacks
102     private OnTimeChangedListener mOnTimeChangedListener;
103 
104     private Calendar mTempCalendar;
105 
106     private Locale mCurrentLocale;
107 
108     /**
109      * The callback interface used to indicate the time has been adjusted.
110      */
111     public interface OnTimeChangedListener {
112 
113         /**
114          * @param view The view associated with this listener.
115          * @param hourOfDay The current hour.
116          * @param minute The current minute.
117          */
onTimeChanged(TimePicker view, int hourOfDay, int minute)118         void onTimeChanged(TimePicker view, int hourOfDay, int minute);
119     }
120 
TimePicker(Context context)121     public TimePicker(Context context) {
122         this(context, null);
123     }
124 
TimePicker(Context context, AttributeSet attrs)125     public TimePicker(Context context, AttributeSet attrs) {
126         this(context, attrs, R.attr.timePickerStyle);
127     }
128 
TimePicker(Context context, AttributeSet attrs, int defStyle)129     public TimePicker(Context context, AttributeSet attrs, int defStyle) {
130         super(context, attrs, defStyle);
131 
132         // initialization based on locale
133         setCurrentLocale(Locale.getDefault());
134 
135         // process style attributes
136         TypedArray attributesArray = context.obtainStyledAttributes(
137                 attrs, R.styleable.TimePicker, defStyle, 0);
138         int layoutResourceId = attributesArray.getResourceId(
139                 R.styleable.TimePicker_internalLayout, R.layout.time_picker);
140         attributesArray.recycle();
141 
142         LayoutInflater inflater = (LayoutInflater) context.getSystemService(
143                 Context.LAYOUT_INFLATER_SERVICE);
144         inflater.inflate(layoutResourceId, this, true);
145 
146         // hour
147         mHourSpinner = (NumberPicker) findViewById(R.id.hour);
148         mHourSpinner.setOnValueChangedListener(new NumberPicker.OnValueChangeListener() {
149             public void onValueChange(NumberPicker spinner, int oldVal, int newVal) {
150                 updateInputState();
151                 if (!is24HourView()) {
152                     if ((oldVal == HOURS_IN_HALF_DAY - 1 && newVal == HOURS_IN_HALF_DAY)
153                             || (oldVal == HOURS_IN_HALF_DAY && newVal == HOURS_IN_HALF_DAY - 1)) {
154                         mIsAm = !mIsAm;
155                         updateAmPmControl();
156                     }
157                 }
158                 onTimeChanged();
159             }
160         });
161         mHourSpinnerInput = (EditText) mHourSpinner.findViewById(R.id.numberpicker_input);
162         mHourSpinnerInput.setImeOptions(EditorInfo.IME_ACTION_NEXT);
163 
164         // divider (only for the new widget style)
165         mDivider = (TextView) findViewById(R.id.divider);
166         if (mDivider != null) {
167             mDivider.setText(R.string.time_picker_separator);
168         }
169 
170         // minute
171         mMinuteSpinner = (NumberPicker) findViewById(R.id.minute);
172         mMinuteSpinner.setMinValue(0);
173         mMinuteSpinner.setMaxValue(59);
174         mMinuteSpinner.setOnLongPressUpdateInterval(100);
175         mMinuteSpinner.setFormatter(NumberPicker.getTwoDigitFormatter());
176         mMinuteSpinner.setOnValueChangedListener(new NumberPicker.OnValueChangeListener() {
177             public void onValueChange(NumberPicker spinner, int oldVal, int newVal) {
178                 updateInputState();
179                 int minValue = mMinuteSpinner.getMinValue();
180                 int maxValue = mMinuteSpinner.getMaxValue();
181                 if (oldVal == maxValue && newVal == minValue) {
182                     int newHour = mHourSpinner.getValue() + 1;
183                     if (!is24HourView() && newHour == HOURS_IN_HALF_DAY) {
184                         mIsAm = !mIsAm;
185                         updateAmPmControl();
186                     }
187                     mHourSpinner.setValue(newHour);
188                 } else if (oldVal == minValue && newVal == maxValue) {
189                     int newHour = mHourSpinner.getValue() - 1;
190                     if (!is24HourView() && newHour == HOURS_IN_HALF_DAY - 1) {
191                         mIsAm = !mIsAm;
192                         updateAmPmControl();
193                     }
194                     mHourSpinner.setValue(newHour);
195                 }
196                 onTimeChanged();
197             }
198         });
199         mMinuteSpinnerInput = (EditText) mMinuteSpinner.findViewById(R.id.numberpicker_input);
200         mMinuteSpinnerInput.setImeOptions(EditorInfo.IME_ACTION_NEXT);
201 
202         /* Get the localized am/pm strings and use them in the spinner */
203         mAmPmStrings = new DateFormatSymbols().getAmPmStrings();
204 
205         // am/pm
206         View amPmView = findViewById(R.id.amPm);
207         if (amPmView instanceof Button) {
208             mAmPmSpinner = null;
209             mAmPmSpinnerInput = null;
210             mAmPmButton = (Button) amPmView;
211             mAmPmButton.setOnClickListener(new OnClickListener() {
212                 public void onClick(View button) {
213                     button.requestFocus();
214                     mIsAm = !mIsAm;
215                     updateAmPmControl();
216                     onTimeChanged();
217                 }
218             });
219         } else {
220             mAmPmButton = null;
221             mAmPmSpinner = (NumberPicker) amPmView;
222             mAmPmSpinner.setMinValue(0);
223             mAmPmSpinner.setMaxValue(1);
224             mAmPmSpinner.setDisplayedValues(mAmPmStrings);
225             mAmPmSpinner.setOnValueChangedListener(new OnValueChangeListener() {
226                 public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
227                     updateInputState();
228                     picker.requestFocus();
229                     mIsAm = !mIsAm;
230                     updateAmPmControl();
231                     onTimeChanged();
232                 }
233             });
234             mAmPmSpinnerInput = (EditText) mAmPmSpinner.findViewById(R.id.numberpicker_input);
235             mAmPmSpinnerInput.setImeOptions(EditorInfo.IME_ACTION_DONE);
236         }
237 
238         // update controls to initial state
239         updateHourControl();
240         updateAmPmControl();
241 
242         setOnTimeChangedListener(NO_OP_CHANGE_LISTENER);
243 
244         // set to current time
245         setCurrentHour(mTempCalendar.get(Calendar.HOUR_OF_DAY));
246         setCurrentMinute(mTempCalendar.get(Calendar.MINUTE));
247 
248         if (!isEnabled()) {
249             setEnabled(false);
250         }
251 
252         // set the content descriptions
253         setContentDescriptions();
254 
255         // If not explicitly specified this view is important for accessibility.
256         if (getImportantForAccessibility() == IMPORTANT_FOR_ACCESSIBILITY_AUTO) {
257             setImportantForAccessibility(IMPORTANT_FOR_ACCESSIBILITY_YES);
258         }
259     }
260 
261     @Override
setEnabled(boolean enabled)262     public void setEnabled(boolean enabled) {
263         if (mIsEnabled == enabled) {
264             return;
265         }
266         super.setEnabled(enabled);
267         mMinuteSpinner.setEnabled(enabled);
268         if (mDivider != null) {
269             mDivider.setEnabled(enabled);
270         }
271         mHourSpinner.setEnabled(enabled);
272         if (mAmPmSpinner != null) {
273             mAmPmSpinner.setEnabled(enabled);
274         } else {
275             mAmPmButton.setEnabled(enabled);
276         }
277         mIsEnabled = enabled;
278     }
279 
280     @Override
isEnabled()281     public boolean isEnabled() {
282         return mIsEnabled;
283     }
284 
285     @Override
onConfigurationChanged(Configuration newConfig)286     protected void onConfigurationChanged(Configuration newConfig) {
287         super.onConfigurationChanged(newConfig);
288         setCurrentLocale(newConfig.locale);
289     }
290 
291     /**
292      * Sets the current locale.
293      *
294      * @param locale The current locale.
295      */
setCurrentLocale(Locale locale)296     private void setCurrentLocale(Locale locale) {
297         if (locale.equals(mCurrentLocale)) {
298             return;
299         }
300         mCurrentLocale = locale;
301         mTempCalendar = Calendar.getInstance(locale);
302     }
303 
304     /**
305      * Used to save / restore state of time picker
306      */
307     private static class SavedState extends BaseSavedState {
308 
309         private final int mHour;
310 
311         private final int mMinute;
312 
SavedState(Parcelable superState, int hour, int minute)313         private SavedState(Parcelable superState, int hour, int minute) {
314             super(superState);
315             mHour = hour;
316             mMinute = minute;
317         }
318 
SavedState(Parcel in)319         private SavedState(Parcel in) {
320             super(in);
321             mHour = in.readInt();
322             mMinute = in.readInt();
323         }
324 
getHour()325         public int getHour() {
326             return mHour;
327         }
328 
getMinute()329         public int getMinute() {
330             return mMinute;
331         }
332 
333         @Override
writeToParcel(Parcel dest, int flags)334         public void writeToParcel(Parcel dest, int flags) {
335             super.writeToParcel(dest, flags);
336             dest.writeInt(mHour);
337             dest.writeInt(mMinute);
338         }
339 
340         @SuppressWarnings({"unused", "hiding"})
341         public static final Parcelable.Creator<SavedState> CREATOR = new Creator<SavedState>() {
342             public SavedState createFromParcel(Parcel in) {
343                 return new SavedState(in);
344             }
345 
346             public SavedState[] newArray(int size) {
347                 return new SavedState[size];
348             }
349         };
350     }
351 
352     @Override
onSaveInstanceState()353     protected Parcelable onSaveInstanceState() {
354         Parcelable superState = super.onSaveInstanceState();
355         return new SavedState(superState, getCurrentHour(), getCurrentMinute());
356     }
357 
358     @Override
onRestoreInstanceState(Parcelable state)359     protected void onRestoreInstanceState(Parcelable state) {
360         SavedState ss = (SavedState) state;
361         super.onRestoreInstanceState(ss.getSuperState());
362         setCurrentHour(ss.getHour());
363         setCurrentMinute(ss.getMinute());
364     }
365 
366     /**
367      * Set the callback that indicates the time has been adjusted by the user.
368      *
369      * @param onTimeChangedListener the callback, should not be null.
370      */
setOnTimeChangedListener(OnTimeChangedListener onTimeChangedListener)371     public void setOnTimeChangedListener(OnTimeChangedListener onTimeChangedListener) {
372         mOnTimeChangedListener = onTimeChangedListener;
373     }
374 
375     /**
376      * @return The current hour in the range (0-23).
377      */
getCurrentHour()378     public Integer getCurrentHour() {
379         int currentHour = mHourSpinner.getValue();
380         if (is24HourView()) {
381             return currentHour;
382         } else if (mIsAm) {
383             return currentHour % HOURS_IN_HALF_DAY;
384         } else {
385             return (currentHour % HOURS_IN_HALF_DAY) + HOURS_IN_HALF_DAY;
386         }
387     }
388 
389     /**
390      * Set the current hour.
391      */
setCurrentHour(Integer currentHour)392     public void setCurrentHour(Integer currentHour) {
393         // why was Integer used in the first place?
394         if (currentHour == null || currentHour == getCurrentHour()) {
395             return;
396         }
397         if (!is24HourView()) {
398             // convert [0,23] ordinal to wall clock display
399             if (currentHour >= HOURS_IN_HALF_DAY) {
400                 mIsAm = false;
401                 if (currentHour > HOURS_IN_HALF_DAY) {
402                     currentHour = currentHour - HOURS_IN_HALF_DAY;
403                 }
404             } else {
405                 mIsAm = true;
406                 if (currentHour == 0) {
407                     currentHour = HOURS_IN_HALF_DAY;
408                 }
409             }
410             updateAmPmControl();
411         }
412         mHourSpinner.setValue(currentHour);
413         onTimeChanged();
414     }
415 
416     /**
417      * Set whether in 24 hour or AM/PM mode.
418      *
419      * @param is24HourView True = 24 hour mode. False = AM/PM.
420      */
setIs24HourView(Boolean is24HourView)421     public void setIs24HourView(Boolean is24HourView) {
422         if (mIs24HourView == is24HourView) {
423             return;
424         }
425         mIs24HourView = is24HourView;
426         // cache the current hour since spinner range changes
427         int currentHour = getCurrentHour();
428         updateHourControl();
429         // set value after spinner range is updated
430         setCurrentHour(currentHour);
431         updateAmPmControl();
432     }
433 
434     /**
435      * @return true if this is in 24 hour view else false.
436      */
is24HourView()437     public boolean is24HourView() {
438         return mIs24HourView;
439     }
440 
441     /**
442      * @return The current minute.
443      */
getCurrentMinute()444     public Integer getCurrentMinute() {
445         return mMinuteSpinner.getValue();
446     }
447 
448     /**
449      * Set the current minute (0-59).
450      */
setCurrentMinute(Integer currentMinute)451     public void setCurrentMinute(Integer currentMinute) {
452         if (currentMinute == getCurrentMinute()) {
453             return;
454         }
455         mMinuteSpinner.setValue(currentMinute);
456         onTimeChanged();
457     }
458 
459     @Override
getBaseline()460     public int getBaseline() {
461         return mHourSpinner.getBaseline();
462     }
463 
464     @Override
dispatchPopulateAccessibilityEvent(AccessibilityEvent event)465     public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) {
466         onPopulateAccessibilityEvent(event);
467         return true;
468     }
469 
470     @Override
onPopulateAccessibilityEvent(AccessibilityEvent event)471     public void onPopulateAccessibilityEvent(AccessibilityEvent event) {
472         super.onPopulateAccessibilityEvent(event);
473 
474         int flags = DateUtils.FORMAT_SHOW_TIME;
475         if (mIs24HourView) {
476             flags |= DateUtils.FORMAT_24HOUR;
477         } else {
478             flags |= DateUtils.FORMAT_12HOUR;
479         }
480         mTempCalendar.set(Calendar.HOUR_OF_DAY, getCurrentHour());
481         mTempCalendar.set(Calendar.MINUTE, getCurrentMinute());
482         String selectedDateUtterance = DateUtils.formatDateTime(mContext,
483                 mTempCalendar.getTimeInMillis(), flags);
484         event.getText().add(selectedDateUtterance);
485     }
486 
487     @Override
onInitializeAccessibilityEvent(AccessibilityEvent event)488     public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
489         super.onInitializeAccessibilityEvent(event);
490         event.setClassName(TimePicker.class.getName());
491     }
492 
493     @Override
onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info)494     public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
495         super.onInitializeAccessibilityNodeInfo(info);
496         info.setClassName(TimePicker.class.getName());
497     }
498 
updateHourControl()499     private void updateHourControl() {
500         if (is24HourView()) {
501             mHourSpinner.setMinValue(0);
502             mHourSpinner.setMaxValue(23);
503             mHourSpinner.setFormatter(NumberPicker.getTwoDigitFormatter());
504         } else {
505             mHourSpinner.setMinValue(1);
506             mHourSpinner.setMaxValue(12);
507             mHourSpinner.setFormatter(null);
508         }
509     }
510 
updateAmPmControl()511     private void updateAmPmControl() {
512         if (is24HourView()) {
513             if (mAmPmSpinner != null) {
514                 mAmPmSpinner.setVisibility(View.GONE);
515             } else {
516                 mAmPmButton.setVisibility(View.GONE);
517             }
518         } else {
519             int index = mIsAm ? Calendar.AM : Calendar.PM;
520             if (mAmPmSpinner != null) {
521                 mAmPmSpinner.setValue(index);
522                 mAmPmSpinner.setVisibility(View.VISIBLE);
523             } else {
524                 mAmPmButton.setText(mAmPmStrings[index]);
525                 mAmPmButton.setVisibility(View.VISIBLE);
526             }
527         }
528         sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_SELECTED);
529     }
530 
onTimeChanged()531     private void onTimeChanged() {
532         sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_SELECTED);
533         if (mOnTimeChangedListener != null) {
534             mOnTimeChangedListener.onTimeChanged(this, getCurrentHour(), getCurrentMinute());
535         }
536     }
537 
setContentDescriptions()538     private void setContentDescriptions() {
539         // Minute
540         trySetContentDescription(mMinuteSpinner, R.id.increment,
541                 R.string.time_picker_increment_minute_button);
542         trySetContentDescription(mMinuteSpinner, R.id.decrement,
543                 R.string.time_picker_decrement_minute_button);
544         // Hour
545         trySetContentDescription(mHourSpinner, R.id.increment,
546                 R.string.time_picker_increment_hour_button);
547         trySetContentDescription(mHourSpinner, R.id.decrement,
548                 R.string.time_picker_decrement_hour_button);
549         // AM/PM
550         if (mAmPmSpinner != null) {
551             trySetContentDescription(mAmPmSpinner, R.id.increment,
552                     R.string.time_picker_increment_set_pm_button);
553             trySetContentDescription(mAmPmSpinner, R.id.decrement,
554                     R.string.time_picker_decrement_set_am_button);
555         }
556     }
557 
trySetContentDescription(View root, int viewId, int contDescResId)558     private void trySetContentDescription(View root, int viewId, int contDescResId) {
559         View target = root.findViewById(viewId);
560         if (target != null) {
561             target.setContentDescription(mContext.getString(contDescResId));
562         }
563     }
564 
updateInputState()565     private void updateInputState() {
566         // Make sure that if the user changes the value and the IME is active
567         // for one of the inputs if this widget, the IME is closed. If the user
568         // changed the value via the IME and there is a next input the IME will
569         // be shown, otherwise the user chose another means of changing the
570         // value and having the IME up makes no sense.
571         InputMethodManager inputMethodManager = InputMethodManager.peekInstance();
572         if (inputMethodManager != null) {
573             if (inputMethodManager.isActive(mHourSpinnerInput)) {
574                 mHourSpinnerInput.clearFocus();
575                 inputMethodManager.hideSoftInputFromWindow(getWindowToken(), 0);
576             } else if (inputMethodManager.isActive(mMinuteSpinnerInput)) {
577                 mMinuteSpinnerInput.clearFocus();
578                 inputMethodManager.hideSoftInputFromWindow(getWindowToken(), 0);
579             } else if (inputMethodManager.isActive(mAmPmSpinnerInput)) {
580                 mAmPmSpinnerInput.clearFocus();
581                 inputMethodManager.hideSoftInputFromWindow(getWindowToken(), 0);
582             }
583         }
584     }
585 }
586