Add domain information to profiles in web UI (#29602)
Co-authored-by: Claire <claire.github-309c@sitedethib.com>
| @ -0,0 +1,86 @@ | ||||
| import PropTypes from 'prop-types'; | ||||
| import { useState, useRef, useCallback } from 'react'; | ||||
| 
 | ||||
| import { FormattedMessage } from 'react-intl'; | ||||
| 
 | ||||
| import classNames from 'classnames'; | ||||
| 
 | ||||
| import Overlay from 'react-overlays/Overlay'; | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| import AlternateEmailIcon from '@/material-icons/400-24px/alternate_email.svg?react'; | ||||
| import BadgeIcon from '@/material-icons/400-24px/badge.svg?react'; | ||||
| import GlobeIcon from '@/material-icons/400-24px/globe.svg?react'; | ||||
| import { Icon } from 'mastodon/components/icon'; | ||||
| 
 | ||||
| export const DomainPill = ({ domain, username, isSelf }) => { | ||||
|   const [open, setOpen] = useState(false); | ||||
|   const [expanded, setExpanded] = useState(false); | ||||
|   const triggerRef = useRef(null); | ||||
| 
 | ||||
|   const handleClick = useCallback(() => { | ||||
|     setOpen(!open); | ||||
|   }, [open, setOpen]); | ||||
| 
 | ||||
|   const handleExpandClick = useCallback(() => { | ||||
|     setExpanded(!expanded); | ||||
|   }, [expanded, setExpanded]); | ||||
| 
 | ||||
|   return ( | ||||
|     <> | ||||
|       <button className={classNames('account__domain-pill', { active: open })} ref={triggerRef} onClick={handleClick}>{domain}</button> | ||||
| 
 | ||||
|       <Overlay show={open} rootClose onHide={handleClick} offset={[5, 5]} target={triggerRef}> | ||||
|         {({ props }) => ( | ||||
|           <div {...props} className='account__domain-pill__popout dropdown-animation'> | ||||
|             <div className='account__domain-pill__popout__header'> | ||||
|               <div className='account__domain-pill__popout__header__icon'><Icon icon={BadgeIcon} /></div> | ||||
|               <h3><FormattedMessage id='domain_pill.whats_in_a_handle' defaultMessage="What's in a handle?" /></h3> | ||||
|             </div> | ||||
| 
 | ||||
|             <div className='account__domain-pill__popout__handle'> | ||||
|               <div className='account__domain-pill__popout__handle__label'>{isSelf ? <FormattedMessage id='domain_pill.your_handle' defaultMessage='Your handle:' /> : <FormattedMessage id='domain_pill.their_handle' defaultMessage='Their handle:' />}</div> | ||||
|               <div className='account__domain-pill__popout__handle__handle'>@{username}@{domain}</div> | ||||
|             </div> | ||||
| 
 | ||||
|             <div className='account__domain-pill__popout__parts'> | ||||
|               <div> | ||||
|                 <div className='account__domain-pill__popout__parts__icon'><Icon icon={AlternateEmailIcon} /></div> | ||||
| 
 | ||||
|                 <div> | ||||
|                   <h6><FormattedMessage id='domain_pill.username' defaultMessage='Username' /></h6> | ||||
|                   <p>{isSelf ? <FormattedMessage id='domain_pill.your_username' defaultMessage='Your unique identifier on this server. It’s possible to find users with the same username on different servers.' /> : <FormattedMessage id='domain_pill.their_username' defaultMessage='Their unique identifier on their server. It’s possible to find users with the same username on different servers.' />}</p> | ||||
|                 </div> | ||||
|               </div> | ||||
| 
 | ||||
|               <div> | ||||
|                 <div className='account__domain-pill__popout__parts__icon'><Icon icon={GlobeIcon} /></div> | ||||
| 
 | ||||
|                 <div> | ||||
|                   <h6><FormattedMessage id='domain_pill.server' defaultMessage='Server' /></h6> | ||||
|                   <p>{isSelf ? <FormattedMessage id='domain_pill.your_server' defaultMessage='Your digital home, where all of your posts live. Don’t like this one? Transfer servers at any time and bring your followers, too.' /> : <FormattedMessage id='domain_pill.their_server' defaultMessage='Their digital home, where all of their posts live.' />}</p> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
| 
 | ||||
|             <p>{isSelf ? <FormattedMessage id='domain_pill.who_you_are' defaultMessage='Because your handle says who you are and where you are, people can interact with you across the social web of <button>ActivityPub-powered platforms</button>.' values={{ button: x => <button onClick={handleExpandClick} className='link-button'>{x}</button> }} /> : <FormattedMessage id='domain_pill.who_they_are' defaultMessage='Since handles say who someone is and where they are, you can interact with people across the social web of <button>ActivityPub-powered platforms</button>.' values={{ button: x => <button onClick={handleExpandClick} className='link-button'>{x}</button> }} />}</p> | ||||
| 
 | ||||
|             {expanded && ( | ||||
|               <> | ||||
|                 <p><FormattedMessage id='domain_pill.activitypub_like_language' defaultMessage='ActivityPub is like the language Mastodon speaks with other social networks.' /></p> | ||||
|                 <p><FormattedMessage id='domain_pill.activitypub_lets_connect' defaultMessage='It lets you connect and interact with people not just on Mastodon, but across different social apps too.' /></p> | ||||
|               </> | ||||
|             )} | ||||
|           </div> | ||||
|         )} | ||||
|       </Overlay> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| DomainPill.propTypes = { | ||||
|   username: PropTypes.string.isRequired, | ||||
|   domain: PropTypes.string.isRequired, | ||||
|   isSelf: PropTypes.bool, | ||||
| }; | ||||
| @ -25,13 +25,15 @@ import { IconButton } from 'mastodon/components/icon_button'; | ||||
| import { LoadingIndicator } from 'mastodon/components/loading_indicator'; | ||||
| import { ShortNumber } from 'mastodon/components/short_number'; | ||||
| import DropdownMenuContainer from 'mastodon/containers/dropdown_menu_container'; | ||||
| import { autoPlayGif, me, domain } from 'mastodon/initial_state'; | ||||
| import { autoPlayGif, me, domain as localDomain } from 'mastodon/initial_state'; | ||||
| import { PERMISSION_MANAGE_USERS, PERMISSION_MANAGE_FEDERATION } from 'mastodon/permissions'; | ||||
| import { WithRouterPropTypes } from 'mastodon/utils/react_router'; | ||||
| 
 | ||||
| import AccountNoteContainer from '../containers/account_note_container'; | ||||
| import FollowRequestNoteContainer from '../containers/follow_request_note_container'; | ||||
| 
 | ||||
| import { DomainPill } from './domain_pill'; | ||||
| 
 | ||||
| const messages = defineMessages({ | ||||
|   unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' }, | ||||
|   follow: { id: 'account.follow', defaultMessage: 'Follow' }, | ||||
| @ -78,7 +80,7 @@ const messages = defineMessages({ | ||||
| 
 | ||||
| const titleFromAccount = account => { | ||||
|   const displayName = account.get('display_name'); | ||||
|   const acct = account.get('acct') === account.get('username') ? `${account.get('username')}@${domain}` : account.get('acct'); | ||||
|   const acct = account.get('acct') === account.get('username') ? `${account.get('username')}@${localDomain}` : account.get('acct'); | ||||
|   const prefix = displayName.trim().length === 0 ? account.get('username') : displayName; | ||||
| 
 | ||||
|   return `${prefix} (@${acct})`; | ||||
| @ -252,7 +254,7 @@ class Header extends ImmutablePureComponent { | ||||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { account, hidden, intl, domain } = this.props; | ||||
|     const { account, hidden, intl } = this.props; | ||||
|     const { signedIn, permissions } = this.context.identity; | ||||
| 
 | ||||
|     if (!account) { | ||||
| @ -393,7 +395,8 @@ class Header extends ImmutablePureComponent { | ||||
|     const displayNameHtml = { __html: account.get('display_name_html') }; | ||||
|     const fields          = account.get('fields'); | ||||
|     const isLocal         = account.get('acct').indexOf('@') === -1; | ||||
|     const acct            = isLocal && domain ? `${account.get('acct')}@${domain}` : account.get('acct'); | ||||
|     const username        = account.get('acct').split('@')[0]; | ||||
|     const domain          = isLocal ? localDomain : account.get('acct').split('@')[1]; | ||||
|     const isIndexable     = !account.get('noindex'); | ||||
| 
 | ||||
|     const badges = []; | ||||
| @ -438,7 +441,9 @@ class Header extends ImmutablePureComponent { | ||||
|             <h1> | ||||
|               <span dangerouslySetInnerHTML={displayNameHtml} /> | ||||
|               <small> | ||||
|                 <span>@{acct}</span> {lockedIcon} | ||||
|                 <span>@{username}<span className='invisible'>@{domain}</span></span> | ||||
|                 <DomainPill username={username} domain={domain} isSelf={me === account.get('id')} /> | ||||
|                 {lockedIcon} | ||||
|               </small> | ||||
|             </h1> | ||||
|           </div> | ||||
|  | ||||
| @ -205,6 +205,19 @@ | ||||
|   "dismissable_banner.explore_statuses": "These are posts from across the social web that are gaining traction today. Newer posts with more boosts and favorites are ranked higher.", | ||||
|   "dismissable_banner.explore_tags": "These are hashtags that are gaining traction on the social web today. Hashtags that are used by more different people are ranked higher.", | ||||
|   "dismissable_banner.public_timeline": "These are the most recent public posts from people on the social web that people on {domain} follow.", | ||||
|   "domain_pill.activitypub_lets_connect": "It lets you connect and interact with people not just on Mastodon, but across different social apps too.", | ||||
|   "domain_pill.activitypub_like_language": "ActivityPub is like the language Mastodon speaks with other social networks.", | ||||
|   "domain_pill.server": "Server", | ||||
|   "domain_pill.their_handle": "Their handle:", | ||||
|   "domain_pill.their_server": "Their digital home, where all of their posts live.", | ||||
|   "domain_pill.their_username": "Their unique identifier on their server. It’s possible to find users with the same username on different servers.", | ||||
|   "domain_pill.username": "Username", | ||||
|   "domain_pill.whats_in_a_handle": "What's in a handle?", | ||||
|   "domain_pill.who_they_are": "Since handles say who someone is and where they are, you can interact with people across the social web of <button>ActivityPub-powered platforms</button>.", | ||||
|   "domain_pill.who_you_are": "Because your handle says who you are and where you are, people can interact with you across the social web of <button>ActivityPub-powered platforms</button>.", | ||||
|   "domain_pill.your_handle": "Your handle:", | ||||
|   "domain_pill.your_server": "Your digital home, where all of your posts live. Don’t like this one? Transfer servers at any time and bring your followers, too.", | ||||
|   "domain_pill.your_username": "Your unique identifier on this server. It’s possible to find users with the same username on different servers.", | ||||
|   "embed.instructions": "Embed this post on your website by copying the code below.", | ||||
|   "embed.preview": "Here is what it will look like:", | ||||
|   "emoji_button.activity": "Activity", | ||||
|  | ||||
							
								
								
									
										1
									
								
								app/javascript/material-icons/400-24px/badge-fill.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="M160-80q-33 0-56.5-23.5T80-160v-440q0-33 23.5-56.5T160-680h200v-120q0-33 23.5-56.5T440-880h80q33 0 56.5 23.5T600-800v120h200q33 0 56.5 23.5T880-600v440q0 33-23.5 56.5T800-80H160Zm80-160h240v-18q0-17-9.5-31.5T444-312q-20-9-40.5-13.5T360-330q-23 0-43.5 4.5T276-312q-17 8-26.5 22.5T240-258v18Zm320-60h160v-60H560v60Zm-200-60q25 0 42.5-17.5T420-420q0-25-17.5-42.5T360-480q-25 0-42.5 17.5T300-420q0 25 17.5 42.5T360-360Zm200-60h160v-60H560v60ZM440-600h80v-200h-80v200Z"/></svg> | ||||
| After Width: | Height: | Size: 569 B | 
							
								
								
									
										1
									
								
								app/javascript/material-icons/400-24px/badge.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="M160-80q-33 0-56.5-23.5T80-160v-440q0-33 23.5-56.5T160-680h200v-120q0-33 23.5-56.5T440-880h80q33 0 56.5 23.5T600-800v120h200q33 0 56.5 23.5T880-600v440q0 33-23.5 56.5T800-80H160Zm0-80h640v-440H600q0 33-23.5 56.5T520-520h-80q-33 0-56.5-23.5T360-600H160v440Zm80-80h240v-18q0-17-9.5-31.5T444-312q-20-9-40.5-13.5T360-330q-23 0-43.5 4.5T276-312q-17 8-26.5 22.5T240-258v18Zm320-60h160v-60H560v60Zm-200-60q25 0 42.5-17.5T420-420q0-25-17.5-42.5T360-480q-25 0-42.5 17.5T300-420q0 25 17.5 42.5T360-360Zm200-60h160v-60H560v60ZM440-600h80v-200h-80v200Zm40 220Z"/></svg> | ||||
| After Width: | Height: | Size: 654 B | 
							
								
								
									
										1
									
								
								app/javascript/material-icons/400-24px/globe-fill.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="M480-80q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q134 0 227-93t93-227q0-7-.5-14.5T799-507q-5 29-27 48t-52 19h-80q-33 0-56.5-23.5T560-520v-40H400v-80q0-33 23.5-56.5T480-720h40q0-23 12.5-40.5T563-789q-20-5-40.5-8t-42.5-3q-134 0-227 93t-93 227h200q66 0 113 47t47 113v40H400v110q20 5 39.5 7.5T480-160Z"/></svg> | ||||
| After Width: | Height: | Size: 548 B | 
							
								
								
									
										1
									
								
								app/javascript/material-icons/400-24px/globe.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="M480-80q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q134 0 227-93t93-227q0-7-.5-14.5T799-507q-5 29-27 48t-52 19h-80q-33 0-56.5-23.5T560-520v-40H400v-80q0-33 23.5-56.5T480-720h40q0-23 12.5-40.5T563-789q-20-5-40.5-8t-42.5-3q-134 0-227 93t-93 227h200q66 0 113 47t47 113v40H400v110q20 5 39.5 7.5T480-160Z"/></svg> | ||||
| After Width: | Height: | Size: 548 B | 
| @ -1 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="m233-80 65-281L80-550l288-25 112-265 112 265 288 25-218 189 65 281-247-149L233-80Z"/></svg> | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="m233-120 65-281L80-590l288-25 112-265 112 265 288 25-218 189 65 281-247-149-247 149Z"/></svg> | ||||
| Before Width: | Height: | Size: 188 B After Width: | Height: | Size: 190 B | 
| @ -1 +1 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="m354-247 126-76 126 77-33-144 111-96-146-13-58-136-58 135-146 13 111 97-33 143ZM233-80l65-281L80-550l288-25 112-265 112 265 288 25-218 189 65 281-247-149L233-80Zm247-350Z"/></svg> | ||||
| <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24"><path d="m354-287 126-76 126 77-33-144 111-96-146-13-58-136-58 135-146 13 111 97-33 143ZM233-120l65-281L80-590l288-25 112-265 112 265 288 25-218 189 65 281-247-149-247 149Zm247-350Z"/></svg> | ||||
| Before Width: | Height: | Size: 276 B After Width: | Height: | Size: 278 B | 
| @ -1792,6 +1792,118 @@ body > [data-popper-placement] { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   &__domain-pill { | ||||
|     display: inline-flex; | ||||
|     background: rgba($highlight-text-color, 0.2); | ||||
|     border-radius: 4px; | ||||
|     border: 0; | ||||
|     color: $highlight-text-color; | ||||
|     font-weight: 500; | ||||
|     font-size: 12px; | ||||
|     line-height: 16px; | ||||
|     padding: 4px 8px; | ||||
| 
 | ||||
|     &.active { | ||||
|       color: $white; | ||||
|       background: $ui-highlight-color; | ||||
|     } | ||||
| 
 | ||||
|     &__popout { | ||||
|       background: var(--dropdown-background-color); | ||||
|       backdrop-filter: var(--background-filter); | ||||
|       border: 1px solid var(--dropdown-border-color); | ||||
|       box-shadow: var(--dropdown-shadow); | ||||
|       max-width: 320px; | ||||
|       padding: 16px; | ||||
|       border-radius: 8px; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       gap: 24px; | ||||
|       font-size: 14px; | ||||
|       line-height: 20px; | ||||
|       color: $darker-text-color; | ||||
| 
 | ||||
|       .link-button { | ||||
|         display: inline; | ||||
|         font-size: inherit; | ||||
|         line-height: inherit; | ||||
|       } | ||||
| 
 | ||||
|       &__header { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|         gap: 12px; | ||||
| 
 | ||||
|         &__icon { | ||||
|           width: 40px; | ||||
|           height: 40px; | ||||
|           background: $ui-highlight-color; | ||||
|           color: $white; | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           justify-content: center; | ||||
|           border-radius: 50%; | ||||
|           flex-shrink: 0; | ||||
|         } | ||||
| 
 | ||||
|         h3 { | ||||
|           font-size: 17px; | ||||
|           line-height: 22px; | ||||
|           color: $primary-text-color; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       &__handle { | ||||
|         border: 2px dashed $highlight-text-color; | ||||
|         background: rgba($highlight-text-color, 0.1); | ||||
|         padding: 12px 8px; | ||||
|         color: $highlight-text-color; | ||||
|         border-radius: 4px; | ||||
| 
 | ||||
|         &__label { | ||||
|           font-size: 11px; | ||||
|           line-height: 16px; | ||||
|           font-weight: 500; | ||||
|         } | ||||
| 
 | ||||
|         &__handle { | ||||
|           user-select: all; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       &__parts { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         gap: 8px; | ||||
|         font-size: 12px; | ||||
|         line-height: 16px; | ||||
| 
 | ||||
|         & > div { | ||||
|           display: flex; | ||||
|           align-items: flex-start; | ||||
|           gap: 12px; | ||||
|         } | ||||
| 
 | ||||
|         &__icon { | ||||
|           width: 40px; | ||||
|           height: 40px; | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           justify-content: center; | ||||
|           flex-shrink: 0; | ||||
|           color: $highlight-text-color; | ||||
|         } | ||||
| 
 | ||||
|         h6 { | ||||
|           font-size: 14px; | ||||
|           line-height: 20px; | ||||
|           font-weight: 500; | ||||
|           color: $primary-text-color; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   &__note { | ||||
|     font-size: 14px; | ||||
|     font-weight: 400; | ||||
| @ -7544,14 +7656,17 @@ noscript { | ||||
|         font-size: 17px; | ||||
|         line-height: 22px; | ||||
|         color: $primary-text-color; | ||||
|         font-weight: 700; | ||||
|         font-weight: 600; | ||||
|         overflow: hidden; | ||||
|         white-space: nowrap; | ||||
|         text-overflow: ellipsis; | ||||
| 
 | ||||
|         small { | ||||
|           display: block; | ||||
|           font-size: 15px; | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           gap: 4px; | ||||
|           font-size: 14px; | ||||
|           line-height: 20px; | ||||
|           color: $darker-text-color; | ||||
|           font-weight: 400; | ||||
|           overflow: hidden; | ||||
| @ -7562,10 +7677,8 @@ noscript { | ||||
|           } | ||||
| 
 | ||||
|           .icon-lock { | ||||
|             height: 16px; | ||||
|             width: 16px; | ||||
|             position: relative; | ||||
|             top: 3px; | ||||
|             height: 18px; | ||||
|             width: 18px; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||