2022-04-29 18:33:26 -07:00
'use strict' ;
import React from 'react' ;
import { defineMessages , useIntl , FormattedMessage } from 'react-intl' ;
2023-06-21 15:13:29 -07:00
import { usePatronUser } from 'soapbox/api/hooks' ;
2022-04-29 18:33:26 -07:00
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' ;
2023-06-27 13:05:39 -07:00
import { useAppSelector , useSoapboxConfig } from 'soapbox/hooks' ;
2022-04-29 18:33:26 -07:00
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 {
2023-06-21 15:13:29 -07:00
account? : Account
2022-04-29 18:33:26 -07:00
/** Username from URL params, in case the account isn't found. */
2023-02-15 13:26:27 -08:00
username : string
2022-04-29 18:33:26 -07:00
}
/** User profile metadata, such as location, birthday, etc. */
const ProfileInfoPanel : React.FC < IProfileInfoPanel > = ( { account , username } ) = > {
const intl = useIntl ( ) ;
const { displayFqn } = useSoapboxConfig ( ) ;
2023-06-21 15:13:29 -07:00
const { patronUser } = usePatronUser ( account ? . url ) ;
2023-06-27 13:05:39 -07:00
const me = useAppSelector ( state = > state . me ) ;
const ownAccount = account ? . id === me ;
2022-04-29 18:33:26 -07:00
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 [ ] = > {
2023-06-21 15:13:29 -07:00
const badges = account ? getAccountBadges ( account ) : [ ] ;
2022-09-11 18:17:40 -07:00
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 ( ) ;
2023-06-21 15:13:29 -07:00
const isPatron = patronUser ? . is_patron === true ;
2022-04-29 18:33:26 -07:00
const badges = [ ] ;
if ( staffBadge ) {
badges . push ( staffBadge ) ;
}
if ( isPatron ) {
2023-08-25 13:43:21 -07:00
badges . push ( < Badge slug = 'patron' title = { < FormattedMessage id = 'account.patron' defaultMessage = 'Patron' / > } key = 'patron' / > ) ;
2022-04-29 18:33:26 -07:00
}
2022-09-11 18:17:40 -07:00
return [ . . . badges , . . . custom ] ;
2022-04-29 18:33:26 -07:00
} ;
const renderBirthday = ( ) : React . ReactNode = > {
2023-06-21 15:13:29 -07:00
const birthday = account ? . pleroma ? . birthday ;
2022-04-29 18:33:26 -07:00
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
2023-02-20 09:37:06 -08:00
src = { require ( '@tabler/icons/balloon.svg' ) }
2023-02-01 14:13:42 -08:00
className = 'h-4 w-4 text-gray-800 dark:text-gray-200'
2022-04-29 18:33:26 -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 >
) ;
} ;
if ( ! account ) {
return (
< div className = 'mt-6 min-w-0 flex-1 sm:px-2' >
< Stack space = { 2 } >
< Stack >
< HStack space = { 1 } alignItems = 'center' >
2023-05-02 11:09:58 -07:00
< Text size = 'sm' theme = 'muted' direction = 'ltr' truncate >
2022-04-29 18:33:26 -07:00
@ { username }
< / Text >
< / HStack >
< / Stack >
< / Stack >
< / div >
) ;
}
2023-06-20 12:24:39 -07:00
const deactivated = account . pleroma ? . deactivated ? ? false ;
2022-04-29 18:33:26 -07:00
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 badges = getBadges ( ) ;
return (
< div className = 'mt-6 min-w-0 flex-1 sm:px-2' >
< Stack space = { 2 } >
< Stack >
< HStack space = { 1 } alignItems = 'center' >
2023-04-10 07:33:24 -07:00
< Text size = 'lg' weight = 'bold' dangerouslySetInnerHTML = { displayNameHtml } truncate / >
2022-04-29 18:33:26 -07:00
{ 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 } >
2023-04-10 07:33:24 -07:00
< Text size = 'sm' theme = 'muted' direction = 'ltr' truncate >
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 ) }
2023-02-01 14:13:42 -08:00
className = 'h-4 w-4 text-gray-600'
2022-04-29 18:33:26 -07:00
/ >
) }
< / HStack >
< / Stack >
< ProfileStats account = { account } / >
2022-11-19 16:13:27 -08:00
{ account . note . length > 0 && (
2023-07-06 10:21:52 -07:00
< Markup size = 'sm' dangerouslySetInnerHTML = { { __html : account.note_emojified } } truncate / >
2022-04-29 18:33:26 -07:00
) }
2023-02-01 14:13:42 -08:00
< div className = 'flex flex-col items-start gap-2 md:flex-row md:flex-wrap md:items-center' >
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' ) }
2023-02-01 14:13:42 -08:00
className = 'h-4 w-4 text-gray-800 dark:text-gray-200'
2022-04-29 18:33:26 -07:00
/ >
< 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' ) }
2023-02-01 14:13:42 -08:00
className = 'h-4 w-4 text-gray-800 dark:text-gray-200'
2022-04-29 18:33:26 -07:00
/ >
< 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' ) }
2023-02-01 14:13:42 -08:00
className = 'h-4 w-4 text-gray-800 dark:text-gray-200'
2022-04-29 18:33:26 -07:00
/ >
< div className = 'max-w-[300px]' >
< Text size = 'sm' truncate >
{ isSafeUrl ( account . website ) ? (
2023-02-01 14:13:42 -08:00
< a className = 'text-primary-600 hover:underline dark:text-accent-blue' 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
2023-06-27 13:05:39 -07:00
{ ownAccount ? null : < ProfileFamiliarFollowers account = { account } / > }
2022-04-29 18:33:26 -07:00
< / Stack >
2022-11-26 12:36:09 -08:00
2023-06-20 12:24:39 -07:00
{ account . fields . length > 0 && (
2022-11-26 12:36:09 -08:00
< 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 ;