2022-01-20 13:28:49 -08:00
|
|
|
|
|
|
|
import PropTypes from 'prop-types';
|
|
|
|
import React from 'react';
|
2022-01-21 14:40:39 -08:00
|
|
|
import { HotKeys } from 'react-hotkeys';
|
2022-01-20 13:28:49 -08:00
|
|
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
|
|
import ImmutablePureComponent from 'react-immutable-pure-component';
|
|
|
|
import { injectIntl, FormattedMessage } from 'react-intl';
|
|
|
|
import { connect } from 'react-redux';
|
|
|
|
import { Link } from 'react-router-dom';
|
|
|
|
|
|
|
|
import { fetchBirthdayReminders } from 'soapbox/actions/accounts';
|
2022-02-02 05:33:12 -08:00
|
|
|
import { openModal } from 'soapbox/actions/modals';
|
2022-01-20 13:28:49 -08:00
|
|
|
import Icon from 'soapbox/components/icon';
|
2022-04-04 12:58:27 -07:00
|
|
|
import { HStack, Text } from 'soapbox/components/ui';
|
2022-01-20 13:28:49 -08:00
|
|
|
import { makeGetAccount } from 'soapbox/selectors';
|
|
|
|
|
|
|
|
const mapStateToProps = (state, props) => {
|
|
|
|
const me = state.get('me');
|
|
|
|
const getAccount = makeGetAccount();
|
|
|
|
|
|
|
|
const birthdays = state.getIn(['user_lists', 'birthday_reminders', me]);
|
|
|
|
|
2022-02-18 18:04:03 -08:00
|
|
|
if (birthdays?.size > 0) {
|
2022-01-20 13:28:49 -08:00
|
|
|
return {
|
|
|
|
birthdays,
|
|
|
|
account: getAccount(state, birthdays.first()),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
birthdays,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export default @connect(mapStateToProps)
|
|
|
|
@injectIntl
|
|
|
|
class BirthdayReminders extends ImmutablePureComponent {
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
birthdays: ImmutablePropTypes.orderedSet,
|
|
|
|
intl: PropTypes.object.isRequired,
|
|
|
|
dispatch: PropTypes.func.isRequired,
|
2022-01-24 14:51:22 -08:00
|
|
|
onMoveDown: PropTypes.func,
|
2022-01-20 13:28:49 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
const { dispatch } = this.props;
|
|
|
|
|
|
|
|
const date = new Date();
|
|
|
|
|
|
|
|
const day = date.getDate();
|
|
|
|
const month = date.getMonth() + 1;
|
|
|
|
|
|
|
|
dispatch(fetchBirthdayReminders(day, month));
|
|
|
|
}
|
|
|
|
|
2022-01-21 14:40:39 -08:00
|
|
|
getHandlers() {
|
|
|
|
return {
|
|
|
|
open: this.handleOpenBirthdaysModal,
|
2022-01-24 14:51:22 -08:00
|
|
|
moveDown: this.props.onMoveDown,
|
2022-01-21 14:40:39 -08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-01-20 13:28:49 -08:00
|
|
|
handleOpenBirthdaysModal = () => {
|
|
|
|
const { dispatch } = this.props;
|
|
|
|
|
|
|
|
dispatch(openModal('BIRTHDAYS'));
|
|
|
|
}
|
|
|
|
|
|
|
|
renderMessage() {
|
|
|
|
const { birthdays, account } = this.props;
|
|
|
|
|
|
|
|
const link = (
|
|
|
|
<bdi>
|
|
|
|
<Link
|
2022-04-04 12:58:27 -07:00
|
|
|
className='text-gray-800 dark:text-gray-200 font-bold hover:underline'
|
2022-01-20 13:28:49 -08:00
|
|
|
title={account.get('acct')}
|
|
|
|
to={`/@${account.get('acct')}`}
|
|
|
|
dangerouslySetInnerHTML={{ __html: account.get('display_name_html') }}
|
|
|
|
/>
|
|
|
|
</bdi>
|
|
|
|
);
|
|
|
|
|
|
|
|
if (birthdays.size === 1) {
|
|
|
|
return <FormattedMessage id='notification.birthday' defaultMessage='{name} has birthday today' values={{ name: link }} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<FormattedMessage
|
|
|
|
id='notification.birthday_plural'
|
|
|
|
defaultMessage='{name} and {more} have birthday today'
|
|
|
|
values={{
|
|
|
|
name: link,
|
|
|
|
more: (
|
|
|
|
<span type='button' role='presentation' onClick={this.handleOpenBirthdaysModal}>
|
|
|
|
<FormattedMessage
|
|
|
|
id='notification.birthday.more'
|
|
|
|
defaultMessage='{count} more {count, plural, one {friend} other {friends}}'
|
|
|
|
values={{ count: birthdays.size - 1 }}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
),
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-24 14:51:22 -08:00
|
|
|
renderMessageForScreenReader = () => {
|
|
|
|
const { intl, birthdays, account } = this.props;
|
|
|
|
|
|
|
|
if (birthdays.size === 1) {
|
|
|
|
return intl.formatMessage({ id: 'notification.birthday', defaultMessage: '{name} has birthday today' }, { name: account.get('display_name') });
|
|
|
|
}
|
|
|
|
|
|
|
|
return intl.formatMessage(
|
|
|
|
{
|
|
|
|
id: 'notification.birthday_plural',
|
|
|
|
defaultMessage: '{name} and {more} have birthday today',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: account.get('display_name'),
|
|
|
|
more: intl.formatMessage(
|
|
|
|
{
|
|
|
|
id: 'notification.birthday.more',
|
|
|
|
defaultMessage: '{count} more {count, plural, one {friend} other {friends}}',
|
|
|
|
},
|
|
|
|
{ count: birthdays.size - 1 },
|
|
|
|
),
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-20 13:28:49 -08:00
|
|
|
render() {
|
|
|
|
const { birthdays } = this.props;
|
|
|
|
|
|
|
|
if (!birthdays || birthdays.size === 0) return null;
|
|
|
|
|
|
|
|
return (
|
2022-01-21 14:40:39 -08:00
|
|
|
<HotKeys handlers={this.getHandlers()}>
|
2022-01-24 14:51:22 -08:00
|
|
|
<div className='notification notification-birthday focusable' tabIndex='0' title={this.renderMessageForScreenReader()}>
|
2022-04-04 12:58:27 -07:00
|
|
|
<div className='p-4 focusable'>
|
|
|
|
<HStack alignItems='center' space={1.5}>
|
|
|
|
<Icon
|
|
|
|
src={require('@tabler/icons/icons/ballon.svg')}
|
|
|
|
className='text-primary-600'
|
|
|
|
/>
|
2022-01-21 14:40:39 -08:00
|
|
|
|
2022-04-04 12:58:27 -07:00
|
|
|
<Text
|
|
|
|
theme='muted'
|
|
|
|
size='sm'
|
|
|
|
>
|
|
|
|
{this.renderMessage()}
|
|
|
|
</Text>
|
|
|
|
</HStack>
|
2022-01-20 13:28:49 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2022-01-21 14:40:39 -08:00
|
|
|
</HotKeys>
|
2022-01-20 13:28:49 -08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-02-18 18:04:03 -08:00
|
|
|
}
|