2021-06-18 09:04:31 -07:00
|
|
|
'use strict';
|
|
|
|
|
2022-01-10 14:17:52 -08:00
|
|
|
import classNames from 'classnames';
|
2021-06-18 09:04:31 -07:00
|
|
|
import PropTypes from 'prop-types';
|
2022-01-10 14:17:52 -08:00
|
|
|
import React from 'react';
|
|
|
|
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
|
|
|
import { connect } from 'react-redux';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2022-05-12 05:17:38 -07:00
|
|
|
import { setSchedule, removeSchedule } from 'soapbox/actions/compose';
|
2022-01-10 14:01:24 -08:00
|
|
|
import IconButton from 'soapbox/components/icon_button';
|
2022-05-12 05:17:38 -07:00
|
|
|
import { HStack, Stack, Text } from 'soapbox/components/ui';
|
2022-03-09 13:57:37 -08:00
|
|
|
import BundleContainer from 'soapbox/features/ui/containers/bundle_container';
|
|
|
|
import { DatePicker } from 'soapbox/features/ui/util/async-components';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2021-06-18 09:04:31 -07:00
|
|
|
const messages = defineMessages({
|
|
|
|
schedule: { id: 'schedule.post_time', defaultMessage: 'Post Date/Time' },
|
2021-06-27 22:06:04 -07:00
|
|
|
remove: { id: 'schedule.remove', defaultMessage: 'Remove schedule' },
|
2021-06-18 09:04:31 -07:00
|
|
|
});
|
|
|
|
|
2021-09-11 16:00:48 -07:00
|
|
|
const mapStateToProps = state => ({
|
|
|
|
active: state.getIn(['compose', 'schedule']) ? true : false,
|
2021-06-27 22:36:44 -07:00
|
|
|
scheduledAt: state.getIn(['compose', 'schedule']),
|
2021-06-27 22:06:04 -07:00
|
|
|
});
|
|
|
|
|
2021-09-11 16:00:48 -07:00
|
|
|
const mapDispatchToProps = dispatch => ({
|
|
|
|
onSchedule(date) {
|
|
|
|
dispatch(setSchedule(date));
|
|
|
|
},
|
|
|
|
|
|
|
|
onRemoveSchedule(date) {
|
|
|
|
dispatch(removeSchedule());
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
export default @connect(mapStateToProps, mapDispatchToProps)
|
2021-06-27 22:06:04 -07:00
|
|
|
@injectIntl
|
2021-06-18 09:04:31 -07:00
|
|
|
class ScheduleForm extends React.Component {
|
|
|
|
|
|
|
|
static propTypes = {
|
2021-06-27 22:36:44 -07:00
|
|
|
scheduledAt: PropTypes.instanceOf(Date),
|
2021-06-18 09:04:31 -07:00
|
|
|
intl: PropTypes.object.isRequired,
|
|
|
|
onSchedule: PropTypes.func.isRequired,
|
2021-09-11 16:00:48 -07:00
|
|
|
onRemoveSchedule: PropTypes.func.isRequired,
|
2021-06-27 22:06:04 -07:00
|
|
|
dispatch: PropTypes.func,
|
2021-06-18 09:04:31 -07:00
|
|
|
active: PropTypes.bool,
|
|
|
|
};
|
|
|
|
|
2021-06-30 19:02:53 -07:00
|
|
|
state = {
|
|
|
|
initialized: false,
|
|
|
|
}
|
|
|
|
|
2021-06-27 22:36:44 -07:00
|
|
|
setSchedule = date => {
|
2021-06-18 09:04:31 -07:00
|
|
|
this.props.onSchedule(date);
|
|
|
|
}
|
|
|
|
|
2021-06-30 19:02:53 -07:00
|
|
|
setRef = c => {
|
|
|
|
this.datePicker = c;
|
|
|
|
}
|
2021-06-18 09:04:31 -07:00
|
|
|
|
2021-06-30 19:02:53 -07:00
|
|
|
openDatePicker = () => {
|
|
|
|
if (!this.datePicker) return;
|
|
|
|
this.datePicker.setOpen(true);
|
2021-06-18 09:04:31 -07:00
|
|
|
}
|
|
|
|
|
2021-06-21 11:39:22 -07:00
|
|
|
isCurrentOrFutureDate(date) {
|
2021-06-18 14:51:14 -07:00
|
|
|
return date && new Date().setHours(0, 0, 0, 0) <= new Date(date).setHours(0, 0, 0, 0);
|
2021-06-18 09:04:31 -07:00
|
|
|
}
|
|
|
|
|
2021-06-21 11:39:22 -07:00
|
|
|
isFiveMinutesFromNow(time) {
|
2021-06-18 09:04:31 -07:00
|
|
|
const fiveMinutesFromNow = new Date(new Date().getTime() + 300000); // now, plus five minutes (Pleroma won't schedule posts )
|
|
|
|
const selectedDate = new Date(time);
|
|
|
|
|
|
|
|
return fiveMinutesFromNow.getTime() < selectedDate.getTime();
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2021-06-18 09:04:31 -07:00
|
|
|
|
2021-06-27 22:06:04 -07:00
|
|
|
handleRemove = e => {
|
2021-09-11 16:00:48 -07:00
|
|
|
this.props.onRemoveSchedule();
|
2021-06-27 22:06:04 -07:00
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
|
2021-06-30 19:02:53 -07:00
|
|
|
initialize = () => {
|
|
|
|
const { initialized } = this.state;
|
|
|
|
|
|
|
|
if (!initialized && this.datePicker) {
|
|
|
|
this.openDatePicker();
|
|
|
|
this.setState({ initialized: true });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidUpdate() {
|
|
|
|
this.initialize();
|
|
|
|
}
|
|
|
|
|
2021-06-18 09:04:31 -07:00
|
|
|
render() {
|
2021-06-27 22:36:44 -07:00
|
|
|
if (!this.props.active) {
|
2021-06-18 09:04:31 -07:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-06-27 22:36:44 -07:00
|
|
|
const { intl, scheduledAt } = this.props;
|
2021-06-18 09:04:31 -07:00
|
|
|
|
|
|
|
return (
|
2022-05-12 05:17:38 -07:00
|
|
|
<Stack>
|
|
|
|
<Text style='muted'>
|
2021-06-27 22:06:04 -07:00
|
|
|
<FormattedMessage id='datepicker.hint' defaultMessage='Scheduled to post at…' />
|
2022-05-12 05:17:38 -07:00
|
|
|
</Text>
|
|
|
|
<HStack className='mb-2' space={2} alignItems='center'>
|
2022-03-09 13:57:37 -08:00
|
|
|
<BundleContainer fetchComponent={DatePicker}>
|
|
|
|
{Component => (<Component
|
|
|
|
selected={scheduledAt}
|
|
|
|
showTimeSelect
|
|
|
|
dateFormat='MMMM d, yyyy h:mm aa'
|
|
|
|
timeIntervals={15}
|
|
|
|
wrapperClassName='react-datepicker-wrapper'
|
|
|
|
onChange={this.setSchedule}
|
|
|
|
placeholderText={this.props.intl.formatMessage(messages.schedule)}
|
|
|
|
filterDate={this.isCurrentOrFutureDate}
|
|
|
|
filterTime={this.isFiveMinutesFromNow}
|
|
|
|
ref={this.setRef}
|
2022-05-12 05:17:38 -07:00
|
|
|
className={classNames({
|
|
|
|
'has-error': !this.isFiveMinutesFromNow(scheduledAt),
|
|
|
|
})}
|
2022-03-09 13:57:37 -08:00
|
|
|
/>)}
|
|
|
|
</BundleContainer>
|
2022-05-12 05:17:38 -07:00
|
|
|
<IconButton
|
|
|
|
iconClassName='w-4 h-4'
|
|
|
|
className='bg-transparent text-gray-400 hover:text-gray-600'
|
|
|
|
src={require('@tabler/icons/icons/x.svg')}
|
|
|
|
onClick={this.handleRemove}
|
|
|
|
title={intl.formatMessage(messages.remove)}
|
|
|
|
/>
|
|
|
|
</HStack>
|
|
|
|
</Stack>
|
2021-06-18 09:04:31 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|