import { TimePicker as TimePickerDefinition } from '.'; import { View, CSSType } from '../core/view'; import { Property } from '../core/properties'; interface Time { hour: number; minute: number; } const dateComparer = (x: Date, y: Date): boolean => x <= y && x >= y; export function getValidTime(picker: TimePickerDefinition, hour: number, minute: number): Time { if (picker.minuteInterval > 1) { const minuteFloor = minute - (minute % picker.minuteInterval); minute = minuteFloor + (minute === minuteFloor + 1 ? picker.minuteInterval : 0); if (minute === 60) { hour++; minute = 0; } } let time = { hour: hour, minute: minute }; if (!isLessThanMaxTime(picker, hour, minute)) { time = { hour: picker.maxHour, minute: picker.maxMinute }; } if (!isGreaterThanMinTime(picker, hour, minute)) { time = { hour: picker.minHour, minute: picker.minMinute }; } return time; } function isValidTime(picker: TimePickerDefinition): boolean { return isGreaterThanMinTime(picker) && isLessThanMaxTime(picker); } function isHourValid(value: number): boolean { return typeof value === 'number' && value >= 0 && value <= 23; } function isMinuteValid(value: number): boolean { return typeof value === 'number' && value >= 0 && value <= 59; } function isMinuteIntervalValid(value: number): boolean { return typeof value === 'number' && value >= 1 && value <= 30 && 60 % value === 0; } function getMinutes(hour: number): number { return hour * 60; } export function isDefined(value: any): boolean { return value !== undefined; } function isGreaterThanMinTime(picker: TimePickerDefinition, hour?: number, minute?: number): boolean { if (picker.minHour === undefined || picker.minMinute === undefined) { return true; } return getMinutes(hour !== undefined ? hour : picker.hour) + (minute !== undefined ? minute : picker.minute) >= getMinutes(picker.minHour) + picker.minMinute; } function isLessThanMaxTime(picker: TimePickerDefinition, hour?: number, minute?: number): boolean { if (!isDefined(picker.maxHour) || !isDefined(picker.maxMinute)) { return true; } return getMinutes(isDefined(hour) ? hour : picker.hour) + (isDefined(minute) ? minute : picker.minute) <= getMinutes(picker.maxHour) + picker.maxMinute; } function toString(value: number | Date): string { if (value instanceof Date) { return value + ''; } return value < 10 ? `0${value}` : `${value}`; } function getMinMaxTimeErrorMessage(picker: TimePickerDefinition): string { return `Min time: (${toString(picker.minHour)}:${toString(picker.minMinute)}), max time: (${toString(picker.maxHour)}:${toString(picker.maxMinute)})`; } function getErrorMessage(picker: TimePickerDefinition, propertyName: string, newValue: number | Date): string { return `${propertyName} property value (${toString(newValue)}:${toString(picker.minute)}) is not valid. ${getMinMaxTimeErrorMessage(picker)}.`; } @CSSType('TimePicker') export abstract class TimePickerBase extends View implements TimePickerDefinition { public static timeChangeEvent: string = 'timeChange'; public hour: number; public minute: number; public time: Date; public minuteInterval: number; public minHour: number; public maxHour: number; public minMinute: number; public maxMinute: number; public iosPreferredDatePickerStyle: number; } TimePickerBase.prototype.recycleNativeView = 'auto'; export const minHourProperty = new Property({ name: 'minHour', defaultValue: 0, valueChanged: (picker, oldValue, newValue) => { if (!isHourValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'minHour', newValue)); } }, valueConverter: (v) => parseInt(v), }); minHourProperty.register(TimePickerBase); export const maxHourProperty = new Property({ name: 'maxHour', defaultValue: 23, valueChanged: (picker, oldValue, newValue) => { if (!isHourValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'maxHour', newValue)); } }, valueConverter: (v) => parseInt(v), }); maxHourProperty.register(TimePickerBase); export const minMinuteProperty = new Property({ name: 'minMinute', defaultValue: 0, valueChanged: (picker, oldValue, newValue) => { if (!isMinuteValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'minMinute', newValue)); } }, valueConverter: (v) => parseInt(v), }); minMinuteProperty.register(TimePickerBase); export const maxMinuteProperty = new Property({ name: 'maxMinute', defaultValue: 59, valueChanged: (picker, oldValue, newValue) => { if (!isMinuteValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'maxMinute', newValue)); } }, valueConverter: (v) => parseInt(v), }); maxMinuteProperty.register(TimePickerBase); export const minuteIntervalProperty = new Property({ name: 'minuteInterval', defaultValue: 1, valueChanged: (picker, oldValue, newValue) => { if (!isMinuteIntervalValid(newValue)) { throw new Error(getErrorMessage(picker, 'minuteInterval', newValue)); } }, valueConverter: (v) => parseInt(v), }); minuteIntervalProperty.register(TimePickerBase); export const minuteProperty = new Property({ name: 'minute', // avoid defaultValue of 0 because it will prevent valueChanged from firing to initialize value due to it already matching defaultValue to start // sometimes user needs to set 0: https://github.com/NativeScript/NativeScript/issues/10457 defaultValue: null, valueChanged: (picker, oldValue, newValue) => { newValue = newValue || 0; if (!isMinuteValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'minute', newValue)); } picker.time = new Date(0, 0, 0, picker.hour, picker.minute); }, valueConverter: (v) => parseInt(v), }); minuteProperty.register(TimePickerBase); export const hourProperty = new Property({ name: 'hour', // avoid defaultValue of 0 because it will prevent valueChanged from firing to initialize value due to it already matching defaultValue to start // sometimes user needs to set 0: https://github.com/NativeScript/NativeScript/issues/10457 defaultValue: null, valueChanged: (picker, oldValue, newValue) => { newValue = newValue || 0; if (!isHourValid(newValue) || !isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'Hour', newValue)); } picker.time = new Date(0, 0, 0, picker.hour, picker.minute); }, valueConverter: (v) => parseInt(v), }); hourProperty.register(TimePickerBase); export const timeProperty = new Property({ name: 'time', defaultValue: new Date(), equalityComparer: dateComparer, valueChanged: (picker, oldValue, newValue) => { if (!isValidTime(picker)) { throw new Error(getErrorMessage(picker, 'time', newValue)); } picker.hour = newValue.getHours(); picker.minute = newValue.getMinutes(); }, }); timeProperty.register(TimePickerBase); export const iosPreferredDatePickerStyleProperty = new Property({ name: 'iosPreferredDatePickerStyle', defaultValue: 0, valueConverter: (v) => parseInt(v), }); iosPreferredDatePickerStyleProperty.register(TimePickerBase);