mirror of
https://github.com/overleaf/overleaf.git
synced 2025-12-05 01:10:29 +00:00
* Update test for the loading spinner component * Create a story for the loading spinner component * Move role and use CSS for spacing instead * Add a classname prop * Reuse LoadingSpinner * Use OLSpinner instead of Spinner * Use data-testid since status role was moved * Wait for journals to load * Use `isLoading` prop instead and fix the button's height * Use `isLoading` prop instead * Use LoadingSpinner instead and remove spacing * Update test for the loading spinner component * Use `isLoading` prop instead * Add aria-describedby to layout button for processing state * Replace `spinner` to `ol-spinner` * Scope status * Remove redundant `div.loading` --------- Co-authored-by: Antoine Clausse <antoine.clausse@overleaf.com> GitOrigin-RevId: 8f43b991f8f458b2abd5a4661913ac9b972d892a
106 lines
3.2 KiB
TypeScript
106 lines
3.2 KiB
TypeScript
import { Fragment } from 'react'
|
|
import { useTranslation, Trans } from 'react-i18next'
|
|
import getMeta from '../../../utils/meta'
|
|
import {
|
|
UserEmailsProvider,
|
|
useUserEmailsContext,
|
|
} from '../context/user-email-context'
|
|
import EmailsHeader from './emails/header'
|
|
import EmailsRow from './emails/row'
|
|
import AddEmail from './emails/add-email'
|
|
import OLNotification from '@/shared/components/ol/ol-notification'
|
|
import LoadingSpinner from '@/shared/components/loading-spinner'
|
|
|
|
function EmailsSectionContent() {
|
|
const { t } = useTranslation()
|
|
const {
|
|
state: { data: userEmailsData },
|
|
isInitializing,
|
|
isInitializingError,
|
|
isInitializingSuccess,
|
|
} = useUserEmailsContext()
|
|
const userEmails = Object.values(userEmailsData.byId)
|
|
const primary = userEmails.find(userEmail => userEmail.default)
|
|
|
|
// Only show the "add email" button if the user has permission to add a secondary email
|
|
const hideAddSecondaryEmail = getMeta('ol-cannot-add-secondary-email')
|
|
|
|
// Sort emails: primary first, then confirmed secondary emails, then unconfirmed secondary emails
|
|
const sortedUserEmails = [...userEmails].sort((a, b) => {
|
|
// Primary email comes first
|
|
if (a.default) return -1
|
|
if (b.default) return 1
|
|
|
|
// Then sort by confirmation status
|
|
if (a.confirmedAt && !b.confirmedAt) return -1
|
|
if (!a.confirmedAt && b.confirmedAt) return 1
|
|
|
|
// If both have the same status, sort by email string
|
|
return a.email.localeCompare(b.email)
|
|
})
|
|
|
|
return (
|
|
<>
|
|
<h2 className="h3">{t('emails_and_affiliations_title')}</h2>
|
|
<p className="small">{t('emails_and_affiliations_explanation')}</p>
|
|
<p className="small">
|
|
<Trans
|
|
i18nKey="change_primary_email_address_instructions"
|
|
components={[
|
|
// eslint-disable-next-line react/jsx-key
|
|
<strong />,
|
|
// eslint-disable-next-line jsx-a11y/anchor-has-content, react/jsx-key
|
|
<a
|
|
href="/learn/how-to/Managing_your_Overleaf_emails"
|
|
target="_blank"
|
|
/>,
|
|
]}
|
|
/>
|
|
</p>
|
|
<>
|
|
<EmailsHeader />
|
|
{isInitializing ? (
|
|
<div className="affiliations-table-row-highlighted">
|
|
<div className="affiliations-table-cell text-center">
|
|
<LoadingSpinner size="sm" />
|
|
</div>
|
|
</div>
|
|
) : (
|
|
<>
|
|
{sortedUserEmails.map(userEmail => (
|
|
<Fragment key={userEmail.email}>
|
|
<EmailsRow userEmailData={userEmail} primary={primary} />
|
|
<div className="horizontal-divider" />
|
|
</Fragment>
|
|
))}
|
|
</>
|
|
)}
|
|
{isInitializingSuccess && !hideAddSecondaryEmail && <AddEmail />}
|
|
{isInitializingError && (
|
|
<OLNotification
|
|
type="error"
|
|
content={t('error_performing_request')}
|
|
/>
|
|
)}
|
|
</>
|
|
</>
|
|
)
|
|
}
|
|
|
|
function EmailsSection() {
|
|
const { hasAffiliationsFeature } = getMeta('ol-ExposedSettings')
|
|
if (!hasAffiliationsFeature) {
|
|
return null
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<UserEmailsProvider>
|
|
<EmailsSectionContent />
|
|
</UserEmailsProvider>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default EmailsSection
|