pleroma/app/soapbox/features/ui/components/profile_info_panel.js

268 lines
8.4 KiB
JavaScript
Raw Normal View History

2020-03-27 13:59:38 -07:00
'use strict';
import { List as ImmutableList } from 'immutable';
import PropTypes from 'prop-types';
2020-03-27 13:59:38 -07:00
import React from 'react';
import ImmutablePropTypes from 'react-immutable-proptypes';
import ImmutablePureComponent from 'react-immutable-pure-component';
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
import { connect } from 'react-redux';
import { initAccountNoteModal } from 'soapbox/actions/account_notes';
import Badge from 'soapbox/components/badge';
2022-03-21 11:09:01 -07:00
import { Icon, HStack, Stack, Text } from 'soapbox/components/ui';
2020-05-28 15:52:07 -07:00
import VerificationBadge from 'soapbox/components/verification_badge';
2022-02-27 20:25:23 -08:00
import { getAcct, isAdmin, isModerator, isLocal } from 'soapbox/utils/accounts';
import { displayFqn } from 'soapbox/utils/state';
2021-09-13 11:09:11 -07:00
import ProfileStats from './profile_stats';
2022-03-21 11:09:01 -07:00
// Basically ensure the URL isn't `javascript:alert('hi')` or something like that
const isSafeUrl = text => {
try {
const url = new URL(text);
return ['http:', 'https:'].includes(url.protocol);
} catch(e) {
return false;
}
};
2020-03-27 13:59:38 -07:00
const messages = defineMessages({
linkVerifiedOn: { id: 'account.link_verified_on', defaultMessage: 'Ownership of this link was checked on {date}' },
account_locked: { id: 'account.locked_info', defaultMessage: 'This account privacy status is set to locked. The owner manually reviews who can follow them.' },
deactivated: { id: 'account.deactivated', defaultMessage: 'Deactivated' },
2021-01-18 10:55:38 -08:00
bot: { id: 'account.badges.bot', defaultMessage: 'Bot' },
2020-03-27 13:59:38 -07:00
});
class ProfileInfoPanel extends ImmutablePureComponent {
static propTypes = {
2022-03-23 10:14:42 -07:00
account: ImmutablePropTypes.record,
2020-03-27 13:59:38 -07:00
identity_proofs: ImmutablePropTypes.list,
intl: PropTypes.object.isRequired,
username: PropTypes.string,
displayFqn: PropTypes.bool,
onShowNote: PropTypes.func,
2020-03-27 13:59:38 -07:00
};
getStaffBadge = () => {
const { account } = this.props;
if (isAdmin(account)) {
return <Badge slug='admin' title='Admin' key='staff' />;
} else if (isModerator(account)) {
return <Badge slug='moderator' title='Moderator' key='staff' />;
} else {
return null;
}
}
getBadges = () => {
const { account } = this.props;
const staffBadge = this.getStaffBadge();
const isPatron = account.getIn(['patron', 'is_patron']);
const badges = [];
if (staffBadge) {
badges.push(staffBadge);
}
if (isPatron) {
badges.push(<Badge slug='patron' title='Patron' key='patron' />);
}
return badges;
}
2022-03-21 11:09:01 -07:00
renderBirthday = () => {
const { account, intl } = this.props;
const birthday = account.get('birthday');
if (!birthday) return null;
2022-01-31 18:18:15 -08:00
const formattedBirthday = intl.formatDate(birthday, { timeZone: 'UTC', day: 'numeric', month: 'long', year: 'numeric' });
const date = new Date(birthday);
const today = new Date();
const hasBirthday = date.getDate() === today.getDate() && date.getMonth() === today.getMonth();
return (
2022-03-21 11:09:01 -07:00
<HStack alignItems='center' space={0.5}>
<Icon
src={require('@tabler/icons/icons/ballon.svg')}
className='w-4 h-4 text-gray-800'
/>
2022-03-21 11:09:01 -07:00
<Text size='sm'>
{hasBirthday ? (
<FormattedMessage id='account.birthday_today' defaultMessage='Birthday is today!' />
) : (
<FormattedMessage id='account.birthday' defaultMessage='Born {date}' values={{ date: formattedBirthday }} />
)}
</Text>
</HStack>
);
}
handleShowNote = e => {
const { account, onShowNote } = this.props;
e.preventDefault();
onShowNote(account);
}
render() {
2022-03-21 11:09:01 -07:00
const { account, displayFqn, intl, username } = this.props;
2020-03-27 13:59:38 -07:00
if (!account) {
return (
2022-03-21 11:09:01 -07:00
<div className='mt-6 min-w-0 flex-1 sm:px-2'>
<Stack space={2}>
<Stack>
<HStack space={1} alignItems='center'>
<Text size='sm' theme='muted'>
@{username}
</Text>
</HStack>
</Stack>
</Stack>
2020-03-27 13:59:38 -07:00
</div>
);
}
const content = { __html: account.get('note_emojified') };
const deactivated = !account.getIn(['pleroma', 'is_active'], true);
const displayNameHtml = deactivated ? { __html: intl.formatMessage(messages.deactivated) } : { __html: account.get('display_name_html') };
2020-03-27 13:59:38 -07:00
const memberSinceDate = intl.formatDate(account.get('created_at'), { month: 'long', year: 'numeric' });
2022-02-27 20:25:23 -08:00
const verified = account.get('verified');
const badges = this.getBadges();
2020-03-27 13:59:38 -07:00
return (
2022-03-21 11:09:01 -07:00
<div className='mt-6 min-w-0 flex-1 sm:px-2'>
<Stack space={2}>
{/* Not sure if this is actual used. */}
{/* <div className='profile-info-panel-content__deactivated'>
<FormattedMessage
id='account.deactivated_description' defaultMessage='This account has been deactivated.'
/>
</div> */}
<Stack>
<HStack space={1} alignItems='center'>
<Text size='lg' weight='bold' dangerouslySetInnerHTML={displayNameHtml} />
2020-03-27 13:59:38 -07:00
{verified && <VerificationBadge />}
2022-03-21 11:09:01 -07:00
2021-01-18 10:55:38 -08:00
{account.get('bot') && <Badge slug='bot' title={intl.formatMessage(messages.bot)} />}
2020-03-27 13:59:38 -07:00
2022-03-21 11:09:01 -07:00
{badges.length > 0 && (
<HStack space={1} alignItems='center'>
{badges}
</HStack>
)}
</HStack>
2020-08-25 11:47:02 -07:00
2022-03-21 11:09:01 -07:00
<HStack alignItems='center' space={0.5}>
<Text size='sm' theme='muted'>
@{getAcct(account, displayFqn)}
</Text>
{account.get('locked') && (
<Icon
src={require('@tabler/icons/icons/lock.svg')}
title={intl.formatMessage(messages.account_locked)}
className='w-4 h-4 text-gray-600'
/>
)}
</HStack>
</Stack>
<ProfileStats account={account} />
2020-03-27 13:59:38 -07:00
2020-08-25 11:47:02 -07:00
{
2020-03-27 13:59:38 -07:00
(account.get('note').length > 0 && account.get('note') !== '<p></p>') &&
2022-03-21 11:09:01 -07:00
<Text size='sm' dangerouslySetInnerHTML={content} />
2020-03-27 13:59:38 -07:00
}
2022-03-24 11:07:08 -07:00
<div className='flex flex-col md:flex-row items-start md:flex-wrap md:items-center gap-2'>
2022-03-21 11:09:01 -07:00
{isLocal(account) ? (
<HStack alignItems='center' space={0.5}>
<Icon
src={require('@tabler/icons/icons/calendar.svg')}
className='w-4 h-4 text-gray-800'
/>
<Text size='sm'>
<FormattedMessage
id='account.member_since' defaultMessage='Joined {date}' values={{
date: memberSinceDate,
}}
/>
</Text>
</HStack>
) : null}
{account.get('location') ? (
<HStack alignItems='center' space={0.5}>
<Icon
src={require('@tabler/icons/icons/map-pin.svg')}
className='w-4 h-4 text-gray-800'
/>
<Text size='sm'>
{account.get('location')}
</Text>
</HStack>
) : null}
{account.get('website') ? (
<HStack alignItems='center' space={0.5}>
<Icon
src={require('@tabler/icons/icons/link.svg')}
className='w-4 h-4 text-gray-800'
/>
2022-03-24 11:07:08 -07:00
<div className='max-w-[300px]'>
<Text size='sm' truncate>
{isSafeUrl(account.get('website')) ? (
<a className='text-primary-600 hover:underline' href={account.get('website')} target='_blank'>{account.get('website')}</a>
) : (
account.get('website')
)}
</Text>
</div>
2022-03-21 11:09:01 -07:00
</HStack>
) : null}
{this.renderBirthday()}
</div>
</Stack>
2020-03-27 13:59:38 -07:00
</div>
);
}
2020-04-14 11:44:40 -07:00
2020-03-27 13:59:38 -07:00
}
const mapStateToProps = (state, { account }) => {
const identity_proofs = account ? state.getIn(['identity_proofs', account.get('id')], ImmutableList()) : ImmutableList();
return {
identity_proofs,
domain: state.getIn(['meta', 'domain']),
displayFqn: displayFqn(state),
2020-03-27 13:59:38 -07:00
};
};
const mapDispatchToProps = (dispatch) => ({
onShowNote(account) {
dispatch(initAccountNoteModal(account));
},
});
2020-03-27 13:59:38 -07:00
export default injectIntl(
connect(mapStateToProps, mapDispatchToProps, null, {
2020-03-27 13:59:38 -07:00
forwardRef: true,
},
2020-04-14 11:44:40 -07:00
)(ProfileInfoPanel));