parent
21b04af524
commit
d0aad1ac85
@ -1,137 +1,241 @@ |
|||||||
|
/* |
||||||
|
|
||||||
|
`<ComposeAdvancedOptions>` |
||||||
|
========================== |
||||||
|
|
||||||
|
> For more information on the contents of this file, please contact: |
||||||
|
> |
||||||
|
> - surinna [@srn@dev.glitch.social] |
||||||
|
|
||||||
|
This adds an advanced options dropdown to the toot compose box, for |
||||||
|
toggles that don't necessarily fit elsewhere. |
||||||
|
|
||||||
|
__Props:__ |
||||||
|
|
||||||
|
- __`values` (`ImmutablePropTypes.contains(…).isRequired`) :__ |
||||||
|
An Immutable map with the following values: |
||||||
|
|
||||||
|
- __`do_not_federate` (`PropTypes.bool.isRequired`) :__ |
||||||
|
Specifies whether or not to federate the status. |
||||||
|
|
||||||
|
- __`onChange` (`PropTypes.func.isRequired`) :__ |
||||||
|
The function to call when a toggle is changed. We pass this from |
||||||
|
our container to the toggle. |
||||||
|
|
||||||
|
- __`intl` (`PropTypes.object.isRequired`) :__ |
||||||
|
Our internationalization object, inserted by `@injectIntl`. |
||||||
|
|
||||||
|
__State:__ |
||||||
|
|
||||||
|
- __`open` :__ |
||||||
|
This tells whether the dropdown is currently open or closed. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Imports: |
||||||
|
-------- |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
// Package imports //
|
// Package imports //
|
||||||
import React from 'react'; |
import React from 'react'; |
||||||
import PropTypes from 'prop-types'; |
import PropTypes from 'prop-types'; |
||||||
import ImmutablePropTypes from 'react-immutable-proptypes'; |
import ImmutablePropTypes from 'react-immutable-proptypes'; |
||||||
import Toggle from 'react-toggle'; |
|
||||||
import { injectIntl, defineMessages } from 'react-intl'; |
import { injectIntl, defineMessages } from 'react-intl'; |
||||||
|
|
||||||
// Mastodon imports //
|
// Mastodon imports //
|
||||||
import IconButton from '../../../../mastodon/components/icon_button'; |
import IconButton from '../../../../mastodon/components/icon_button'; |
||||||
|
|
||||||
|
// Our imports //
|
||||||
|
import ComposeAdvancedOptionsToggle from './toggle'; |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Inital setup: |
||||||
|
------------- |
||||||
|
|
||||||
|
The `messages` constant is used to define any messages that we need |
||||||
|
from inside props. These are the various titles and labels on our |
||||||
|
toggles. |
||||||
|
|
||||||
|
`iconStyle` styles the icon used for the dropdown button. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
const messages = defineMessages({ |
const messages = defineMessages({ |
||||||
local_only_short: { id: 'advanced-options.local-only.short', defaultMessage: 'Local-only' }, |
local_only_short : |
||||||
local_only_long: { id: 'advanced-options.local-only.long', defaultMessage: 'Do not post to other instances' }, |
{ id: 'advanced-options.local-only.short', defaultMessage: 'Local-only' }, |
||||||
advanced_options_icon_title: { id: 'advanced_options.icon_title', defaultMessage: 'Advanced options' }, |
local_only_long : |
||||||
|
{ id: 'advanced-options.local-only.long', defaultMessage: 'Do not post to other instances' }, |
||||||
|
advanced_options_icon_title : |
||||||
|
{ id: 'advanced_options.icon_title', defaultMessage: 'Advanced options' }, |
||||||
}); |
}); |
||||||
|
|
||||||
const iconStyle = { |
const iconStyle = { |
||||||
height: null, |
height : null, |
||||||
lineHeight: '27px', |
lineHeight : '27px', |
||||||
}; |
}; |
||||||
|
|
||||||
class AdvancedOptionToggle extends React.PureComponent { |
/* |
||||||
|
|
||||||
static propTypes = { |
|
||||||
onChange: PropTypes.func.isRequired, |
|
||||||
active: PropTypes.bool.isRequired, |
|
||||||
name: PropTypes.string.isRequired, |
|
||||||
shortText: PropTypes.string.isRequired, |
|
||||||
longText: PropTypes.string.isRequired, |
|
||||||
} |
|
||||||
|
|
||||||
onToggle = () => { |
|
||||||
this.props.onChange(this.props.name); |
|
||||||
} |
|
||||||
|
|
||||||
render() { |
Implementation: |
||||||
const { active, shortText, longText } = this.props; |
--------------- |
||||||
return ( |
|
||||||
<div role='button' tabIndex='0' className='advanced-options-dropdown__option' onClick={this.onToggle}> |
|
||||||
<div className='advanced-options-dropdown__option__toggle'> |
|
||||||
<Toggle checked={active} onChange={this.onToggle} /> |
|
||||||
</div> |
|
||||||
<div className='advanced-options-dropdown__option__content'> |
|
||||||
<strong>{shortText}</strong> |
|
||||||
{longText} |
|
||||||
</div> |
|
||||||
</div> |
|
||||||
); |
|
||||||
} |
|
||||||
|
|
||||||
} |
*/ |
||||||
|
|
||||||
@injectIntl |
@injectIntl |
||||||
export default class ComposeAdvancedOptions extends React.PureComponent { |
export default class ComposeAdvancedOptions extends React.PureComponent { |
||||||
|
|
||||||
static propTypes = { |
static propTypes = { |
||||||
values: ImmutablePropTypes.contains({ |
values : ImmutablePropTypes.contains({ |
||||||
do_not_federate: PropTypes.bool.isRequired, |
do_not_federate : PropTypes.bool.isRequired, |
||||||
}).isRequired, |
}).isRequired, |
||||||
onChange: PropTypes.func.isRequired, |
onChange : PropTypes.func.isRequired, |
||||||
intl: PropTypes.object.isRequired, |
intl : PropTypes.object.isRequired, |
||||||
}; |
}; |
||||||
|
|
||||||
|
state = { |
||||||
|
open: false, |
||||||
|
}; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `onToggleDropdown()` |
||||||
|
|
||||||
|
This function toggles the opening and closing of the advanced options |
||||||
|
dropdown. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
onToggleDropdown = () => { |
onToggleDropdown = () => { |
||||||
this.setState({ open: !this.state.open }); |
this.setState({ open: !this.state.open }); |
||||||
}; |
}; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `onGlobalClick(e)` |
||||||
|
|
||||||
|
This function closes the advanced options dropdown if you click |
||||||
|
anywhere else on the screen. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
onGlobalClick = (e) => { |
onGlobalClick = (e) => { |
||||||
if (e.target !== this.node && !this.node.contains(e.target) && this.state.open) { |
if (e.target !== this.node && !this.node.contains(e.target) && this.state.open) { |
||||||
this.setState({ open: false }); |
this.setState({ open: false }); |
||||||
} |
} |
||||||
} |
} |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `componentDidMount()`, `componentWillUnmount()` |
||||||
|
|
||||||
|
This function closes the advanced options dropdown if you click |
||||||
|
anywhere else on the screen. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
componentDidMount () { |
componentDidMount () { |
||||||
window.addEventListener('click', this.onGlobalClick); |
window.addEventListener('click', this.onGlobalClick); |
||||||
window.addEventListener('touchstart', this.onGlobalClick); |
window.addEventListener('touchstart', this.onGlobalClick); |
||||||
} |
} |
||||||
|
|
||||||
componentWillUnmount () { |
componentWillUnmount () { |
||||||
window.removeEventListener('click', this.onGlobalClick); |
window.removeEventListener('click', this.onGlobalClick); |
||||||
window.removeEventListener('touchstart', this.onGlobalClick); |
window.removeEventListener('touchstart', this.onGlobalClick); |
||||||
} |
} |
||||||
|
|
||||||
state = { |
/* |
||||||
open: false, |
|
||||||
}; |
|
||||||
|
|
||||||
handleClick = (e) => { |
### `setRef(c)` |
||||||
const option = e.currentTarget.getAttribute('data-index'); |
|
||||||
e.preventDefault(); |
`setRef()` stores a reference to the dropdown's `<div> in `this.node`.
|
||||||
this.props.onChange(option); |
|
||||||
} |
*/ |
||||||
|
|
||||||
setRef = (c) => { |
setRef = (c) => { |
||||||
this.node = c; |
this.node = c; |
||||||
} |
} |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `render()` |
||||||
|
|
||||||
|
`render()` actually puts our component on the screen. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
render () { |
render () { |
||||||
const { open } = this.state; |
const { open } = this.state; |
||||||
const { intl, values } = this.props; |
const { intl, values } = this.props; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
The `options` array provides all of the available advanced options |
||||||
|
alongside their icon, text, and name. |
||||||
|
|
||||||
|
*/ |
||||||
const options = [ |
const options = [ |
||||||
{ icon: 'wifi', shortText: messages.local_only_short, longText: messages.local_only_long, key: 'do_not_federate' }, |
{ icon: 'wifi', shortText: messages.local_only_short, longText: messages.local_only_long, name: 'do_not_federate' }, |
||||||
]; |
]; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
`anyEnabled` tells us if any of our advanced options have been enabled. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
const anyEnabled = values.some((enabled) => enabled); |
const anyEnabled = values.some((enabled) => enabled); |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
`optionElems` takes our `options` and creates |
||||||
|
`<ComposeAdvancedOptionsToggle>`s out of them. We use the `name` of the |
||||||
|
toggle as its `key` so that React can keep track of it. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
const optionElems = options.map((option) => { |
const optionElems = options.map((option) => { |
||||||
return ( |
return ( |
||||||
<AdvancedOptionToggle |
<ComposeAdvancedOptionsToggle |
||||||
onChange={this.props.onChange} |
onChange={this.props.onChange} |
||||||
active={values.get(option.key)} |
active={values.get(option.name)} |
||||||
key={option.key} |
key={option.name} |
||||||
name={option.key} |
name={option.name} |
||||||
shortText={intl.formatMessage(option.shortText)} |
shortText={intl.formatMessage(option.shortText)} |
||||||
longText={intl.formatMessage(option.longText)} |
longText={intl.formatMessage(option.longText)} |
||||||
/> |
/> |
||||||
); |
); |
||||||
}); |
}); |
||||||
|
|
||||||
return (<div ref={this.setRef} className={`advanced-options-dropdown ${open ? 'open' : ''} ${anyEnabled ? 'active' : ''} `}> |
/* |
||||||
<div className='advanced-options-dropdown__value'> |
|
||||||
<IconButton |
Finally, we can render our component. |
||||||
className='advanced-options-dropdown__value' |
|
||||||
title={intl.formatMessage(messages.advanced_options_icon_title)} |
*/ |
||||||
icon='ellipsis-h' active={open || anyEnabled} |
|
||||||
size={18} |
return ( |
||||||
style={iconStyle} |
<div ref={this.setRef} className={`advanced-options-dropdown ${open ? 'open' : ''} ${anyEnabled ? 'active' : ''} `}> |
||||||
onClick={this.onToggleDropdown} |
<div className='advanced-options-dropdown__value'> |
||||||
/> |
<IconButton |
||||||
</div> |
className='advanced-options-dropdown__value' |
||||||
<div className='advanced-options-dropdown__dropdown'> |
title={intl.formatMessage(messages.advanced_options_icon_title)} |
||||||
{optionElems} |
icon='ellipsis-h' active={open || anyEnabled} |
||||||
|
size={18} |
||||||
|
style={iconStyle} |
||||||
|
onClick={this.onToggleDropdown} |
||||||
|
/> |
||||||
|
</div> |
||||||
|
<div className='advanced-options-dropdown__dropdown'> |
||||||
|
{optionElems} |
||||||
|
</div> |
||||||
</div> |
</div> |
||||||
</div>); |
); |
||||||
} |
} |
||||||
|
|
||||||
} |
} |
||||||
|
@ -0,0 +1,103 @@ |
|||||||
|
/* |
||||||
|
|
||||||
|
`<ComposeAdvancedOptionsToggle>` |
||||||
|
================================ |
||||||
|
|
||||||
|
> For more information on the contents of this file, please contact: |
||||||
|
> |
||||||
|
> - surinna [@srn@dev.glitch.social] |
||||||
|
|
||||||
|
This creates the toggle used by `<ComposeAdvancedOptions>`. |
||||||
|
|
||||||
|
__Props:__ |
||||||
|
|
||||||
|
- __`onChange` (`PropTypes.func`) :__ |
||||||
|
This provides the function to call when the toggle is |
||||||
|
(de-?)activated. |
||||||
|
|
||||||
|
- __`active` (`PropTypes.bool`) :__ |
||||||
|
This prop controls whether the toggle is currently active or not. |
||||||
|
|
||||||
|
- __`name` (`PropTypes.string`) :__ |
||||||
|
This identifies the toggle, and is sent to `onChange()` when it is |
||||||
|
called. |
||||||
|
|
||||||
|
- __`shortText` (`PropTypes.string`) :__ |
||||||
|
This is a short string used as the title of the toggle. |
||||||
|
|
||||||
|
- __`longText` (`PropTypes.string`) :__ |
||||||
|
This is a longer string used as a subtitle for the toggle. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Imports: |
||||||
|
-------- |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
// Package imports //
|
||||||
|
import React from 'react'; |
||||||
|
import PropTypes from 'prop-types'; |
||||||
|
import Toggle from 'react-toggle'; |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Implementation: |
||||||
|
--------------- |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
export default class ComposeAdvancedOptionsToggle extends React.PureComponent { |
||||||
|
|
||||||
|
static propTypes = { |
||||||
|
onChange: PropTypes.func.isRequired, |
||||||
|
active: PropTypes.bool.isRequired, |
||||||
|
name: PropTypes.string.isRequired, |
||||||
|
shortText: PropTypes.string.isRequired, |
||||||
|
longText: PropTypes.string.isRequired, |
||||||
|
} |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `onToggle()` |
||||||
|
|
||||||
|
The `onToggle()` function simply calls the `onChange()` prop with the |
||||||
|
toggle's `name`. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
onToggle = () => { |
||||||
|
this.props.onChange(this.props.name); |
||||||
|
} |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `render()` |
||||||
|
|
||||||
|
The `render()` function is used to render our component. We just render |
||||||
|
a `<Toggle>` and place next to it our text. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
render() { |
||||||
|
const { active, shortText, longText } = this.props; |
||||||
|
return ( |
||||||
|
<div role='button' tabIndex='0' className='advanced-options-dropdown__option' onClick={this.onToggle}> |
||||||
|
<div className='advanced-options-dropdown__option__toggle'> |
||||||
|
<Toggle checked={active} onChange={this.onToggle} /> |
||||||
|
</div> |
||||||
|
<div className='advanced-options-dropdown__option__content'> |
||||||
|
<strong>{shortText}</strong> |
||||||
|
{longText} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,171 @@ |
|||||||
|
/* |
||||||
|
|
||||||
|
`<NotificationFollow>` |
||||||
|
====================== |
||||||
|
|
||||||
|
This component renders a follow notification. |
||||||
|
|
||||||
|
__Props:__ |
||||||
|
|
||||||
|
- __`id` (`PropTypes.number.isRequired`) :__ |
||||||
|
This is the id of the notification. |
||||||
|
|
||||||
|
- __`onDeleteNotification` (`PropTypes.func.isRequired`) :__ |
||||||
|
The function to call when a notification should be |
||||||
|
dismissed/deleted. |
||||||
|
|
||||||
|
- __`account` (`PropTypes.object.isRequired`) :__ |
||||||
|
The account associated with the follow notification, ie the account |
||||||
|
which followed the user. |
||||||
|
|
||||||
|
- __`intl` (`PropTypes.object.isRequired`) :__ |
||||||
|
Our internationalization object, inserted by `@injectIntl`. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Imports: |
||||||
|
-------- |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
// Package imports //
|
||||||
|
import React from 'react'; |
||||||
|
import ImmutablePropTypes from 'react-immutable-proptypes'; |
||||||
|
import PropTypes from 'prop-types'; |
||||||
|
import { defineMessages, FormattedMessage, injectIntl } from 'react-intl'; |
||||||
|
import escapeTextContentForBrowser from 'escape-html'; |
||||||
|
import ImmutablePureComponent from 'react-immutable-pure-component'; |
||||||
|
|
||||||
|
// Mastodon imports //
|
||||||
|
import emojify from '../../../mastodon/emoji'; |
||||||
|
import Permalink from '../../../mastodon/components/permalink'; |
||||||
|
import AccountContainer from '../../../mastodon/containers/account_container'; |
||||||
|
|
||||||
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Inital setup: |
||||||
|
------------- |
||||||
|
|
||||||
|
The `messages` constant is used to define any messages that we need |
||||||
|
from inside props. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
const messages = defineMessages({ |
||||||
|
deleteNotification : |
||||||
|
{ id: 'status.dismiss_notification', defaultMessage: 'Dismiss notification' }, |
||||||
|
}); |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
Implementation: |
||||||
|
--------------- |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
@injectIntl |
||||||
|
export default class NotificationFollow extends ImmutablePureComponent { |
||||||
|
|
||||||
|
static propTypes = { |
||||||
|
id : PropTypes.number.isRequired, |
||||||
|
onDeleteNotification : PropTypes.func.isRequired, |
||||||
|
account : ImmutablePropTypes.map.isRequired, |
||||||
|
intl : PropTypes.object.isRequired, |
||||||
|
}; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `handleNotificationDeleteClick()` |
||||||
|
|
||||||
|
This function just calls our `onDeleteNotification()` prop with the |
||||||
|
notification's `id`. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
handleNotificationDeleteClick = () => { |
||||||
|
this.props.onDeleteNotification(this.props.id); |
||||||
|
} |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
### `render()` |
||||||
|
|
||||||
|
This actually renders the component. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
render () { |
||||||
|
const { account, intl } = this.props; |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
`dismiss` creates the notification dismissal button. Its title is given |
||||||
|
by `dismissTitle`. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
const dismissTitle = intl.formatMessage(messages.deleteNotification); |
||||||
|
const dismiss = ( |
||||||
|
<button |
||||||
|
aria-label={dismissTitle} |
||||||
|
title={dismissTitle} |
||||||
|
onClick={this.handleNotificationDeleteClick} |
||||||
|
className='status__prepend-dismiss-button' |
||||||
|
> |
||||||
|
<i className='fa fa-eraser' /> |
||||||
|
</button> |
||||||
|
); |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
`link` is a container for the account's `displayName`, which links to |
||||||
|
the account timeline using a `<Permalink>`. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
const displayName = account.get('display_name') || account.get('username'); |
||||||
|
const displayNameHTML = { __html: emojify(escapeTextContentForBrowser(displayName)) }; |
||||||
|
const link = ( |
||||||
|
<Permalink |
||||||
|
className='notification__display-name' |
||||||
|
href={account.get('url')} |
||||||
|
title={account.get('acct')} |
||||||
|
to={`/accounts/${account.get('id')}`} |
||||||
|
dangerouslySetInnerHTML={displayNameHTML} |
||||||
|
/> |
||||||
|
); |
||||||
|
|
||||||
|
/* |
||||||
|
|
||||||
|
We can now render our component. |
||||||
|
|
||||||
|
*/ |
||||||
|
|
||||||
|
return ( |
||||||
|
<div className='notification notification-follow'> |
||||||
|
<div className='notification__message'> |
||||||
|
<div className='notification__favourite-icon-wrapper'> |
||||||
|
<i className='fa fa-fw fa-user-plus' /> |
||||||
|
</div> |
||||||
|
|
||||||
|
<FormattedMessage |
||||||
|
id='notification.follow' |
||||||
|
defaultMessage='{name} followed you' |
||||||
|
values={{ name: link }} |
||||||
|
/> |
||||||
|
|
||||||
|
{dismiss} |
||||||
|
</div> |
||||||
|
|
||||||
|
<AccountContainer id={account.get('id')} withNote={false} /> |
||||||
|
</div> |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -1,78 +0,0 @@ |
|||||||
// Package imports //
|
|
||||||
import React from 'react'; |
|
||||||
import ImmutablePropTypes from 'react-immutable-proptypes'; |
|
||||||
import PropTypes from 'prop-types'; |
|
||||||
import { defineMessages, FormattedMessage, injectIntl } from 'react-intl'; |
|
||||||
import escapeTextContentForBrowser from 'escape-html'; |
|
||||||
import ImmutablePureComponent from 'react-immutable-pure-component'; |
|
||||||
|
|
||||||
// Mastodon imports //
|
|
||||||
import emojify from '../../../mastodon/emoji'; |
|
||||||
import Permalink from '../../../mastodon/components/permalink'; |
|
||||||
import AccountContainer from '../../../mastodon/containers/account_container'; |
|
||||||
|
|
||||||
const messages = defineMessages({ |
|
||||||
deleteNotification: { id: 'status.dismiss_notification', defaultMessage: 'Dismiss notification' }, |
|
||||||
}); |
|
||||||
|
|
||||||
|
|
||||||
@injectIntl |
|
||||||
export default class FollowNotification extends ImmutablePureComponent { |
|
||||||
|
|
||||||
static contextTypes = { |
|
||||||
router: PropTypes.object, |
|
||||||
}; |
|
||||||
|
|
||||||
static propTypes = { |
|
||||||
notificationId: PropTypes.number.isRequired, |
|
||||||
onDeleteNotification: PropTypes.func.isRequired, |
|
||||||
account: ImmutablePropTypes.map.isRequired, |
|
||||||
intl: PropTypes.object.isRequired, |
|
||||||
}; |
|
||||||
|
|
||||||
// Avoid checking props that are functions (and whose equality will always
|
|
||||||
// evaluate to false. See react-immutable-pure-component for usage.
|
|
||||||
updateOnProps = [ |
|
||||||
'account', |
|
||||||
] |
|
||||||
|
|
||||||
handleNotificationDeleteClick = () => { |
|
||||||
this.props.onDeleteNotification(this.props.notificationId); |
|
||||||
} |
|
||||||
|
|
||||||
render () { |
|
||||||
const { account, intl } = this.props; |
|
||||||
|
|
||||||
const dismissTitle = intl.formatMessage(messages.deleteNotification); |
|
||||||
const dismiss = ( |
|
||||||
<button |
|
||||||
aria-label={dismissTitle} |
|
||||||
title={dismissTitle} |
|
||||||
onClick={this.handleNotificationDeleteClick} |
|
||||||
className='status__prepend-dismiss-button' |
|
||||||
> |
|
||||||
<i className='fa fa-eraser' /> |
|
||||||
</button> |
|
||||||
); |
|
||||||
|
|
||||||
const displayName = account.get('display_name').length > 0 ? account.get('display_name') : account.get('username'); |
|
||||||
const displayNameHTML = { __html: emojify(escapeTextContentForBrowser(displayName)) }; |
|
||||||
const link = <Permalink className='notification__display-name' href={account.get('url')} title={account.get('acct')} to={`/accounts/${account.get('id')}`} dangerouslySetInnerHTML={displayNameHTML} />; |
|
||||||
return ( |
|
||||||
<div className='notification notification-follow'> |
|
||||||
<div className='notification__message'> |
|
||||||
<div className='notification__favourite-icon-wrapper'> |
|
||||||
<i className='fa fa-fw fa-user-plus' /> |
|
||||||
</div> |
|
||||||
|
|
||||||
<FormattedMessage id='notification.follow' defaultMessage='{name} followed you' values={{ name: link }} /> |
|
||||||
|
|
||||||
{dismiss} |
|
||||||
</div> |
|
||||||
|
|
||||||
<AccountContainer id={account.get('id')} withNote={false} /> |
|
||||||
</div> |
|
||||||
); |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
Loading…
Reference in new issue