import PropTypes from 'prop-types'; import React, { Component } from 'react'; import { connect } from 'react-redux'; import { createSelector } from 'reselect'; import * as commandNames from 'Commands/commandNames'; import * as calendarActions from 'Store/Actions/calendarActions'; import { clearEpisodeFiles, fetchEpisodeFiles } from 'Store/Actions/episodeFileActions'; import { clearQueueDetails, fetchQueueDetails } from 'Store/Actions/queueActions'; import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector'; import hasDifferentItems from 'Utilities/Object/hasDifferentItems'; import selectUniqueIds from 'Utilities/Object/selectUniqueIds'; import { registerPagePopulator, unregisterPagePopulator } from 'Utilities/pagePopulator'; import Calendar from './Calendar'; const UPDATE_DELAY = 3600000; // 1 hour function createMapStateToProps() { return createSelector( (state) => state.calendar, (state) => state.settings.ui.item.firstDayOfWeek, createCommandExecutingSelector(commandNames.REFRESH_SERIES), (calendar, firstDayOfWeek, isRefreshingSeries) => { return { ...calendar, isRefreshingSeries, firstDayOfWeek }; } ); } const mapDispatchToProps = { ...calendarActions, fetchEpisodeFiles, clearEpisodeFiles, fetchQueueDetails, clearQueueDetails }; class CalendarConnector extends Component { // // Lifecycle constructor(props, context) { super(props, context); this.updateTimeoutId = null; } componentDidMount() { const { useCurrentPage, fetchCalendar, gotoCalendarToday } = this.props; registerPagePopulator(this.repopulate, ['episodeFileUpdated', 'episodeFileDeleted']); if (useCurrentPage) { fetchCalendar(); } else { gotoCalendarToday(); } this.scheduleUpdate(); } componentDidUpdate(prevProps) { const { items, time, view, isRefreshingSeries, firstDayOfWeek } = this.props; if (hasDifferentItems(prevProps.items, items)) { const episodeIds = selectUniqueIds(items, 'id'); const episodeFileIds = selectUniqueIds(items, 'episodeFileId'); if (items.length) { this.props.fetchQueueDetails({ episodeIds }); } if (episodeFileIds.length) { this.props.fetchEpisodeFiles({ episodeFileIds }); } } if (prevProps.time !== time) { this.scheduleUpdate(); } if (prevProps.firstDayOfWeek !== firstDayOfWeek) { this.props.fetchCalendar({ time, view }); } if (prevProps.isRefreshingSeries && !isRefreshingSeries) { this.props.fetchCalendar({ time, view }); } } componentWillUnmount() { unregisterPagePopulator(this.repopulate); this.props.clearCalendar(); this.props.clearQueueDetails(); this.props.clearEpisodeFiles(); this.clearUpdateTimeout(); } // // Control repopulate = () => { const { time, view } = this.props; this.props.fetchQueueDetails({ time, view }); this.props.fetchCalendar({ time, view }); }; scheduleUpdate = () => { this.clearUpdateTimeout(); this.updateTimeoutId = setTimeout(this.updateCalendar, UPDATE_DELAY); }; clearUpdateTimeout = () => { if (this.updateTimeoutId) { clearTimeout(this.updateTimeoutId); } }; updateCalendar = () => { this.props.gotoCalendarToday(); this.scheduleUpdate(); }; // // Listeners onCalendarViewChange = (view) => { this.props.setCalendarView({ view }); }; onTodayPress = () => { this.props.gotoCalendarToday(); }; onPreviousPress = () => { this.props.gotoCalendarPreviousRange(); }; onNextPress = () => { this.props.gotoCalendarNextRange(); }; // // Render render() { return ( ); } } CalendarConnector.propTypes = { useCurrentPage: PropTypes.bool.isRequired, time: PropTypes.string, view: PropTypes.string.isRequired, firstDayOfWeek: PropTypes.number.isRequired, items: PropTypes.arrayOf(PropTypes.object).isRequired, isRefreshingSeries: PropTypes.bool.isRequired, setCalendarView: PropTypes.func.isRequired, gotoCalendarToday: PropTypes.func.isRequired, gotoCalendarPreviousRange: PropTypes.func.isRequired, gotoCalendarNextRange: PropTypes.func.isRequired, clearCalendar: PropTypes.func.isRequired, fetchCalendar: PropTypes.func.isRequired, fetchEpisodeFiles: PropTypes.func.isRequired, clearEpisodeFiles: PropTypes.func.isRequired, fetchQueueDetails: PropTypes.func.isRequired, clearQueueDetails: PropTypes.func.isRequired }; export default connect(createMapStateToProps, mapDispatchToProps)(CalendarConnector);