224 lines
6.0 KiB
JavaScript
Raw Normal View History

2020-02-20 19:57:29 -05:00
import ImmutablePropTypes from 'react-immutable-proptypes'
import ImmutablePureComponent from 'react-immutable-pure-component'
import debounce from 'lodash.debounce'
2020-04-23 23:17:27 -04:00
import { me } from '../initial_state'
import {
CX,
POPOVER_USER_INFO,
} from '../constants'
2020-02-28 10:20:47 -05:00
import { openPopover, closePopover } from '../actions/popover'
2020-02-20 19:57:29 -05:00
import Icon from './icon'
2020-04-23 23:17:27 -04:00
import Text from './text'
2020-02-28 10:20:47 -05:00
2020-04-11 18:29:19 -04:00
const mapDispatchToProps = (dispatch) => ({
2020-03-11 19:56:18 -04:00
openUserInfoPopover(props) {
2020-04-23 23:17:27 -04:00
dispatch(openPopover(POPOVER_USER_INFO, props))
2020-02-28 10:20:47 -05:00
},
closeUserInfoPopover() {
2020-04-23 23:17:27 -04:00
dispatch(closePopover(POPOVER_USER_INFO))
2020-02-28 10:20:47 -05:00
}
})
export default
@connect(null, mapDispatchToProps)
class DisplayName extends ImmutablePureComponent {
2020-02-08 01:12:01 -05:00
static propTypes = {
2020-03-24 00:39:12 -04:00
account: ImmutablePropTypes.map,
2020-02-28 10:20:47 -05:00
openUserInfoPopover: PropTypes.func.isRequired,
closeUserInfoPopover: PropTypes.func.isRequired,
2020-04-23 23:17:27 -04:00
isLarge: PropTypes.bool,
isMultiline: PropTypes.bool,
isSmall: PropTypes.bool,
2020-02-28 10:20:47 -05:00
noHover: PropTypes.bool,
2020-04-23 23:17:27 -04:00
noRelationship: PropTypes.bool,
2020-03-03 22:45:16 -05:00
noUsername: PropTypes.bool,
2020-05-06 19:40:54 -04:00
isComment: PropTypes.bool,
2020-05-09 23:26:58 -04:00
isCentered: PropTypes.bool,
2020-02-28 10:20:47 -05:00
}
2020-04-23 23:17:27 -04:00
updateOnProps = [
'account',
'isMultiline',
'isSmall',
'isLarge',
'noHover',
'noRelationship',
'noUsername',
2020-05-06 19:40:54 -04:00
'isComment',
2020-05-09 23:26:58 -04:00
'isCentered',
2020-04-23 23:17:27 -04:00
]
2020-04-23 02:13:29 -04:00
mouseOverTimeout = null
componentWillUnmount () {
document.removeEventListener('mousemove', this.handleMouseMove, true)
clearTimeout(this.mouseOverTimeout)
}
2020-04-23 02:13:29 -04:00
handleMouseEnter = () => {
if (this.mouseOverTimeout) return null
this.mouseOverTimeout = setTimeout(() => {
this.props.openUserInfoPopover({
targetRef: this.node,
position: 'top',
accountId: this.props.account.get('id'),
2020-04-23 02:13:29 -04:00
})
document.addEventListener('mousemove', this.handleMouseMove, true)
}, 1250)
2020-04-23 02:13:29 -04:00
}
2020-03-11 19:56:18 -04:00
handleMouseLeave = debounce((e) => {
this.attemptToHidePopover(e)
}, 250)
handleMouseMove = debounce((e) => {
this.attemptToHidePopover(e)
}, 100)
attemptToHidePopover = (e) => {
const lastTarget = e.toElement || e.relatedTarget
const isElement = (lastTarget instanceof Element || lastTarget instanceof HTMLDocument)
const userInfoPopoverEl = document.getElementById('user-info-popover')
if (this.mouseOverTimeout &&
(
!isElement && !userInfoPopoverEl ||
(userInfoPopoverEl && isElement && lastTarget && !userInfoPopoverEl.contains(lastTarget)) ||
(!userInfoPopoverEl && isElement && lastTarget && this.node && !this.node.contains(lastTarget))
)) {
document.removeEventListener('mousemove', this.handleMouseMove, true)
2020-04-23 02:13:29 -04:00
clearTimeout(this.mouseOverTimeout)
this.mouseOverTimeout = null
this.props.closeUserInfoPopover()
}
}
2020-02-28 10:20:47 -05:00
2020-03-11 19:56:18 -04:00
setRef = (n) => {
2020-04-23 02:13:29 -04:00
this.node = n
2020-02-20 19:57:29 -05:00
}
2020-02-08 01:12:01 -05:00
2020-02-28 10:20:47 -05:00
render() {
2020-03-03 22:45:16 -05:00
const {
account,
2020-04-23 23:17:27 -04:00
isMultiline,
isLarge,
2020-03-03 22:45:16 -05:00
noHover,
2020-03-08 19:02:28 -04:00
noUsername,
2020-04-23 23:17:27 -04:00
noRelationship,
2020-05-06 19:40:54 -04:00
isSmall,
isComment,
2020-05-09 23:26:58 -04:00
isCentered,
2020-03-03 22:45:16 -05:00
} = this.props
2020-02-28 10:20:47 -05:00
if (!account) return null
2020-04-23 23:17:27 -04:00
const containerClassName = CX({
default: 1,
maxWidth100PC: 1,
alignItemsCenter: !isMultiline,
flexRow: !isMultiline,
cursorPointer: !noHover,
2020-05-09 23:26:58 -04:00
alignItemsCenter: isCentered,
2020-04-23 23:17:27 -04:00
})
2020-02-28 10:20:47 -05:00
2020-04-23 23:17:27 -04:00
const displayNameClasses = CX({
2020-02-28 10:20:47 -05:00
text: 1,
overflowWrapBreakWord: 1,
whiteSpaceNoWrap: 1,
fontWeightBold: 1,
colorPrimary: 1,
2020-03-11 19:56:18 -04:00
mr2: 1,
2020-04-23 23:17:27 -04:00
lineHeight125: !isSmall,
2020-04-29 18:32:49 -04:00
fs14PX: isSmall,
fs15PX: !isLarge,
fs24PX: isLarge && !isSmall,
2020-02-28 10:20:47 -05:00
})
2020-04-23 23:17:27 -04:00
const usernameClasses = CX({
2020-02-28 10:20:47 -05:00
text: 1,
displayFlex: 1,
flexNormal: 1,
flexShrink1: 1,
overflowWrapBreakWord: 1,
textOverflowEllipsis: 1,
colorSecondary: 1,
fontWeightNormal: 1,
2020-04-23 23:17:27 -04:00
lineHeight15: isMultiline,
lineHeight125: !isMultiline,
ml5: !isMultiline,
2020-04-29 18:32:49 -04:00
fs14PX: isSmall,
fs15PX: !isLarge,
fs16PX: isLarge && !isSmall,
2020-02-28 10:20:47 -05:00
})
2020-03-08 19:02:28 -04:00
const iconSize =
2020-05-09 23:26:58 -04:00
!!isLarge ? 19 :
!!isComment ? 12 :
!!isSmall ? 14 : 15
2020-02-08 01:12:01 -05:00
2020-04-23 23:17:27 -04:00
let relationshipLabel
if (me && account) {
const accountId = account.get('id')
const isFollowedBy = (me !== accountId && account.getIn(['relationship', 'followed_by']))
if (isFollowedBy) {
relationshipLabel = 'Follows you'//intl.formatMessage(messages.accountFollowsYou)
}
}
// {
// /* : todo : audio-mute, bot
2020-04-23 23:17:27 -04:00
// account.getIn(['relationship', 'muting'])
// */
// }
// bot: { id: 'account.badges.bot', defaultMessage: 'Bot' },
2020-04-23 02:13:29 -04:00
2020-02-08 01:12:01 -05:00
return (
2020-04-23 23:17:27 -04:00
<div
className={containerClassName}
onMouseEnter={noHover ? undefined : this.handleMouseEnter}
onMouseLeave={noHover ? undefined : this.handleMouseLeave}
ref={this.setRef}
>
2020-05-09 23:26:58 -04:00
<span className={[_s.default, _s.flexRow, _s.alignItemsCenter, _s.maxWidth100PC].join(' ')}>
2020-02-28 10:20:47 -05:00
<bdi className={[_s.text, _s.whiteSpaceNoWrap, _s.textOverflowEllipsis].join(' ')}>
<strong
className={displayNameClasses}
dangerouslySetInnerHTML={{ __html: account.get('display_name_html') }}
/>
2020-04-23 23:17:27 -04:00
{
2020-05-09 23:26:58 -04:00
account.get('locked') &&
<Icon id='lock-filled' size={`${iconSize - 3}px`} className={[_s.fillPrimary, _s.ml5].join(' ')} />
2020-04-23 23:17:27 -04:00
}
2020-02-28 10:20:47 -05:00
</bdi>
{
account.get('is_verified') &&
2020-05-09 23:26:58 -04:00
<Icon id='verified' size={`${iconSize}px`} className={[_s.ml5, _s.default].join(' ')} />
2020-02-28 10:20:47 -05:00
}
2020-04-23 23:17:27 -04:00
</span>
2020-03-03 22:45:16 -05:00
{
!noUsername &&
<span className={usernameClasses}>
@{account.get('acct')}
2020-04-23 23:17:27 -04:00
{
!noRelationship && !!relationshipLabel &&
<span className={[_s.default, _s.ml5, _s.justifyContentCenter].join(' ')}>
<Text
size='extraSmall'
isBadge
color='tertiary'
2020-04-29 18:32:49 -04:00
className={[_s.bgSecondary, _s.py2].join(' ')}
2020-04-23 23:17:27 -04:00
>
{relationshipLabel}
</Text>
</span>
}
2020-03-03 22:45:16 -05:00
</span>
}
2020-04-23 23:17:27 -04:00
</div>
2020-02-20 19:57:29 -05:00
)
2020-02-08 01:12:01 -05:00
}
}