Merge "Change high/low bound logic for manual date/time"
This commit is contained in:
@@ -18,15 +18,16 @@ package com.android.settings.datetime;
|
|||||||
|
|
||||||
import static android.app.time.Capabilities.CAPABILITY_POSSESSED;
|
import static android.app.time.Capabilities.CAPABILITY_POSSESSED;
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.app.DatePickerDialog;
|
import android.app.DatePickerDialog;
|
||||||
import android.app.time.TimeCapabilities;
|
import android.app.time.TimeCapabilities;
|
||||||
import android.app.time.TimeManager;
|
import android.app.time.TimeManager;
|
||||||
import android.app.timedetector.ManualTimeSuggestion;
|
import android.app.timedetector.ManualTimeSuggestion;
|
||||||
import android.app.timedetector.TimeDetector;
|
import android.app.timedetector.TimeDetector;
|
||||||
|
import android.app.timedetector.TimeDetectorHelper;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.text.format.DateFormat;
|
import android.text.format.DateFormat;
|
||||||
|
import android.util.Log;
|
||||||
import android.widget.DatePicker;
|
import android.widget.DatePicker;
|
||||||
|
|
||||||
import androidx.annotation.VisibleForTesting;
|
import androidx.annotation.VisibleForTesting;
|
||||||
@@ -47,6 +48,7 @@ public class DatePreferenceController extends AbstractPreferenceController
|
|||||||
|
|
||||||
public static final int DIALOG_DATEPICKER = 0;
|
public static final int DIALOG_DATEPICKER = 0;
|
||||||
|
|
||||||
|
private static final String TAG = "DatePreferenceController";
|
||||||
private static final String KEY_DATE = "date";
|
private static final String KEY_DATE = "date";
|
||||||
|
|
||||||
private final DatePreferenceHost mHost;
|
private final DatePreferenceHost mHost;
|
||||||
@@ -96,22 +98,32 @@ public class DatePreferenceController extends AbstractPreferenceController
|
|||||||
mHost.updateTimeAndDateDisplay(mContext);
|
mHost.updateTimeAndDateDisplay(mContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
public DatePickerDialog buildDatePicker(Activity activity) {
|
/**
|
||||||
|
* Builds a {@link DatePickerDialog} that can be used to request the current date from the user.
|
||||||
|
*/
|
||||||
|
public DatePickerDialog buildDatePicker(
|
||||||
|
Context parentContext, TimeDetectorHelper timeDetectorHelper) {
|
||||||
final Calendar calendar = Calendar.getInstance();
|
final Calendar calendar = Calendar.getInstance();
|
||||||
final DatePickerDialog d = new DatePickerDialog(
|
final DatePickerDialog dialog = new DatePickerDialog(
|
||||||
activity,
|
parentContext,
|
||||||
this,
|
this,
|
||||||
calendar.get(Calendar.YEAR),
|
calendar.get(Calendar.YEAR),
|
||||||
calendar.get(Calendar.MONTH),
|
calendar.get(Calendar.MONTH),
|
||||||
calendar.get(Calendar.DAY_OF_MONTH));
|
calendar.get(Calendar.DAY_OF_MONTH));
|
||||||
// The system clock can't represent dates outside this range.
|
|
||||||
|
// Limit the dates the user can pick to a sensible range.
|
||||||
|
DatePicker datePicker = dialog.getDatePicker();
|
||||||
|
|
||||||
calendar.clear();
|
calendar.clear();
|
||||||
calendar.set(2007, Calendar.JANUARY, 1);
|
int minYear = timeDetectorHelper.getManualDateSelectionYearMin();
|
||||||
d.getDatePicker().setMinDate(calendar.getTimeInMillis());
|
calendar.set(minYear, Calendar.JANUARY, 1);
|
||||||
|
datePicker.setMinDate(calendar.getTimeInMillis());
|
||||||
|
|
||||||
|
int maxYear = timeDetectorHelper.getManualDateSelectionYearMax();
|
||||||
calendar.clear();
|
calendar.clear();
|
||||||
calendar.set(2037, Calendar.DECEMBER, 31);
|
calendar.set(maxYear, Calendar.DECEMBER, 31);
|
||||||
d.getDatePicker().setMaxDate(calendar.getTimeInMillis());
|
datePicker.setMaxDate(calendar.getTimeInMillis());
|
||||||
return d;
|
return dialog;
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
@@ -121,13 +133,16 @@ public class DatePreferenceController extends AbstractPreferenceController
|
|||||||
c.set(Calendar.YEAR, year);
|
c.set(Calendar.YEAR, year);
|
||||||
c.set(Calendar.MONTH, month);
|
c.set(Calendar.MONTH, month);
|
||||||
c.set(Calendar.DAY_OF_MONTH, day);
|
c.set(Calendar.DAY_OF_MONTH, day);
|
||||||
long when = Math.max(c.getTimeInMillis(), DatePreferenceHost.MIN_DATE);
|
long when = c.getTimeInMillis();
|
||||||
|
|
||||||
if (when / 1000 < Integer.MAX_VALUE) {
|
TimeDetector timeDetector = mContext.getSystemService(TimeDetector.class);
|
||||||
TimeDetector timeDetector = mContext.getSystemService(TimeDetector.class);
|
ManualTimeSuggestion manualTimeSuggestion =
|
||||||
ManualTimeSuggestion manualTimeSuggestion =
|
TimeDetector.createManualTimeSuggestion(when, "Settings: Set date");
|
||||||
TimeDetector.createManualTimeSuggestion(when, "Settings: Set date");
|
boolean success = timeDetector.suggestManualTime(manualTimeSuggestion);
|
||||||
timeDetector.suggestManualTime(manualTimeSuggestion);
|
if (!success) {
|
||||||
|
// This implies the system server is applying tighter bounds than the settings app or
|
||||||
|
// the date/time cannot be set for other reasons, e.g. perhaps "auto time" is turned on.
|
||||||
|
Log.w(TAG, "Unable to set date with suggestion=" + manualTimeSuggestion);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -19,6 +19,7 @@ package com.android.settings.datetime;
|
|||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.app.Dialog;
|
import android.app.Dialog;
|
||||||
import android.app.settings.SettingsEnums;
|
import android.app.settings.SettingsEnums;
|
||||||
|
import android.app.timedetector.TimeDetectorHelper;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
|
||||||
@@ -104,7 +105,7 @@ public class DateTimeSettings extends DashboardFragment implements
|
|||||||
switch (id) {
|
switch (id) {
|
||||||
case DatePreferenceController.DIALOG_DATEPICKER:
|
case DatePreferenceController.DIALOG_DATEPICKER:
|
||||||
return use(DatePreferenceController.class)
|
return use(DatePreferenceController.class)
|
||||||
.buildDatePicker(getActivity());
|
.buildDatePicker(getActivity(), TimeDetectorHelper.INSTANCE);
|
||||||
case TimePreferenceController.DIALOG_TIMEPICKER:
|
case TimePreferenceController.DIALOG_TIMEPICKER:
|
||||||
return use(TimePreferenceController.class)
|
return use(TimePreferenceController.class)
|
||||||
.buildTimePicker(getActivity());
|
.buildTimePicker(getActivity());
|
||||||
|
@@ -16,13 +16,13 @@
|
|||||||
|
|
||||||
package com.android.settings.datetime;
|
package com.android.settings.datetime;
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.app.TimePickerDialog;
|
import android.app.TimePickerDialog;
|
||||||
import android.app.timedetector.ManualTimeSuggestion;
|
import android.app.timedetector.ManualTimeSuggestion;
|
||||||
import android.app.timedetector.TimeDetector;
|
import android.app.timedetector.TimeDetector;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.text.format.DateFormat;
|
import android.text.format.DateFormat;
|
||||||
|
import android.util.Log;
|
||||||
import android.widget.TimePicker;
|
import android.widget.TimePicker;
|
||||||
|
|
||||||
import androidx.preference.Preference;
|
import androidx.preference.Preference;
|
||||||
@@ -42,6 +42,7 @@ public class TimePreferenceController extends AbstractPreferenceController
|
|||||||
|
|
||||||
public static final int DIALOG_TIMEPICKER = 1;
|
public static final int DIALOG_TIMEPICKER = 1;
|
||||||
|
|
||||||
|
private static final String TAG = "TimePreferenceController";
|
||||||
private static final String KEY_TIME = "time";
|
private static final String KEY_TIME = "time";
|
||||||
|
|
||||||
private final DatePreferenceController mDatePreferenceController;
|
private final DatePreferenceController mDatePreferenceController;
|
||||||
@@ -99,14 +100,17 @@ public class TimePreferenceController extends AbstractPreferenceController
|
|||||||
// SystemClock time.
|
// SystemClock time.
|
||||||
}
|
}
|
||||||
|
|
||||||
public TimePickerDialog buildTimePicker(Activity activity) {
|
/**
|
||||||
|
* Builds a {@link TimePickerDialog} that can be used to request the current time from the user.
|
||||||
|
*/
|
||||||
|
public TimePickerDialog buildTimePicker(Context parentContext) {
|
||||||
final Calendar calendar = Calendar.getInstance();
|
final Calendar calendar = Calendar.getInstance();
|
||||||
return new TimePickerDialog(
|
return new TimePickerDialog(
|
||||||
activity,
|
parentContext,
|
||||||
this,
|
this,
|
||||||
calendar.get(Calendar.HOUR_OF_DAY),
|
calendar.get(Calendar.HOUR_OF_DAY),
|
||||||
calendar.get(Calendar.MINUTE),
|
calendar.get(Calendar.MINUTE),
|
||||||
DateFormat.is24HourFormat(activity));
|
DateFormat.is24HourFormat(parentContext));
|
||||||
}
|
}
|
||||||
|
|
||||||
void setTime(int hourOfDay, int minute) {
|
void setTime(int hourOfDay, int minute) {
|
||||||
@@ -116,13 +120,16 @@ public class TimePreferenceController extends AbstractPreferenceController
|
|||||||
c.set(Calendar.MINUTE, minute);
|
c.set(Calendar.MINUTE, minute);
|
||||||
c.set(Calendar.SECOND, 0);
|
c.set(Calendar.SECOND, 0);
|
||||||
c.set(Calendar.MILLISECOND, 0);
|
c.set(Calendar.MILLISECOND, 0);
|
||||||
long when = Math.max(c.getTimeInMillis(), TimePreferenceHost.MIN_DATE);
|
long when = c.getTimeInMillis();
|
||||||
|
|
||||||
if (when / 1000 < Integer.MAX_VALUE) {
|
TimeDetector timeDetector = mContext.getSystemService(TimeDetector.class);
|
||||||
TimeDetector timeDetector = mContext.getSystemService(TimeDetector.class);
|
ManualTimeSuggestion manualTimeSuggestion =
|
||||||
ManualTimeSuggestion manualTimeSuggestion =
|
TimeDetector.createManualTimeSuggestion(when, "Settings: Set time");
|
||||||
TimeDetector.createManualTimeSuggestion(when, "Settings: Set time");
|
boolean success = timeDetector.suggestManualTime(manualTimeSuggestion);
|
||||||
timeDetector.suggestManualTime(manualTimeSuggestion);
|
if (!success) {
|
||||||
|
// This implies the system server is applying tighter bounds than the settings app or
|
||||||
|
// the date/time cannot be set for other reasons, e.g. perhaps "auto time" is turned on.
|
||||||
|
Log.w(TAG, "Unable to set time with suggestion=" + manualTimeSuggestion);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -19,8 +19,5 @@ package com.android.settings.datetime;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
|
||||||
public interface UpdateTimeAndDateCallback {
|
public interface UpdateTimeAndDateCallback {
|
||||||
// Minimum time is Nov 5, 2007, 0:00.
|
|
||||||
long MIN_DATE = 1194220800000L;
|
|
||||||
|
|
||||||
void updateTimeAndDateDisplay(Context context);
|
void updateTimeAndDateDisplay(Context context);
|
||||||
}
|
}
|
||||||
|
@@ -18,15 +18,19 @@ package com.android.settings.datetime;
|
|||||||
|
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import android.app.DatePickerDialog;
|
||||||
import android.app.time.Capabilities;
|
import android.app.time.Capabilities;
|
||||||
import android.app.time.TimeCapabilities;
|
import android.app.time.TimeCapabilities;
|
||||||
import android.app.time.TimeCapabilitiesAndConfig;
|
import android.app.time.TimeCapabilitiesAndConfig;
|
||||||
import android.app.time.TimeConfiguration;
|
import android.app.time.TimeConfiguration;
|
||||||
import android.app.time.TimeManager;
|
import android.app.time.TimeManager;
|
||||||
import android.app.timedetector.TimeDetector;
|
import android.app.timedetector.TimeDetector;
|
||||||
|
import android.app.timedetector.TimeDetectorHelper;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.UserHandle;
|
import android.os.UserHandle;
|
||||||
|
|
||||||
@@ -40,17 +44,20 @@ import org.mockito.MockitoAnnotations;
|
|||||||
import org.robolectric.RobolectricTestRunner;
|
import org.robolectric.RobolectricTestRunner;
|
||||||
import org.robolectric.RuntimeEnvironment;
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.GregorianCalendar;
|
||||||
|
|
||||||
@RunWith(RobolectricTestRunner.class)
|
@RunWith(RobolectricTestRunner.class)
|
||||||
public class DatePreferenceControllerTest {
|
public class DatePreferenceControllerTest {
|
||||||
|
|
||||||
@Mock
|
@Mock
|
||||||
private Context mContext;
|
private Context mContext;
|
||||||
@Mock
|
@Mock
|
||||||
private TimeDetector mTimeDetector;
|
private DatePreferenceController.DatePreferenceHost mHost;
|
||||||
@Mock
|
@Mock
|
||||||
private TimeManager mTimeManager;
|
private TimeManager mTimeManager;
|
||||||
@Mock
|
@Mock
|
||||||
private DatePreferenceController.DatePreferenceHost mHost;
|
private TimeDetector mTimeDetector;
|
||||||
|
|
||||||
private RestrictedPreference mPreference;
|
private RestrictedPreference mPreference;
|
||||||
private DatePreferenceController mController;
|
private DatePreferenceController mController;
|
||||||
@@ -114,6 +121,26 @@ public class DatePreferenceControllerTest {
|
|||||||
verify(mHost).showDatePicker();
|
verify(mHost).showDatePicker();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBuildDatePicker() {
|
||||||
|
TimeDetectorHelper timeDetectorHelper = mock(TimeDetectorHelper.class);
|
||||||
|
when(timeDetectorHelper.getManualDateSelectionYearMin()).thenReturn(2015);
|
||||||
|
when(timeDetectorHelper.getManualDateSelectionYearMax()).thenReturn(2020);
|
||||||
|
|
||||||
|
Context context = RuntimeEnvironment.application;
|
||||||
|
DatePickerDialog dialog = mController.buildDatePicker(context, timeDetectorHelper);
|
||||||
|
|
||||||
|
GregorianCalendar calendar = new GregorianCalendar();
|
||||||
|
|
||||||
|
long minDate = dialog.getDatePicker().getMinDate();
|
||||||
|
calendar.setTimeInMillis(minDate);
|
||||||
|
assertEquals(2015, calendar.get(Calendar.YEAR));
|
||||||
|
|
||||||
|
long maxDate = dialog.getDatePicker().getMaxDate();
|
||||||
|
calendar.setTimeInMillis(maxDate);
|
||||||
|
assertEquals(2020, calendar.get(Calendar.YEAR));
|
||||||
|
}
|
||||||
|
|
||||||
private static TimeCapabilitiesAndConfig createCapabilitiesAndConfig(
|
private static TimeCapabilitiesAndConfig createCapabilitiesAndConfig(
|
||||||
boolean suggestManualAllowed) {
|
boolean suggestManualAllowed) {
|
||||||
int suggestManualCapability = suggestManualAllowed ? Capabilities.CAPABILITY_POSSESSED
|
int suggestManualCapability = suggestManualAllowed ? Capabilities.CAPABILITY_POSSESSED
|
||||||
|
Reference in New Issue
Block a user