box-ui-elements-mlh
Version:
528 lines (446 loc) • 22.3 kB
JavaScript
function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); }
function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }
function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); if (superClass) _setPrototypeOf(subClass, superClass); }
function _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
function _possibleConstructorReturn(self, call) { if (call && (_typeof(call) === "object" || typeof call === "function")) { return call; } return _assertThisInitialized(self); }
function _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return self; }
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
import * as React from 'react';
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
import classNames from 'classnames';
import Pikaday from 'pikaday';
import noop from 'lodash/noop';
import range from 'lodash/range';
import uniqueId from 'lodash/uniqueId'; // @ts-ignore flow import
import { RESIN_TAG_TARGET } from '../../common/variables';
import IconAlert from '../../icons/general/IconAlert';
import IconCalendar from '../../icons/general/IconCalendar';
import IconClear from '../../icons/general/IconClear';
import { ButtonType } from '../button';
import Label from '../label';
import PlainButton from '../plain-button';
import Tooltip, { TooltipPosition, TooltipTheme } from '../tooltip'; // @ts-ignore flow import
import { convertDateToUnixMidnightTime } from '../../utils/datetime';
import './DatePicker.scss';
var messages = defineMessages({
previousMonth: {
"id": "boxui.base.previousMonth",
"defaultMessage": "Previous Month"
},
nextMonth: {
"id": "boxui.base.nextMonth",
"defaultMessage": "Next Month"
},
iconAlertText: {
"id": "boxui.datePicker.iconAlertText",
"defaultMessage": "Invalid Date"
},
dateClearButton: {
"id": "boxui.datePicker.dateClearButton",
"defaultMessage": "Clear Date"
},
chooseDate: {
"id": "boxui.datePicker.chooseDate",
"defaultMessage": "Choose Date"
}
});
var TOGGLE_DELAY_MS = 300;
var ENTER_KEY = 'Enter';
var ESCAPE_KEY = 'Escape';
var TAB_KEY = 'Tab';
export var DateFormat;
/**
* Converts date from being relative to GMT, to being relative to browser
* timezone. E.g., Thu Jun 29 2017 00:00:00 GMT =>
* Thu Jun 29 2017 00:00:00 GMT-0700 (PDT)
* @param {Date} date UTC date
* @returns {Date} date Local date
*/
(function (DateFormat) {
DateFormat["ISO_STRING_DATE_FORMAT"] = "isoString";
DateFormat["UTC_TIME_DATE_FORMAT"] = "utcTime";
DateFormat["UNIX_TIME_DATE_FORMAT"] = "unixTime";
DateFormat["UTC_ISO_STRING_DATE_FORMAT"] = "utcISOString";
})(DateFormat || (DateFormat = {}));
function convertUTCToLocal(date) {
var dateString = date.toUTCString(); // Remove ` GMT` from the timestamp string
var dateStringWithoutTimeZone = dateString.slice(0, -4);
return new Date(dateStringWithoutTimeZone);
}
function getFormattedDate(date, format) {
if (!date) {
return '';
}
var utcDate;
switch (format) {
case DateFormat.ISO_STRING_DATE_FORMAT:
return date.toISOString();
case DateFormat.UTC_TIME_DATE_FORMAT:
return convertDateToUnixMidnightTime(date);
case DateFormat.UTC_ISO_STRING_DATE_FORMAT:
utcDate = new Date(convertDateToUnixMidnightTime(date));
return utcDate.toISOString();
default:
return date.getTime();
}
}
var localesWhereWeekStartsOnSunday = ['en-US', 'en-CA', 'jp-JP'];
var DatePicker = /*#__PURE__*/function (_React$Component) {
_inherits(DatePicker, _React$Component);
var _super = _createSuper(DatePicker);
function DatePicker() {
var _this;
_classCallCheck(this, DatePicker);
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
_this = _super.call.apply(_super, [this].concat(args));
_defineProperty(_assertThisInitialized(_this), "errorMessageID", uniqueId('errorMessage'));
_defineProperty(_assertThisInitialized(_this), "descriptionID", uniqueId('description'));
_defineProperty(_assertThisInitialized(_this), "onSelectHandler", function () {
var date = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : null;
var onChange = _this.props.onChange;
if (onChange) {
var formattedDate = _this.formatValue(date);
onChange(date, formattedDate);
}
});
_defineProperty(_assertThisInitialized(_this), "datePicker", null);
_defineProperty(_assertThisInitialized(_this), "shouldStayClosed", false);
_defineProperty(_assertThisInitialized(_this), "focusDatePicker", function () {
// By default, this will open the datepicker too
if (_this.dateInputEl) {
_this.dateInputEl.focus();
}
});
_defineProperty(_assertThisInitialized(_this), "handleInputKeyDown", function (event) {
var isTextInputAllowed = _this.props.isTextInputAllowed;
if (_this.datePicker && _this.datePicker.isVisible()) {
event.stopPropagation();
} // Stops up/down arrow & spacebar from moving page scroll position since pikaday does not preventDefault correctly
if (!isTextInputAllowed && event.key !== TAB_KEY) {
event.preventDefault();
}
if (isTextInputAllowed && event.key === ENTER_KEY) {
event.preventDefault();
}
if (event.key === ENTER_KEY || event.key === ESCAPE_KEY || event.key === ' ') {
// Since pikaday auto-selects when you move the select box, enter/space don't do anything but close the date picker
if (_this.datePicker && _this.datePicker.isVisible()) {
_this.datePicker.hide();
}
}
});
_defineProperty(_assertThisInitialized(_this), "handleInputBlur", function (event) {
var _this$props = _this.props,
onBlur = _this$props.onBlur,
isTextInputAllowed = _this$props.isTextInputAllowed;
var nextActiveElement = event.relatedTarget || document.activeElement; // This is mostly here to cancel out the pikaday hide() on blur
if (_this.datePicker && _this.datePicker.isVisible() && nextActiveElement && nextActiveElement === _this.datePickerButtonEl) {
_this.shouldStayClosed = true;
setTimeout(function () {
_this.shouldStayClosed = false;
}, TOGGLE_DELAY_MS);
}
if (onBlur) {
onBlur(event);
} // Since we Fire parent onChange event if isTextInputAllowed
// fire it on blur if the user typed a correct date format
var inputDate = null;
if (_this.dateInputEl) {
inputDate = new Date(_this.dateInputEl.value);
}
if (isTextInputAllowed && inputDate && inputDate.getDate()) {
_this.onSelectHandler(inputDate);
}
});
_defineProperty(_assertThisInitialized(_this), "handleButtonClick", function (event) {
event.preventDefault();
event.stopPropagation();
if (!_this.shouldStayClosed) {
_this.focusDatePicker();
}
});
_defineProperty(_assertThisInitialized(_this), "formatDisplay", function (date) {
var _this$props2 = _this.props,
displayFormat = _this$props2.displayFormat,
intl = _this$props2.intl;
return date ? intl.formatDate(date, displayFormat) : '';
});
_defineProperty(_assertThisInitialized(_this), "formatValue", function (date) {
var dateFormat = _this.props.dateFormat;
return dateFormat ? getFormattedDate(date, dateFormat) : '';
});
_defineProperty(_assertThisInitialized(_this), "clearDate", function (event) {
event.preventDefault(); // so datepicker doesn't open after clearing
if (_this.datePicker) {
_this.datePicker.setDate(null);
}
_this.onSelectHandler(null);
});
return _this;
}
_createClass(DatePicker, [{
key: "componentDidMount",
value: function componentDidMount() {
var _this$props3 = this.props,
customInput = _this$props3.customInput,
dateFormat = _this$props3.dateFormat,
intl = _this$props3.intl,
isAlwaysVisible = _this$props3.isAlwaysVisible,
isTextInputAllowed = _this$props3.isTextInputAllowed,
maxDate = _this$props3.maxDate,
minDate = _this$props3.minDate,
value = _this$props3.value,
yearRange = _this$props3.yearRange;
var formatDate = intl.formatDate,
formatMessage = intl.formatMessage;
var nextMonth = messages.nextMonth,
previousMonth = messages.previousMonth;
var defaultValue = value; // When date format is utcTime, initial date needs to be converted from being relative to GMT to being
// relative to browser timezone
if (dateFormat === DateFormat.UTC_TIME_DATE_FORMAT && value) {
defaultValue = convertUTCToLocal(value);
} // Make sure the DST detection algorithm in browsers is up-to-date
var year = new Date().getFullYear();
var i18n = {
previousMonth: formatMessage(previousMonth),
nextMonth: formatMessage(nextMonth),
months: range(12).map(function (month) {
return formatDate(new Date(year, month, 15), {
month: 'long'
});
}),
// weekdays must start with Sunday, so array of dates below is May 1st-8th, 2016
weekdays: range(1, 8).map(function (date) {
return formatDate(new Date(2016, 4, date), {
weekday: 'long'
});
}),
weekdaysShort: range(1, 8).map(function (date) {
return formatDate(new Date(2016, 4, date), {
weekday: 'narrow'
});
})
}; // If "bound" is true (default), the DatePicker will be appended at the end of the document, with absolute positioning
// If "bound" is false, the DatePicker will be appended to the DOM right after the input, with relative positioning
this.datePicker = new Pikaday({
bound: !customInput,
blurFieldOnSelect: false,
// Available in pikaday > 1.5.1
setDefaultDate: true,
defaultDate: defaultValue === null ? undefined : defaultValue,
field: this.dateInputEl,
firstDay: localesWhereWeekStartsOnSunday.includes(intl.locale) ? 0 : 1,
maxDate: maxDate,
minDate: minDate,
position: 'bottom left',
i18n: i18n,
showDaysInNextAndPreviousMonths: true,
onSelect: this.onSelectHandler,
yearRange: yearRange,
toString: this.formatDisplay
});
if (isTextInputAllowed) {
this.updateDateInputValue(this.formatDisplay(defaultValue));
}
if (isAlwaysVisible) {
this.datePicker.show();
this.datePicker.hide = noop;
}
} // eslint-disable-next-line camelcase
}, {
key: "UNSAFE_componentWillReceiveProps",
value: function UNSAFE_componentWillReceiveProps(nextProps) {
if (!this.datePicker) return;
var _nextProps$value = nextProps.value,
nextValue = _nextProps$value === void 0 ? null : _nextProps$value,
_nextProps$minDate = nextProps.minDate,
nextMinDate = _nextProps$minDate === void 0 ? null : _nextProps$minDate,
_nextProps$maxDate = nextProps.maxDate,
nextMaxDate = _nextProps$maxDate === void 0 ? null : _nextProps$maxDate;
var _this$props4 = this.props,
value = _this$props4.value,
minDate = _this$props4.minDate,
maxDate = _this$props4.maxDate,
isTextInputAllowed = _this$props4.isTextInputAllowed;
var selectedDate = this.datePicker && this.datePicker.getDate(); // only set date when props change
if (nextValue && !value || !nextValue && value || nextValue && value && nextValue.getTime() !== value.getTime()) {
this.datePicker.setDate(nextValue);
} // If text input is allowed the dateInputEl will act as an uncontrolled input and
// we need to set formatted value manually.
if (isTextInputAllowed) {
this.updateDateInputValue(this.formatDisplay(nextValue));
}
if (nextMinDate && !minDate || nextMinDate && minDate || nextMinDate && minDate && nextMinDate.getTime() !== minDate.getTime()) {
this.datePicker.setMinDate(nextMinDate);
if (selectedDate && selectedDate < nextMinDate) {
this.datePicker.gotoDate(nextMinDate);
}
}
if (nextMaxDate && !maxDate || !nextMaxDate && maxDate || nextMaxDate && maxDate && nextMaxDate.getTime() !== maxDate.getTime()) {
this.datePicker.setMaxDate(nextMaxDate);
if (selectedDate && nextMaxDate && selectedDate > nextMaxDate) {
this.datePicker.gotoDate(nextMaxDate);
}
}
}
}, {
key: "componentWillUnmount",
value: function componentWillUnmount() {
if (this.datePicker) {
this.datePicker.destroy();
}
}
}, {
key: "updateDateInputValue",
value: function updateDateInputValue(value) {
if (this.dateInputEl) {
this.dateInputEl.value = value;
}
}
}, {
key: "render",
value: function render() {
var _this2 = this;
var _this$props5 = this.props,
className = _this$props5.className,
customInput = _this$props5.customInput,
description = _this$props5.description,
error = _this$props5.error,
errorTooltipPosition = _this$props5.errorTooltipPosition,
hideLabel = _this$props5.hideLabel,
hideOptionalLabel = _this$props5.hideOptionalLabel,
inputProps = _this$props5.inputProps,
intl = _this$props5.intl,
isAlwaysVisible = _this$props5.isAlwaysVisible,
isClearable = _this$props5.isClearable,
isDisabled = _this$props5.isDisabled,
isRequired = _this$props5.isRequired,
isTextInputAllowed = _this$props5.isTextInputAllowed,
label = _this$props5.label,
name = _this$props5.name,
onFocus = _this$props5.onFocus,
placeholder = _this$props5.placeholder,
resinTarget = _this$props5.resinTarget,
value = _this$props5.value;
var formatMessage = intl.formatMessage;
var classes = classNames(className, 'date-picker-wrapper', {
'show-clear-btn': !!value,
'show-error': !!error
});
var hasError = !!error;
var ariaAttrs = {
'aria-invalid': hasError,
'aria-required': isRequired,
'aria-errormessage': this.errorMessageID,
'aria-describedby': description ? this.descriptionID : undefined
};
var resinTargetAttr = resinTarget ? _defineProperty({}, RESIN_TAG_TARGET, resinTarget) : {};
var valueAttr = isTextInputAllowed ? {
defaultValue: this.formatDisplay(value)
} : {
value: this.formatDisplay(value)
};
var onChangeAttr = isTextInputAllowed ? {} : {
onChange: noop
};
/* fixes proptype error about readonly field (not adding readonly so constraint validation works) */
return /*#__PURE__*/React.createElement("div", {
className: classes
}, /*#__PURE__*/React.createElement("span", {
className: "date-picker-icon-holder"
}, /*#__PURE__*/React.createElement(Label, {
hideLabel: hideLabel,
showOptionalText: !hideOptionalLabel && !isRequired,
text: label
}, /*#__PURE__*/React.createElement(React.Fragment, null, !!description && /*#__PURE__*/React.createElement("div", {
id: this.descriptionID,
className: "date-picker-description"
}, description), /*#__PURE__*/React.createElement(Tooltip, {
className: "date-picker-error-tooltip",
isShown: !!error,
position: errorTooltipPosition,
text: error || '',
theme: TooltipTheme.ERROR
}, customInput ? /*#__PURE__*/React.cloneElement(customInput, _objectSpread(_objectSpread({
disabled: isDisabled,
ref: function ref(_ref2) {
_this2.dateInputEl = _ref2;
},
required: isRequired
}, resinTargetAttr), ariaAttrs)) : /*#__PURE__*/React.createElement("input", _extends({
ref: function ref(_ref3) {
_this2.dateInputEl = _ref3;
},
className: "date-picker-input",
disabled: isDisabled,
onBlur: this.handleInputBlur,
placeholder: placeholder || formatMessage(messages.chooseDate),
required: isRequired,
type: "text"
}, onChangeAttr, {
onFocus: onFocus,
onKeyDown: this.handleInputKeyDown
}, resinTargetAttr, ariaAttrs, inputProps, valueAttr))), /*#__PURE__*/React.createElement("span", {
id: this.errorMessageID,
className: "accessibility-hidden",
role: "alert"
}, error))), isClearable && !!value && !isDisabled ? /*#__PURE__*/React.createElement(PlainButton, {
"aria-label": formatMessage(messages.dateClearButton),
className: "date-picker-clear-btn",
onClick: this.clearDate,
type: ButtonType.BUTTON
}, /*#__PURE__*/React.createElement(IconClear, {
height: 12,
width: 12
})) : null, error ? /*#__PURE__*/React.createElement(IconAlert, {
className: "date-picker-icon-alert",
height: 13,
title: /*#__PURE__*/React.createElement(FormattedMessage, messages.iconAlertText),
width: 13
}) : null, !isAlwaysVisible && /*#__PURE__*/React.createElement(PlainButton, {
"aria-label": formatMessage(messages.chooseDate),
className: "date-picker-open-btn",
getDOMRef: function getDOMRef(ref) {
_this2.datePickerButtonEl = ref;
},
isDisabled: isDisabled,
onClick: this.handleButtonClick,
type: ButtonType.BUTTON
}, /*#__PURE__*/React.createElement(IconCalendar, {
height: 17,
width: 16
})), /*#__PURE__*/React.createElement("input", {
className: "date-picker-unix-time-input",
name: name,
readOnly: true,
type: "hidden",
value: value ? this.formatValue(value) : ''
})));
}
}]);
return DatePicker;
}(React.Component);
_defineProperty(DatePicker, "defaultProps", {
className: '',
dateFormat: DateFormat.UNIX_TIME_DATE_FORMAT,
displayFormat: {},
error: '',
errorTooltipPosition: TooltipPosition.BOTTOM_LEFT,
inputProps: {},
isClearable: true,
isTextInputAllowed: false,
yearRange: 10
});
export { DatePicker as DatePickerBase };
export default injectIntl(DatePicker);
//# sourceMappingURL=DatePicker.js.map