UNPKG

react-schedule-selector-custom

Version:

A mobile-friendly when2meet-style grid-based schedule selector

406 lines (335 loc) 49.2 kB
"use strict"; require("core-js/modules/web.dom-collections.iterator"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = exports.preventScroll = exports.GridCell = void 0; var React = _interopRequireWildcard(require("react")); var _styledComponents = _interopRequireDefault(require("styled-components")); var _add_minutes = _interopRequireDefault(require("date-fns/add_minutes")); var _add_days = _interopRequireDefault(require("date-fns/add_days")); var _start_of_day = _interopRequireDefault(require("date-fns/start_of_day")); var _is_same_minute = _interopRequireDefault(require("date-fns/is_same_minute")); var _format = _interopRequireDefault(require("date-fns/format")); var _typography = require("./typography"); var _colors = _interopRequireDefault(require("./colors")); var _selectionSchemes = _interopRequireDefault(require("./selection-schemes")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; } // Import only the methods we need from date-fns in order to keep build size small const Wrapper = _styledComponents.default.div.withConfig({ displayName: "ScheduleSelector__Wrapper", componentId: "sc-1jag5hl-0" })(["display:flex;align-items:center;width:100%;user-select:none;"]); const Grid = _styledComponents.default.div.withConfig({ displayName: "ScheduleSelector__Grid", componentId: "sc-1jag5hl-1" })(["display:grid;grid-template-columns:auto repeat(", ",1fr);grid-template-rows:auto repeat(", ",1fr);column-gap:", ";row-gap:", ";width:100%;"], props => props.columns, props => props.rows, props => props.columnGap, props => props.rowGap); const GridCell = _styledComponents.default.div.withConfig({ displayName: "ScheduleSelector__GridCell", componentId: "sc-1jag5hl-2" })(["place-self:stretch;touch-action:none;"]); exports.GridCell = GridCell; const DateCell = _styledComponents.default.div.withConfig({ displayName: "ScheduleSelector__DateCell", componentId: "sc-1jag5hl-3" })(["width:100%;height:25px;background-color:", ";&:hover{background-color:", ";}"], props => props.selected ? props.selectedColor : props.unselectedColor, props => props.hoveredColor); const DateLabel = (0, _styledComponents.default)(_typography.Subtitle).withConfig({ displayName: "ScheduleSelector__DateLabel", componentId: "sc-1jag5hl-4" })(["@media (max-width:699px){font-size:12px;}margin:0;margin-bottom:4px;"]); const TimeText = (0, _styledComponents.default)(_typography.Text).withConfig({ displayName: "ScheduleSelector__TimeText", componentId: "sc-1jag5hl-5" })(["@media (max-width:699px){font-size:10px;}text-align:right;margin:0;margin-right:4px;"]); const preventScroll = e => { e.preventDefault(); }; exports.preventScroll = preventScroll; class ScheduleSelector extends React.Component { // documentMouseUpHandler: () => void = () => {} // endSelection: () => void = () => {} // handleTouchMoveEvent: (event: React.SyntheticTouchEvent<*>) => void // handleTouchEndEvent: () => void // handleMouseUpEvent: (date: Date) => void // handleMouseEnterEvent: (date: Date) => void // handleSelectionStartEvent: (date: Date) => void static getDerivedStateFromProps(props, state) { // As long as the user isn't in the process of selecting, allow prop changes to re-populate selection state if (state.selectionStart == null) { return { selectionDraft: [...props.selection], dates: ScheduleSelector.computeDatesMatrix(props) }; } return null; } static computeDatesMatrix(props) { const startTime = (0, _start_of_day.default)(props.startDate); const dates = []; for (let d = 0; d < props.numDays; d += 1) { const currentDay = []; let currentTime = (0, _start_of_day.default)((0, _add_days.default)(startTime, d)); // Start at the beginning of the day // Number of chunks in a day const numChunks = 24 * 60 / props.minutesChunks; while (currentDay.length < numChunks) { currentDay.push(new Date(currentTime)); currentTime = (0, _add_minutes.default)(currentTime, props.minutesChunks); // Increment by 'minutes' } dates.push(currentDay); } return dates; } constructor(props) { super(props); this.cellToDate = new Map(); this.gridRef = null; this.renderDateCellWrapper = time => { const startHandler = () => { this.handleSelectionStartEvent(time); }; const selected = Boolean(this.state.selectionDraft.find(a => (0, _is_same_minute.default)(a, time))); return /*#__PURE__*/React.createElement(GridCell, { className: "rgdp__grid-cell", role: "presentation", key: time.toISOString() // Mouse handlers , onMouseDown: startHandler, onMouseEnter: () => { this.handleMouseEnterEvent(time); }, onMouseUp: () => { this.handleMouseUpEvent(time); } // Touch handlers // Since touch events fire on the event where the touch-drag started, there's no point in passing // in the time parameter, instead these handlers will do their job using the default Event // parameters , onTouchStart: startHandler, onTouchMove: this.handleTouchMoveEvent, onTouchEnd: this.handleTouchEndEvent }, this.renderDateCell(time, selected)); }; this.renderDateCell = (time, selected) => { const refSetter = dateCell => { if (dateCell) { this.cellToDate.set(dateCell, time); } }; if (this.props.renderDateCell) { return this.props.renderDateCell(time, selected, refSetter); } else { return /*#__PURE__*/React.createElement(DateCell, { selected: selected, ref: refSetter, selectedColor: this.props.selectedColor, unselectedColor: this.props.unselectedColor, hoveredColor: this.props.hoveredColor }); } }; this.renderTimeLabel = time => { if (this.props.renderTimeLabel) { return this.props.renderTimeLabel(time); } else { return /*#__PURE__*/React.createElement(TimeText, null, (0, _format.default)(time, this.props.timeFormat)); } }; this.renderDateLabel = date => { if (this.props.renderDateLabel) { return this.props.renderDateLabel(date); } else { return /*#__PURE__*/React.createElement(DateLabel, null, (0, _format.default)(date, this.props.dateFormat)); } }; this.state = { selectionDraft: [...this.props.selection], // copy it over selectionType: null, selectionStart: null, isTouchDragging: false, dates: ScheduleSelector.computeDatesMatrix(props) }; this.selectionSchemeHandlers = { linear: _selectionSchemes.default.linear, square: _selectionSchemes.default.square }; this.endSelection = this.endSelection.bind(this); this.handleMouseUpEvent = this.handleMouseUpEvent.bind(this); this.handleMouseEnterEvent = this.handleMouseEnterEvent.bind(this); this.handleTouchMoveEvent = this.handleTouchMoveEvent.bind(this); this.handleTouchEndEvent = this.handleTouchEndEvent.bind(this); this.handleSelectionStartEvent = this.handleSelectionStartEvent.bind(this); } componentDidMount() { // We need to add the endSelection event listener to the document itself in order // to catch the cases where the users ends their mouse-click somewhere besides // the date cells (in which case none of the DateCell's onMouseUp handlers would fire) // // This isn't necessary for touch events since the `touchend` event fires on // the element where the touch/drag started so it's always caught. document.addEventListener('mouseup', this.endSelection); // Prevent page scrolling when user is dragging on the date cells this.cellToDate.forEach((value, dateCell) => { if (dateCell && dateCell.addEventListener) { // @ts-ignore dateCell.addEventListener('touchmove', preventScroll, { passive: false }); } }); } componentWillUnmount() { document.removeEventListener('mouseup', this.endSelection); this.cellToDate.forEach((value, dateCell) => { if (dateCell && dateCell.removeEventListener) { // @ts-ignore dateCell.removeEventListener('touchmove', preventScroll); } }); } // Performs a lookup into this.cellToDate to retrieve the Date that corresponds to // the cell where this touch event is right now. Note that this method will only work // if the event is a `touchmove` event since it's the only one that has a `touches` list. getTimeFromTouchEvent(event) { const { touches } = event; if (!touches || touches.length === 0) return null; const { clientX, clientY } = touches[0]; const targetElement = document.elementFromPoint(clientX, clientY); if (targetElement) { const cellTime = this.cellToDate.get(targetElement); return cellTime !== null && cellTime !== void 0 ? cellTime : null; } return null; } endSelection() { this.props.onChange(this.state.selectionDraft); this.setState({ selectionType: null, selectionStart: null }); } // Given an ending Date, determines all the dates that should be selected in this draft updateAvailabilityDraft(selectionEnd, callback) { const { selectionType, selectionStart } = this.state; if (selectionType === null || selectionStart === null) return; let newSelection = []; if (selectionStart && selectionEnd && selectionType) { newSelection = this.selectionSchemeHandlers[this.props.selectionScheme](selectionStart, selectionEnd, this.state.dates); } let nextDraft = [...this.props.selection]; if (selectionType === 'add') { nextDraft = Array.from(new Set([...nextDraft, ...newSelection])); } else if (selectionType === 'remove') { nextDraft = nextDraft.filter(a => !newSelection.find(b => (0, _is_same_minute.default)(a, b))); } this.setState({ selectionDraft: nextDraft }, callback); } // Isomorphic (mouse and touch) handler since starting a selection works the same way for both classes of user input handleSelectionStartEvent(startTime) { // Check if the startTime cell is selected/unselected to determine if this drag-select should // add values or remove values const timeSelected = this.props.selection.find(a => (0, _is_same_minute.default)(a, startTime)); this.setState({ selectionType: timeSelected ? 'remove' : 'add', selectionStart: startTime }); } handleMouseEnterEvent(time) { // Need to update selection draft on mouseup as well in order to catch the cases // where the user just clicks on a single cell (because no mouseenter events fire // in this scenario) this.updateAvailabilityDraft(time); } handleMouseUpEvent(time) { this.updateAvailabilityDraft(time); // Don't call this.endSelection() here because the document mouseup handler will do it } handleTouchMoveEvent(event) { this.setState({ isTouchDragging: true }); const cellTime = this.getTimeFromTouchEvent(event); if (cellTime) { this.updateAvailabilityDraft(cellTime); } } handleTouchEndEvent() { if (!this.state.isTouchDragging) { // Going down this branch means the user tapped but didn't drag -- which // means the availability draft hasn't yet been updated (since // handleTouchMoveEvent was never called) so we need to do it now this.updateAvailabilityDraft(null, () => { this.endSelection(); }); } else { this.endSelection(); } this.setState({ isTouchDragging: false }); } renderFullDateGrid() { let flattenedDates = []; const numDays = this.state.dates.length; const numTimes = this.state.dates[0].length; for (let j = 0; j < numTimes; j += 1) { for (let i = 0; i < numDays; i += 1) { flattenedDates.push(this.state.dates[i][j]); } } // Filter out any null dates or undefined dates flattenedDates = flattenedDates.filter(a => a !== null && a !== undefined); const dateGridElements = flattenedDates.map(this.renderDateCellWrapper); for (let i = 0; i < numTimes; i += 1) { const index = i * numDays; const time = this.state.dates[0][i]; // Inject the time label at the start of every row dateGridElements.splice(index + i, 0, this.renderTimeLabel(time)); } return [ /*#__PURE__*/ // Empty top left corner React.createElement("div", { key: "topleft" }), // Top row of dates ...this.state.dates.map((dayOfTimes, index) => /*#__PURE__*/React.cloneElement(this.renderDateLabel(dayOfTimes[0]), { key: "date-".concat(index) })), // Every row after that ...dateGridElements.map((element, index) => /*#__PURE__*/React.cloneElement(element, { key: "time-".concat(index) }))]; } render() { return /*#__PURE__*/React.createElement(Wrapper, null, /*#__PURE__*/React.createElement(Grid, { columns: this.state.dates.length, rows: this.state.dates[0].length, columnGap: this.props.columnGap, rowGap: this.props.rowGap, ref: el => { this.gridRef = el; } }, this.renderFullDateGrid())); } } exports.default = ScheduleSelector; ScheduleSelector.defaultProps = { selection: [], selectionScheme: 'square', numDays: 7, minutesChunks: 60, startDate: new Date(), timeFormat: 'ha', dateFormat: 'M/D', columnGap: '4px', rowGap: '4px', selectedColor: _colors.default.blue, unselectedColor: _colors.default.paleBlue, hoveredColor: _colors.default.lightBlue, onChange: () => {} }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/lib/ScheduleSelector.tsx"],"names":["Wrapper","styled","div","Grid","props","columns","rows","columnGap","rowGap","GridCell","DateCell","selected","selectedColor","unselectedColor","hoveredColor","DateLabel","Subtitle","TimeText","Text","preventScroll","e","preventDefault","ScheduleSelector","React","Component","getDerivedStateFromProps","state","selectionStart","selectionDraft","selection","dates","computeDatesMatrix","startTime","startDate","d","numDays","currentDay","currentTime","numChunks","minutesChunks","length","push","Date","constructor","cellToDate","Map","gridRef","renderDateCellWrapper","time","startHandler","handleSelectionStartEvent","Boolean","find","a","toISOString","handleMouseEnterEvent","handleMouseUpEvent","handleTouchMoveEvent","handleTouchEndEvent","renderDateCell","refSetter","dateCell","set","renderTimeLabel","timeFormat","renderDateLabel","date","dateFormat","selectionType","isTouchDragging","selectionSchemeHandlers","linear","selectionSchemes","square","endSelection","bind","componentDidMount","document","addEventListener","forEach","value","passive","componentWillUnmount","removeEventListener","getTimeFromTouchEvent","event","touches","clientX","clientY","targetElement","elementFromPoint","cellTime","get","onChange","setState","updateAvailabilityDraft","selectionEnd","callback","newSelection","selectionScheme","nextDraft","Array","from","Set","filter","b","timeSelected","renderFullDateGrid","flattenedDates","numTimes","j","i","undefined","dateGridElements","map","index","splice","dayOfTimes","cloneElement","key","element","render","el","defaultProps","colors","blue","paleBlue","lightBlue"],"mappings":";;;;;;;;;AAAA;;AACA;;AAGA;;AAEA;;AACA;;AACA;;AACA;;AAEA;;AACA;;AACA;;;;;;;;AAVA;AAaA,MAAMA,OAAO,GAAGC,0BAAOC,GAAV;AAAA;AAAA;AAAA,oEAAb;;AAOA,MAAMC,IAAI,GAAGF,0BAAOC,GAAV;AAAA;AAAA;AAAA,mJAE6BE,KAAK,IAAIA,KAAK,CAACC,OAF5C,EAG0BD,KAAK,IAAIA,KAAK,CAACE,IAHzC,EAIMF,KAAK,IAAIA,KAAK,CAACG,SAJrB,EAKGH,KAAK,IAAIA,KAAK,CAACI,MALlB,CAAV;;AASO,MAAMC,QAAQ,GAAGR,0BAAOC,GAAV;AAAA;AAAA;AAAA,6CAAd;;;;AAKP,MAAMQ,QAAQ,GAAGT,0BAAOC,GAAV;AAAA;AAAA;AAAA,qFAQQE,KAAK,IAAKA,KAAK,CAACO,QAAN,GAAiBP,KAAK,CAACQ,aAAvB,GAAuCR,KAAK,CAACS,eAR/D,EAWUT,KAAK,IAAIA,KAAK,CAACU,YAXzB,CAAd;;AAeA,MAAMC,SAAS,GAAG,+BAAOC,oBAAP,CAAH;AAAA;AAAA;AAAA,4EAAf;AAQA,MAAMC,QAAQ,GAAG,+BAAOC,gBAAP,CAAH;AAAA;AAAA;AAAA,4FAAd;;AAsCO,MAAMC,aAAa,GAAIC,CAAD,IAAmB;AAC9CA,EAAAA,CAAC,CAACC,cAAF;AACD,CAFM;;;;AAIQ,MAAMC,gBAAN,SAA+BC,KAAK,CAACC,SAArC,CAAqE;AAGlF;AACA;AACA;AACA;AACA;AACA;AACA;AAmBA,SAAOC,wBAAP,CAAgCrB,KAAhC,EAAkDsB,KAAlD,EAA+F;AAC7F;AACA,QAAIA,KAAK,CAACC,cAAN,IAAwB,IAA5B,EAAkC;AAChC,aAAO;AACLC,QAAAA,cAAc,EAAE,CAAC,GAAGxB,KAAK,CAACyB,SAAV,CADX;AAELC,QAAAA,KAAK,EAAER,gBAAgB,CAACS,kBAAjB,CAAoC3B,KAApC;AAFF,OAAP;AAID;;AACD,WAAO,IAAP;AACD;;AAED,SAAO2B,kBAAP,CAA0B3B,KAA1B,EAAgE;AAC9D,UAAM4B,SAAS,GAAG,2BAAW5B,KAAK,CAAC6B,SAAjB,CAAlB;AACA,UAAMH,KAAyB,GAAG,EAAlC;;AACA,SAAK,IAAII,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAG9B,KAAK,CAAC+B,OAA1B,EAAmCD,CAAC,IAAI,CAAxC,EAA2C;AACzC,YAAME,UAAkB,GAAG,EAA3B;AACA,UAAIC,WAAW,GAAG,2BAAW,uBAAQL,SAAR,EAAmBE,CAAnB,CAAX,CAAlB,CAFyC,CAEW;AACpD;;AACA,YAAMI,SAAS,GAAI,KAAK,EAAN,GAAYlC,KAAK,CAACmC,aAApC;;AACA,aAAOH,UAAU,CAACI,MAAX,GAAoBF,SAA3B,EAAsC;AACpCF,QAAAA,UAAU,CAACK,IAAX,CAAgB,IAAIC,IAAJ,CAASL,WAAT,CAAhB;AACAA,QAAAA,WAAW,GAAG,0BAAWA,WAAX,EAAwBjC,KAAK,CAACmC,aAA9B,CAAd,CAFoC,CAEuB;AAC5D;;AACDT,MAAAA,KAAK,CAACW,IAAN,CAAWL,UAAX;AACD;;AACD,WAAON,KAAP;AACD;;AAEDa,EAAAA,WAAW,CAACvC,KAAD,EAAmB;AAC5B,UAAMA,KAAN;AAD4B,SAtD9BwC,UAsD8B,GAtDG,IAAIC,GAAJ,EAsDH;AAAA,SA9C9BC,OA8C8B,GA9CA,IA8CA;;AAAA,SAiJ9BC,qBAjJ8B,GAiJLC,IAAD,IAA6B;AACnD,YAAMC,YAAY,GAAG,MAAM;AACzB,aAAKC,yBAAL,CAA+BF,IAA/B;AACD,OAFD;;AAIA,YAAMrC,QAAQ,GAAGwC,OAAO,CAAC,KAAKzB,KAAL,CAAWE,cAAX,CAA0BwB,IAA1B,CAA+BC,CAAC,IAAI,6BAAaA,CAAb,EAAgBL,IAAhB,CAApC,CAAD,CAAxB;AAEA,0BACE,oBAAC,QAAD;AACE,QAAA,SAAS,EAAC,iBADZ;AAEE,QAAA,IAAI,EAAC,cAFP;AAGE,QAAA,GAAG,EAAEA,IAAI,CAACM,WAAL,EAHP,CAIE;AAJF;AAKE,QAAA,WAAW,EAAEL,YALf;AAME,QAAA,YAAY,EAAE,MAAM;AAClB,eAAKM,qBAAL,CAA2BP,IAA3B;AACD,SARH;AASE,QAAA,SAAS,EAAE,MAAM;AACf,eAAKQ,kBAAL,CAAwBR,IAAxB;AACD,SAXH,CAYE;AACA;AACA;AACA;AAfF;AAgBE,QAAA,YAAY,EAAEC,YAhBhB;AAiBE,QAAA,WAAW,EAAE,KAAKQ,oBAjBpB;AAkBE,QAAA,UAAU,EAAE,KAAKC;AAlBnB,SAoBG,KAAKC,cAAL,CAAoBX,IAApB,EAA0BrC,QAA1B,CApBH,CADF;AAwBD,KAhL6B;;AAAA,SAkL9BgD,cAlL8B,GAkLb,CAACX,IAAD,EAAarC,QAAb,KAAgD;AAC/D,YAAMiD,SAAS,GAAIC,QAAD,IAAkC;AAClD,YAAIA,QAAJ,EAAc;AACZ,eAAKjB,UAAL,CAAgBkB,GAAhB,CAAoBD,QAApB,EAA8Bb,IAA9B;AACD;AACF,OAJD;;AAKA,UAAI,KAAK5C,KAAL,CAAWuD,cAAf,EAA+B;AAC7B,eAAO,KAAKvD,KAAL,CAAWuD,cAAX,CAA0BX,IAA1B,EAAgCrC,QAAhC,EAA0CiD,SAA1C,CAAP;AACD,OAFD,MAEO;AACL,4BACE,oBAAC,QAAD;AACE,UAAA,QAAQ,EAAEjD,QADZ;AAEE,UAAA,GAAG,EAAEiD,SAFP;AAGE,UAAA,aAAa,EAAE,KAAKxD,KAAL,CAAWQ,aAH5B;AAIE,UAAA,eAAe,EAAE,KAAKR,KAAL,CAAWS,eAJ9B;AAKE,UAAA,YAAY,EAAE,KAAKT,KAAL,CAAWU;AAL3B,UADF;AASD;AACF,KArM6B;;AAAA,SAuM9BiD,eAvM8B,GAuMXf,IAAD,IAA6B;AAC7C,UAAI,KAAK5C,KAAL,CAAW2D,eAAf,EAAgC;AAC9B,eAAO,KAAK3D,KAAL,CAAW2D,eAAX,CAA2Bf,IAA3B,CAAP;AACD,OAFD,MAEO;AACL,4BAAO,oBAAC,QAAD,QAAW,qBAAWA,IAAX,EAAiB,KAAK5C,KAAL,CAAW4D,UAA5B,CAAX,CAAP;AACD;AACF,KA7M6B;;AAAA,SA+M9BC,eA/M8B,GA+MXC,IAAD,IAA6B;AAC7C,UAAI,KAAK9D,KAAL,CAAW6D,eAAf,EAAgC;AAC9B,eAAO,KAAK7D,KAAL,CAAW6D,eAAX,CAA2BC,IAA3B,CAAP;AACD,OAFD,MAEO;AACL,4BAAO,oBAAC,SAAD,QAAY,qBAAWA,IAAX,EAAiB,KAAK9D,KAAL,CAAW+D,UAA5B,CAAZ,CAAP;AACD;AACF,KArN6B;;AAG5B,SAAKzC,KAAL,GAAa;AACXE,MAAAA,cAAc,EAAE,CAAC,GAAG,KAAKxB,KAAL,CAAWyB,SAAf,CADL;AACgC;AAC3CuC,MAAAA,aAAa,EAAE,IAFJ;AAGXzC,MAAAA,cAAc,EAAE,IAHL;AAIX0C,MAAAA,eAAe,EAAE,KAJN;AAKXvC,MAAAA,KAAK,EAAER,gBAAgB,CAACS,kBAAjB,CAAoC3B,KAApC;AALI,KAAb;AAQA,SAAKkE,uBAAL,GAA+B;AAC7BC,MAAAA,MAAM,EAAEC,0BAAiBD,MADI;AAE7BE,MAAAA,MAAM,EAAED,0BAAiBC;AAFI,KAA/B;AAKA,SAAKC,YAAL,GAAoB,KAAKA,YAAL,CAAkBC,IAAlB,CAAuB,IAAvB,CAApB;AACA,SAAKnB,kBAAL,GAA0B,KAAKA,kBAAL,CAAwBmB,IAAxB,CAA6B,IAA7B,CAA1B;AACA,SAAKpB,qBAAL,GAA6B,KAAKA,qBAAL,CAA2BoB,IAA3B,CAAgC,IAAhC,CAA7B;AACA,SAAKlB,oBAAL,GAA4B,KAAKA,oBAAL,CAA0BkB,IAA1B,CAA+B,IAA/B,CAA5B;AACA,SAAKjB,mBAAL,GAA2B,KAAKA,mBAAL,CAAyBiB,IAAzB,CAA8B,IAA9B,CAA3B;AACA,SAAKzB,yBAAL,GAAiC,KAAKA,yBAAL,CAA+ByB,IAA/B,CAAoC,IAApC,CAAjC;AACD;;AAEDC,EAAAA,iBAAiB,GAAG;AAClB;AACA;AACA;AACA;AACA;AACA;AACAC,IAAAA,QAAQ,CAACC,gBAAT,CAA0B,SAA1B,EAAqC,KAAKJ,YAA1C,EAPkB,CASlB;;AACA,SAAK9B,UAAL,CAAgBmC,OAAhB,CAAwB,CAACC,KAAD,EAAQnB,QAAR,KAAqB;AAC3C,UAAIA,QAAQ,IAAIA,QAAQ,CAACiB,gBAAzB,EAA2C;AACzC;AACAjB,QAAAA,QAAQ,CAACiB,gBAAT,CAA0B,WAA1B,EAAuC3D,aAAvC,EAAsD;AAAE8D,UAAAA,OAAO,EAAE;AAAX,SAAtD;AACD;AACF,KALD;AAMD;;AAEDC,EAAAA,oBAAoB,GAAG;AACrBL,IAAAA,QAAQ,CAACM,mBAAT,CAA6B,SAA7B,EAAwC,KAAKT,YAA7C;AACA,SAAK9B,UAAL,CAAgBmC,OAAhB,CAAwB,CAACC,KAAD,EAAQnB,QAAR,KAAqB;AAC3C,UAAIA,QAAQ,IAAIA,QAAQ,CAACsB,mBAAzB,EAA8C;AAC5C;AACAtB,QAAAA,QAAQ,CAACsB,mBAAT,CAA6B,WAA7B,EAA0ChE,aAA1C;AACD;AACF,KALD;AAMD,GA1GiF,CA4GlF;AACA;AACA;;;AACAiE,EAAAA,qBAAqB,CAACC,KAAD,EAA4C;AAC/D,UAAM;AAAEC,MAAAA;AAAF,QAAcD,KAApB;AACA,QAAI,CAACC,OAAD,IAAYA,OAAO,CAAC9C,MAAR,KAAmB,CAAnC,EAAsC,OAAO,IAAP;AACtC,UAAM;AAAE+C,MAAAA,OAAF;AAAWC,MAAAA;AAAX,QAAuBF,OAAO,CAAC,CAAD,CAApC;AACA,UAAMG,aAAa,GAAGZ,QAAQ,CAACa,gBAAT,CAA0BH,OAA1B,EAAmCC,OAAnC,CAAtB;;AACA,QAAIC,aAAJ,EAAmB;AACjB,YAAME,QAAQ,GAAG,KAAK/C,UAAL,CAAgBgD,GAAhB,CAAoBH,aAApB,CAAjB;AACA,aAAOE,QAAP,aAAOA,QAAP,cAAOA,QAAP,GAAmB,IAAnB;AACD;;AACD,WAAO,IAAP;AACD;;AAEDjB,EAAAA,YAAY,GAAG;AACb,SAAKtE,KAAL,CAAWyF,QAAX,CAAoB,KAAKnE,KAAL,CAAWE,cAA/B;AACA,SAAKkE,QAAL,CAAc;AACZ1B,MAAAA,aAAa,EAAE,IADH;AAEZzC,MAAAA,cAAc,EAAE;AAFJ,KAAd;AAID,GAjIiF,CAmIlF;;;AACAoE,EAAAA,uBAAuB,CAACC,YAAD,EAA4BC,QAA5B,EAAmD;AACxE,UAAM;AAAE7B,MAAAA,aAAF;AAAiBzC,MAAAA;AAAjB,QAAoC,KAAKD,KAA/C;AAEA,QAAI0C,aAAa,KAAK,IAAlB,IAA0BzC,cAAc,KAAK,IAAjD,EAAuD;AAEvD,QAAIuE,YAAyB,GAAG,EAAhC;;AACA,QAAIvE,cAAc,IAAIqE,YAAlB,IAAkC5B,aAAtC,EAAqD;AACnD8B,MAAAA,YAAY,GAAG,KAAK5B,uBAAL,CAA6B,KAAKlE,KAAL,CAAW+F,eAAxC,EACbxE,cADa,EAEbqE,YAFa,EAGb,KAAKtE,KAAL,CAAWI,KAHE,CAAf;AAKD;;AAED,QAAIsE,SAAS,GAAG,CAAC,GAAG,KAAKhG,KAAL,CAAWyB,SAAf,CAAhB;;AACA,QAAIuC,aAAa,KAAK,KAAtB,EAA6B;AAC3BgC,MAAAA,SAAS,GAAGC,KAAK,CAACC,IAAN,CAAW,IAAIC,GAAJ,CAAQ,CAAC,GAAGH,SAAJ,EAAe,GAAGF,YAAlB,CAAR,CAAX,CAAZ;AACD,KAFD,MAEO,IAAI9B,aAAa,KAAK,QAAtB,EAAgC;AACrCgC,MAAAA,SAAS,GAAGA,SAAS,CAACI,MAAV,CAAiBnD,CAAC,IAAI,CAAC6C,YAAY,CAAC9C,IAAb,CAAkBqD,CAAC,IAAI,6BAAapD,CAAb,EAAgBoD,CAAhB,CAAvB,CAAvB,CAAZ;AACD;;AAED,SAAKX,QAAL,CAAc;AAAElE,MAAAA,cAAc,EAAEwE;AAAlB,KAAd,EAA6CH,QAA7C;AACD,GA1JiF,CA4JlF;;;AACA/C,EAAAA,yBAAyB,CAAClB,SAAD,EAAkB;AACzC;AACA;AACA,UAAM0E,YAAY,GAAG,KAAKtG,KAAL,CAAWyB,SAAX,CAAqBuB,IAArB,CAA0BC,CAAC,IAAI,6BAAaA,CAAb,EAAgBrB,SAAhB,CAA/B,CAArB;AACA,SAAK8D,QAAL,CAAc;AACZ1B,MAAAA,aAAa,EAAEsC,YAAY,GAAG,QAAH,GAAc,KAD7B;AAEZ/E,MAAAA,cAAc,EAAEK;AAFJ,KAAd;AAID;;AAEDuB,EAAAA,qBAAqB,CAACP,IAAD,EAAa;AAChC;AACA;AACA;AACA,SAAK+C,uBAAL,CAA6B/C,IAA7B;AACD;;AAEDQ,EAAAA,kBAAkB,CAACR,IAAD,EAAa;AAC7B,SAAK+C,uBAAL,CAA6B/C,IAA7B,EAD6B,CAE7B;AACD;;AAEDS,EAAAA,oBAAoB,CAAC4B,KAAD,EAA0B;AAC5C,SAAKS,QAAL,CAAc;AAAEzB,MAAAA,eAAe,EAAE;AAAnB,KAAd;AACA,UAAMsB,QAAQ,GAAG,KAAKP,qBAAL,CAA2BC,KAA3B,CAAjB;;AACA,QAAIM,QAAJ,EAAc;AACZ,WAAKI,uBAAL,CAA6BJ,QAA7B;AACD;AACF;;AAEDjC,EAAAA,mBAAmB,GAAG;AACpB,QAAI,CAAC,KAAKhC,KAAL,CAAW2C,eAAhB,EAAiC;AAC/B;AACA;AACA;AACA,WAAK0B,uBAAL,CAA6B,IAA7B,EAAmC,MAAM;AACvC,aAAKrB,YAAL;AACD,OAFD;AAGD,KAPD,MAOO;AACL,WAAKA,YAAL;AACD;;AACD,SAAKoB,QAAL,CAAc;AAAEzB,MAAAA,eAAe,EAAE;AAAnB,KAAd;AACD;;AAwEDsC,EAAAA,kBAAkB,GAAuB;AACvC,QAAIC,cAAsB,GAAG,EAA7B;AACA,UAAMzE,OAAO,GAAG,KAAKT,KAAL,CAAWI,KAAX,CAAiBU,MAAjC;AACA,UAAMqE,QAAQ,GAAG,KAAKnF,KAAL,CAAWI,KAAX,CAAiB,CAAjB,EAAoBU,MAArC;;AACA,SAAK,IAAIsE,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAGD,QAApB,EAA8BC,CAAC,IAAI,CAAnC,EAAsC;AACpC,WAAK,IAAIC,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAG5E,OAApB,EAA6B4E,CAAC,IAAI,CAAlC,EAAqC;AACnCH,QAAAA,cAAc,CAACnE,IAAf,CAAoB,KAAKf,KAAL,CAAWI,KAAX,CAAiBiF,CAAjB,EAAoBD,CAApB,CAApB;AACD;AACF,KARsC,CASvC;;;AACAF,IAAAA,cAAc,GAAGA,cAAc,CAACJ,MAAf,CAAsBnD,CAAC,IAAIA,CAAC,KAAK,IAAN,IAAcA,CAAC,KAAK2D,SAA/C,CAAjB;AACA,UAAMC,gBAAgB,GAAGL,cAAc,CAACM,GAAf,CAAmB,KAAKnE,qBAAxB,CAAzB;;AACA,SAAK,IAAIgE,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAGF,QAApB,EAA8BE,CAAC,IAAI,CAAnC,EAAsC;AACpC,YAAMI,KAAK,GAAGJ,CAAC,GAAG5E,OAAlB;AACA,YAAMa,IAAI,GAAG,KAAKtB,KAAL,CAAWI,KAAX,CAAiB,CAAjB,EAAoBiF,CAApB,CAAb,CAFoC,CAGpC;;AACAE,MAAAA,gBAAgB,CAACG,MAAjB,CAAwBD,KAAK,GAAGJ,CAAhC,EAAmC,CAAnC,EAAsC,KAAKhD,eAAL,CAAqBf,IAArB,CAAtC;AACD;;AACD,WAAO;AAAA;AACL;AACA;AAAK,MAAA,GAAG,EAAC;AAAT,MAFK,EAGL;AACA,OAAG,KAAKtB,KAAL,CAAWI,KAAX,CAAiBoF,GAAjB,CAAqB,CAACG,UAAD,EAAaF,KAAb,kBACtB5F,KAAK,CAAC+F,YAAN,CAAmB,KAAKrD,eAAL,CAAqBoD,UAAU,CAAC,CAAD,CAA/B,CAAnB,EAAwD;AAAEE,MAAAA,GAAG,iBAAUJ,KAAV;AAAL,KAAxD,CADC,CAJE,EAOL;AACA,OAAGF,gBAAgB,CAACC,GAAjB,CAAqB,CAACM,OAAD,EAAUL,KAAV,kBAAoB5F,KAAK,CAAC+F,YAAN,CAAmBE,OAAnB,EAA4B;AAAED,MAAAA,GAAG,iBAAUJ,KAAV;AAAL,KAA5B,CAAzC,CARE,CAAP;AAUD;;AAEDM,EAAAA,MAAM,GAAgB;AACpB,wBACE,oBAAC,OAAD,qBACE,oBAAC,IAAD;AACE,MAAA,OAAO,EAAE,KAAK/F,KAAL,CAAWI,KAAX,CAAiBU,MAD5B;AAEE,MAAA,IAAI,EAAE,KAAKd,KAAL,CAAWI,KAAX,CAAiB,CAAjB,EAAoBU,MAF5B;AAGE,MAAA,SAAS,EAAE,KAAKpC,KAAL,CAAWG,SAHxB;AAIE,MAAA,MAAM,EAAE,KAAKH,KAAL,CAAWI,MAJrB;AAKE,MAAA,GAAG,EAAEkH,EAAE,IAAI;AACT,aAAK5E,OAAL,GAAe4E,EAAf;AACD;AAPH,OASG,KAAKf,kBAAL,EATH,CADF,CADF;AAeD;;AA7TiF;;;AAA/DrF,gB,CAYZqG,Y,GAAmC;AACxC9F,EAAAA,SAAS,EAAE,EAD6B;AAExCsE,EAAAA,eAAe,EAAE,QAFuB;AAGxChE,EAAAA,OAAO,EAAE,CAH+B;AAIxCI,EAAAA,aAAa,EAAE,EAJyB;AAKxCN,EAAAA,SAAS,EAAE,IAAIS,IAAJ,EAL6B;AAMxCsB,EAAAA,UAAU,EAAE,IAN4B;AAOxCG,EAAAA,UAAU,EAAE,KAP4B;AAQxC5D,EAAAA,SAAS,EAAE,KAR6B;AASxCC,EAAAA,MAAM,EAAE,KATgC;AAUxCI,EAAAA,aAAa,EAAEgH,gBAAOC,IAVkB;AAWxChH,EAAAA,eAAe,EAAE+G,gBAAOE,QAXgB;AAYxChH,EAAAA,YAAY,EAAE8G,gBAAOG,SAZmB;AAaxClC,EAAAA,QAAQ,EAAE,MAAM,CAAE;AAbsB,C","sourcesContent":["import * as React from 'react'\nimport styled from 'styled-components'\n\n// Import only the methods we need from date-fns in order to keep build size small\nimport addMinutes from 'date-fns/add_minutes'\nimport addHours from 'date-fns/add_hours'\nimport addDays from 'date-fns/add_days'\nimport startOfDay from 'date-fns/start_of_day'\nimport isSameMinute from 'date-fns/is_same_minute'\nimport formatDate from 'date-fns/format'\n\nimport { Text, Subtitle } from './typography'\nimport colors from './colors'\nimport selectionSchemes, { SelectionSchemeType, SelectionType } from './selection-schemes'\nimport { getDay } from 'date-fns'\n\nconst Wrapper = styled.div`\n  display: flex;\n  align-items: center;\n  width: 100%;\n  user-select: none;\n`\n\nconst Grid = styled.div<{ columns: number; rows: number; columnGap: string; rowGap: string }>`\n  display: grid;\n  grid-template-columns: auto repeat(${props => props.columns}, 1fr);\n  grid-template-rows: auto repeat(${props => props.rows}, 1fr);\n  column-gap: ${props => props.columnGap};\n  row-gap: ${props => props.rowGap};\n  width: 100%;\n`\n\nexport const GridCell = styled.div`\n  place-self: stretch;\n  touch-action: none;\n`\n\nconst DateCell = styled.div<{\n  selected: boolean\n  selectedColor: string\n  unselectedColor: string\n  hoveredColor: string\n}>`\n  width: 100%;\n  height: 25px;\n  background-color: ${props => (props.selected ? props.selectedColor : props.unselectedColor)};\n\n  &:hover {\n    background-color: ${props => props.hoveredColor};\n  }\n`\n\nconst DateLabel = styled(Subtitle)`\n  @media (max-width: 699px) {\n    font-size: 12px;\n  }\n  margin: 0;\n  margin-bottom: 4px;\n`\n\nconst TimeText = styled(Text)`\n  @media (max-width: 699px) {\n    font-size: 10px;\n  }\n  text-align: right;\n  margin: 0;\n  margin-right: 4px;\n`\n\ntype PropsType = {\n  selection: Array<Date>\n  selectionScheme: SelectionSchemeType\n  onChange: (newSelection: Array<Date>) => void\n  startDate: Date\n  numDays: number\n  minutesChunks: number\n  dateFormat: string\n  timeFormat: string\n  columnGap: string\n  rowGap: string\n  unselectedColor: string\n  selectedColor: string\n  hoveredColor: string\n  renderDateCell?: (datetime: Date, selected: boolean, refSetter: (dateCellElement: HTMLElement) => void) => JSX.Element\n  renderTimeLabel?: (time: Date) => JSX.Element\n  renderDateLabel?: (date: Date) => JSX.Element\n}\n\ntype StateType = {\n  // In the case that a user is drag-selecting, we don't want to call this.props.onChange() until they have completed\n  // the drag-select. selectionDraft serves as a temporary copy during drag-selects.\n  selectionDraft: Array<Date>\n  selectionType: SelectionType | null\n  selectionStart: Date | null\n  isTouchDragging: boolean\n  dates: Array<Array<Date>>\n}\n\nexport const preventScroll = (e: TouchEvent) => {\n  e.preventDefault()\n}\n\nexport default class ScheduleSelector extends React.Component<PropsType, StateType> {\n  selectionSchemeHandlers: { [key: string]: (startDate: Date, endDate: Date, foo: Array<Array<Date>>) => Date[] }\n  cellToDate: Map<Element, Date> = new Map()\n  // documentMouseUpHandler: () => void = () => {}\n  // endSelection: () => void = () => {}\n  // handleTouchMoveEvent: (event: React.SyntheticTouchEvent<*>) => void\n  // handleTouchEndEvent: () => void\n  // handleMouseUpEvent: (date: Date) => void\n  // handleMouseEnterEvent: (date: Date) => void\n  // handleSelectionStartEvent: (date: Date) => void\n  gridRef: HTMLElement | null = null\n\n  static defaultProps: Partial<PropsType> = {\n    selection: [],\n    selectionScheme: 'square',\n    numDays: 7,\n    minutesChunks: 60,\n    startDate: new Date(),\n    timeFormat: 'ha',\n    dateFormat: 'M/D',\n    columnGap: '4px',\n    rowGap: '4px',\n    selectedColor: colors.blue,\n    unselectedColor: colors.paleBlue,\n    hoveredColor: colors.lightBlue,\n    onChange: () => {}\n  }\n\n  static getDerivedStateFromProps(props: PropsType, state: StateType): Partial<StateType> | null {\n    // As long as the user isn't in the process of selecting, allow prop changes to re-populate selection state\n    if (state.selectionStart == null) {\n      return {\n        selectionDraft: [...props.selection],\n        dates: ScheduleSelector.computeDatesMatrix(props)\n      }\n    }\n    return null\n  }\n\n  static computeDatesMatrix(props: PropsType): Array<Array<Date>> {\n    const startTime = startOfDay(props.startDate)\n    const dates: Array<Array<Date>> = []\n    for (let d = 0; d < props.numDays; d += 1) {\n      const currentDay: Date[] = []\n      let currentTime = startOfDay(addDays(startTime, d)) // Start at the beginning of the day\n      // Number of chunks in a day\n      const numChunks = (24 * 60) / props.minutesChunks\n      while (currentDay.length < numChunks) {\n        currentDay.push(new Date(currentTime))\n        currentTime = addMinutes(currentTime, props.minutesChunks) // Increment by 'minutes'\n      }\n      dates.push(currentDay)\n    }\n    return dates\n  }\n\n  constructor(props: PropsType) {\n    super(props)\n\n    this.state = {\n      selectionDraft: [...this.props.selection], // copy it over\n      selectionType: null,\n      selectionStart: null,\n      isTouchDragging: false,\n      dates: ScheduleSelector.computeDatesMatrix(props)\n    }\n\n    this.selectionSchemeHandlers = {\n      linear: selectionSchemes.linear,\n      square: selectionSchemes.square\n    }\n\n    this.endSelection = this.endSelection.bind(this)\n    this.handleMouseUpEvent = this.handleMouseUpEvent.bind(this)\n    this.handleMouseEnterEvent = this.handleMouseEnterEvent.bind(this)\n    this.handleTouchMoveEvent = this.handleTouchMoveEvent.bind(this)\n    this.handleTouchEndEvent = this.handleTouchEndEvent.bind(this)\n    this.handleSelectionStartEvent = this.handleSelectionStartEvent.bind(this)\n  }\n\n  componentDidMount() {\n    // We need to add the endSelection event listener to the document itself in order\n    // to catch the cases where the users ends their mouse-click somewhere besides\n    // the date cells (in which case none of the DateCell's onMouseUp handlers would fire)\n    //\n    // This isn't necessary for touch events since the `touchend` event fires on\n    // the element where the touch/drag started so it's always caught.\n    document.addEventListener('mouseup', this.endSelection)\n\n    // Prevent page scrolling when user is dragging on the date cells\n    this.cellToDate.forEach((value, dateCell) => {\n      if (dateCell && dateCell.addEventListener) {\n        // @ts-ignore\n        dateCell.addEventListener('touchmove', preventScroll, { passive: false })\n      }\n    })\n  }\n\n  componentWillUnmount() {\n    document.removeEventListener('mouseup', this.endSelection)\n    this.cellToDate.forEach((value, dateCell) => {\n      if (dateCell && dateCell.removeEventListener) {\n        // @ts-ignore\n        dateCell.removeEventListener('touchmove', preventScroll)\n      }\n    })\n  }\n\n  // Performs a lookup into this.cellToDate to retrieve the Date that corresponds to\n  // the cell where this touch event is right now. Note that this method will only work\n  // if the event is a `touchmove` event since it's the only one that has a `touches` list.\n  getTimeFromTouchEvent(event: React.TouchEvent<any>): Date | null {\n    const { touches } = event\n    if (!touches || touches.length === 0) return null\n    const { clientX, clientY } = touches[0]\n    const targetElement = document.elementFromPoint(clientX, clientY)\n    if (targetElement) {\n      const cellTime = this.cellToDate.get(targetElement)\n      return cellTime ?? null\n    }\n    return null\n  }\n\n  endSelection() {\n    this.props.onChange(this.state.selectionDraft)\n    this.setState({\n      selectionType: null,\n      selectionStart: null\n    })\n  }\n\n  // Given an ending Date, determines all the dates that should be selected in this draft\n  updateAvailabilityDraft(selectionEnd: Date | null, callback?: () => void) {\n    const { selectionType, selectionStart } = this.state\n\n    if (selectionType === null || selectionStart === null) return\n\n    let newSelection: Array<Date> = []\n    if (selectionStart && selectionEnd && selectionType) {\n      newSelection = this.selectionSchemeHandlers[this.props.selectionScheme](\n        selectionStart,\n        selectionEnd,\n        this.state.dates\n      )\n    }\n\n    let nextDraft = [...this.props.selection]\n    if (selectionType === 'add') {\n      nextDraft = Array.from(new Set([...nextDraft, ...newSelection]))\n    } else if (selectionType === 'remove') {\n      nextDraft = nextDraft.filter(a => !newSelection.find(b => isSameMinute(a, b)))\n    }\n\n    this.setState({ selectionDraft: nextDraft }, callback)\n  }\n\n  // Isomorphic (mouse and touch) handler since starting a selection works the same way for both classes of user input\n  handleSelectionStartEvent(startTime: Date) {\n    // Check if the startTime cell is selected/unselected to determine if this drag-select should\n    // add values or remove values\n    const timeSelected = this.props.selection.find(a => isSameMinute(a, startTime))\n    this.setState({\n      selectionType: timeSelected ? 'remove' : 'add',\n      selectionStart: startTime\n    })\n  }\n\n  handleMouseEnterEvent(time: Date) {\n    // Need to update selection draft on mouseup as well in order to catch the cases\n    // where the user just clicks on a single cell (because no mouseenter events fire\n    // in this scenario)\n    this.updateAvailabilityDraft(time)\n  }\n\n  handleMouseUpEvent(time: Date) {\n    this.updateAvailabilityDraft(time)\n    // Don't call this.endSelection() here because the document mouseup handler will do it\n  }\n\n  handleTouchMoveEvent(event: React.TouchEvent) {\n    this.setState({ isTouchDragging: true })\n    const cellTime = this.getTimeFromTouchEvent(event)\n    if (cellTime) {\n      this.updateAvailabilityDraft(cellTime)\n    }\n  }\n\n  handleTouchEndEvent() {\n    if (!this.state.isTouchDragging) {\n      // Going down this branch means the user tapped but didn't drag -- which\n      // means the availability draft hasn't yet been updated (since\n      // handleTouchMoveEvent was never called) so we need to do it now\n      this.updateAvailabilityDraft(null, () => {\n        this.endSelection()\n      })\n    } else {\n      this.endSelection()\n    }\n    this.setState({ isTouchDragging: false })\n  }\n\n  renderDateCellWrapper = (time: Date): JSX.Element => {\n    const startHandler = () => {\n      this.handleSelectionStartEvent(time)\n    }\n\n    const selected = Boolean(this.state.selectionDraft.find(a => isSameMinute(a, time)))\n\n    return (\n      <GridCell\n        className=\"rgdp__grid-cell\"\n        role=\"presentation\"\n        key={time.toISOString()}\n        // Mouse handlers\n        onMouseDown={startHandler}\n        onMouseEnter={() => {\n          this.handleMouseEnterEvent(time)\n        }}\n        onMouseUp={() => {\n          this.handleMouseUpEvent(time)\n        }}\n        // Touch handlers\n        // Since touch events fire on the event where the touch-drag started, there's no point in passing\n        // in the time parameter, instead these handlers will do their job using the default Event\n        // parameters\n        onTouchStart={startHandler}\n        onTouchMove={this.handleTouchMoveEvent}\n        onTouchEnd={this.handleTouchEndEvent}\n      >\n        {this.renderDateCell(time, selected)}\n      </GridCell>\n    )\n  }\n\n  renderDateCell = (time: Date, selected: boolean): JSX.Element => {\n    const refSetter = (dateCell: HTMLElement | null) => {\n      if (dateCell) {\n        this.cellToDate.set(dateCell, time)\n      }\n    }\n    if (this.props.renderDateCell) {\n      return this.props.renderDateCell(time, selected, refSetter)\n    } else {\n      return (\n        <DateCell\n          selected={selected}\n          ref={refSetter}\n          selectedColor={this.props.selectedColor}\n          unselectedColor={this.props.unselectedColor}\n          hoveredColor={this.props.hoveredColor}\n        />\n      )\n    }\n  }\n\n  renderTimeLabel = (time: Date): JSX.Element => {\n    if (this.props.renderTimeLabel) {\n      return this.props.renderTimeLabel(time)\n    } else {\n      return <TimeText>{formatDate(time, this.props.timeFormat)}</TimeText>\n    }\n  }\n\n  renderDateLabel = (date: Date): JSX.Element => {\n    if (this.props.renderDateLabel) {\n      return this.props.renderDateLabel(date)\n    } else {\n      return <DateLabel>{formatDate(date, this.props.dateFormat)}</DateLabel>\n    }\n  }\n\n  renderFullDateGrid(): Array<JSX.Element> {\n    let flattenedDates: Date[] = []\n    const numDays = this.state.dates.length\n    const numTimes = this.state.dates[0].length\n    for (let j = 0; j < numTimes; j += 1) {\n      for (let i = 0; i < numDays; i += 1) {\n        flattenedDates.push(this.state.dates[i][j])\n      }\n    }\n    // Filter out any null dates or undefined dates\n    flattenedDates = flattenedDates.filter(a => a !== null && a !== undefined)\n    const dateGridElements = flattenedDates.map(this.renderDateCellWrapper)\n    for (let i = 0; i < numTimes; i += 1) {\n      const index = i * numDays\n      const time = this.state.dates[0][i]\n      // Inject the time label at the start of every row\n      dateGridElements.splice(index + i, 0, this.renderTimeLabel(time))\n    }\n    return [\n      // Empty top left corner\n      <div key=\"topleft\" />,\n      // Top row of dates\n      ...this.state.dates.map((dayOfTimes, index) =>\n        React.cloneElement(this.renderDateLabel(dayOfTimes[0]), { key: `date-${index}` })\n      ),\n      // Every row after that\n      ...dateGridElements.map((element, index) => React.cloneElement(element, { key: `time-${index}` }))\n    ]\n  }\n\n  render(): JSX.Element {\n    return (\n      <Wrapper>\n        <Grid\n          columns={this.state.dates.length}\n          rows={this.state.dates[0].length}\n          columnGap={this.props.columnGap}\n          rowGap={this.props.rowGap}\n          ref={el => {\n            this.gridRef = el\n          }}\n        >\n          {this.renderFullDateGrid()}\n        </Grid>\n      </Wrapper>\n    )\n  }\n}\n"]}