2022-04-29 18:33:26 -07:00
'use strict' ;
import React from 'react' ;
import { defineMessages , useIntl , FormattedMessage } from 'react-intl' ;
import Badge from 'soapbox/components/badge' ;
2022-11-19 16:13:27 -08:00
import Markup from 'soapbox/components/markup' ;
2022-04-29 18:33:26 -07:00
import { Icon , HStack , Stack , Text } from 'soapbox/components/ui' ;
2022-11-15 08:00:49 -08:00
import VerificationBadge from 'soapbox/components/verification-badge' ;
2022-04-29 18:33:26 -07:00
import { useSoapboxConfig } from 'soapbox/hooks' ;
import { isLocal } from 'soapbox/utils/accounts' ;
2022-09-11 18:17:40 -07:00
import { badgeToTag , getBadges as getAccountBadges } from 'soapbox/utils/badges' ;
import { capitalize } from 'soapbox/utils/strings' ;
2022-04-29 18:33:26 -07:00
2022-11-16 05:32:32 -08:00
import ProfileFamiliarFollowers from './profile-familiar-followers' ;
2022-11-26 12:36:09 -08:00
import ProfileField from './profile-field' ;
2022-11-16 05:32:32 -08:00
import ProfileStats from './profile-stats' ;
2022-04-29 18:33:26 -07:00
import type { Account } from 'soapbox/types/entities' ;
/** Basically ensure the URL isn't `javascript:alert('hi')` or something like that */
const isSafeUrl = ( text : string ) : boolean = > {
try {
const url = new URL ( text ) ;
return [ 'http:' , 'https:' ] . includes ( url . protocol ) ;
} catch ( e ) {
return false ;
}
} ;
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' } ,
bot : { id : 'account.badges.bot' , defaultMessage : 'Bot' } ,
} ) ;
interface IProfileInfoPanel {
account : Account ,
/** Username from URL params, in case the account isn't found. */
username : string ,
}
/** User profile metadata, such as location, birthday, etc. */
const ProfileInfoPanel : React.FC < IProfileInfoPanel > = ( { account , username } ) = > {
const intl = useIntl ( ) ;
const { displayFqn } = useSoapboxConfig ( ) ;
const getStaffBadge = ( ) : React . ReactNode = > {
if ( account ? . admin ) {
2023-01-21 08:15:17 -08:00
return < Badge slug = 'admin' title = { < FormattedMessage id = 'account_moderation_modal.roles.admin' defaultMessage = 'Admin' / > } key = 'staff' / > ;
2022-04-29 18:33:26 -07:00
} else if ( account ? . moderator ) {
2023-01-21 08:15:17 -08:00
return < Badge slug = 'moderator' title = { < FormattedMessage id = 'account_moderation_modal.roles.moderator' defaultMessage = 'Moderator' / > } key = 'staff' / > ;
2022-04-29 18:33:26 -07:00
} else {
return null ;
}
} ;
2022-09-11 18:17:40 -07:00
const getCustomBadges = ( ) : React . ReactNode [ ] = > {
const badges = getAccountBadges ( account ) ;
return badges . map ( badge = > (
< Badge
key = { badge }
slug = { badge }
title = { capitalize ( badgeToTag ( badge ) ) }
/ >
) ) ;
} ;
2022-04-29 18:33:26 -07:00
const getBadges = ( ) : React . ReactNode [ ] = > {
2022-09-11 18:17:40 -07:00
const custom = getCustomBadges ( ) ;
2022-04-29 18:33:26 -07:00
const staffBadge = getStaffBadge ( ) ;
const isPatron = account . getIn ( [ 'patron' , 'is_patron' ] ) === true ;
const badges = [ ] ;
if ( staffBadge ) {
badges . push ( staffBadge ) ;
}
if ( isPatron ) {
badges . push ( < Badge slug = 'patron' title = 'Patron' key = 'patron' / > ) ;
}
2022-09-11 18:17:40 -07:00
return [ . . . badges , . . . custom ] ;
2022-04-29 18:33:26 -07:00
} ;
const renderBirthday = ( ) : React . ReactNode = > {
const birthday = account . birthday ;
if ( ! birthday ) return null ;
const formattedBirthday = intl . formatDate ( birthday , { timeZone : 'UTC' , day : 'numeric' , month : 'long' , year : 'numeric' } ) ;
2022-08-24 06:20:25 -07:00
const date = new Date ( birthday ) ;
2022-04-29 18:33:26 -07:00
const today = new Date ( ) ;
const hasBirthday = date . getDate ( ) === today . getDate ( ) && date . getMonth ( ) === today . getMonth ( ) ;
return (
< HStack alignItems = 'center' space = { 0.5 } >
< Icon
2022-07-09 09:20:02 -07:00
src = { require ( '@tabler/icons/ballon.svg' ) }
2022-04-29 18:33:26 -07:00
className = 'w-4 h-4 text-gray-800 dark:text-gray-200'
/ >
< 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 >
) ;
} ;
if ( ! account ) {
return (
< div className = 'mt-6 min-w-0 flex-1 sm:px-2' >
< Stack space = { 2 } >
< Stack >
< HStack space = { 1 } alignItems = 'center' >
2022-12-23 15:34:14 -08:00
< Text size = 'sm' theme = 'muted' direction = 'ltr' >
2022-04-29 18:33:26 -07:00
@ { username }
< / Text >
< / HStack >
< / Stack >
< / Stack >
< / div >
) ;
}
const content = { __html : account.note_emojified } ;
const deactivated = ! account . pleroma . get ( 'is_active' , true ) === true ;
const displayNameHtml = deactivated ? { __html : intl.formatMessage ( messages . deactivated ) } : { __html : account.display_name_html } ;
const memberSinceDate = intl . formatDate ( account . created_at , { month : 'long' , year : 'numeric' } ) ;
const verified = account . verified ;
const badges = getBadges ( ) ;
return (
< div className = 'mt-6 min-w-0 flex-1 sm:px-2' >
< Stack space = { 2 } >
< Stack >
< HStack space = { 1 } alignItems = 'center' >
< Text size = 'lg' weight = 'bold' dangerouslySetInnerHTML = { displayNameHtml } / >
{ verified && < VerificationBadge / > }
{ account . bot && < Badge slug = 'bot' title = { intl . formatMessage ( messages . bot ) } / > }
{ badges . length > 0 && (
< HStack space = { 1 } alignItems = 'center' >
{ badges }
< / HStack >
) }
< / HStack >
< HStack alignItems = 'center' space = { 0.5 } >
2022-12-23 15:34:14 -08:00
< Text size = 'sm' theme = 'muted' direction = 'ltr' >
2022-04-29 18:33:26 -07:00
@ { displayFqn ? account.fqn : account.acct }
< / Text >
{ account . locked && (
< Icon
2022-07-09 09:20:02 -07:00
src = { require ( '@tabler/icons/lock.svg' ) }
2022-04-29 18:33:26 -07:00
alt = { intl . formatMessage ( messages . account_locked ) }
className = 'w-4 h-4 text-gray-600'
/ >
) }
< / HStack >
< / Stack >
< ProfileStats account = { account } / >
2022-11-19 16:13:27 -08:00
{ account . note . length > 0 && (
< Markup size = 'sm' dangerouslySetInnerHTML = { content } / >
2022-04-29 18:33:26 -07:00
) }
< div className = 'flex flex-col md:flex-row items-start md:flex-wrap md:items-center gap-2' >
2022-06-19 11:38:51 -07:00
{ isLocal ( account ) ? (
2022-04-29 18:33:26 -07:00
< HStack alignItems = 'center' space = { 0.5 } >
< Icon
2022-07-09 09:20:02 -07:00
src = { require ( '@tabler/icons/calendar.svg' ) }
2022-04-29 18:33:26 -07:00
className = 'w-4 h-4 text-gray-800 dark:text-gray-200'
/ >
< Text size = 'sm' >
< FormattedMessage
id = 'account.member_since' defaultMessage = 'Joined {date}' values = { {
date : memberSinceDate ,
} }
/ >
< / Text >
< / HStack >
) : null }
{ account . location ? (
< HStack alignItems = 'center' space = { 0.5 } >
< Icon
2022-07-09 09:20:02 -07:00
src = { require ( '@tabler/icons/map-pin.svg' ) }
2022-04-29 18:33:26 -07:00
className = 'w-4 h-4 text-gray-800 dark:text-gray-200'
/ >
< Text size = 'sm' >
{ account . location }
< / Text >
< / HStack >
) : null }
{ account . website ? (
< HStack alignItems = 'center' space = { 0.5 } >
< Icon
2022-07-09 09:20:02 -07:00
src = { require ( '@tabler/icons/link.svg' ) }
2022-04-29 18:33:26 -07:00
className = 'w-4 h-4 text-gray-800 dark:text-gray-200'
/ >
< div className = 'max-w-[300px]' >
< Text size = 'sm' truncate >
{ isSafeUrl ( account . website ) ? (
2022-08-24 06:20:25 -07:00
< a className = 'text-primary-600 dark:text-accent-blue hover:underline' href = { account . website } target = '_blank' > { account . website } < / a >
2022-04-29 18:33:26 -07:00
) : (
account . website
) }
< / Text >
< / div >
< / HStack >
) : null }
{ renderBirthday ( ) }
< / div >
2022-05-13 14:14:55 -07:00
< ProfileFamiliarFollowers account = { account } / >
2022-04-29 18:33:26 -07:00
< / Stack >
2022-11-26 12:36:09 -08:00
{ account . fields . size > 0 && (
< Stack space = { 2 } className = 'mt-4 xl:hidden' >
{ account . fields . map ( ( field , i ) = > (
< ProfileField field = { field } key = { i } / >
) ) }
< / Stack >
) }
2022-04-29 18:33:26 -07:00
< / div >
) ;
} ;
export default ProfileInfoPanel ;