Files
overleaf-cep/services/web/frontend/js/features/settings/components/emails-section.tsx
T
Rebeka Dekany 960eea7129 Bootstrap files and folders cleanup (#27692)
* Remove icons folder

* Create folders for badge, button, and dropdown components

* Remove Bootstrap 5 from test

* Rename `getBootstrap5Breakpoint` to `getBootstrapBreakpoint`

* Cleanup and update BS 5 comments

* Move components to the shared folder

* Rename `tooltips-bs5` to `tooltip`

* Remove `-bs5` suffix

* Fix path

* Delete BS3 version file

* Rename `_form_marketing-bootstrap-5` to `_form_marketing`

* Delete BS3 version file

* Rename `_contact_general_modal-marketing-bootstrap-5` to `_contact_general_modal-marketing`

* Delete BS3 version file

* Rename `_contact_modal-marketing-bootstrap-5` to `_contact_modal-marketing`

* Delete BS3 version file

* Rename `thin-footer-bootstrap-5` to `thin-footer`

* Delete BS3 version file

* Rename `language-picker-bootstrap-5` to `language-picker`

* Rename `fat-footer-react-bootstrap-5` to `fat-footer-react`

* Delete BS3 version file

* Rename `navbar-marketing-bootstrap-5` to `navbar-marketing`

* Rename `navbar-marketing-react-bootstrap-5` to `navbar-marketing-react`

* Delete BS3 version file

* Rename `layout-website-redesign-cms-bootstrap-5` to `layout-website-redesign-cms`

* Source format

* Fix path

GitOrigin-RevId: cf0f5db7c84cf545c69213dcc271d9ff17fe5db7
2025-08-11 08:06:16 +00:00

108 lines
3.3 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 OLSpinner from '@/shared/components/ol/ol-spinner'
import { LeaversSurveyAlert } from './leavers-survey-alert'
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">
<OLSpinner size="sm" /> {t('loading')}...
</div>
</div>
) : (
<>
{sortedUserEmails.map(userEmail => (
<Fragment key={userEmail.email}>
<EmailsRow userEmailData={userEmail} primary={primary} />
<div className="horizontal-divider" />
</Fragment>
))}
</>
)}
{isInitializingSuccess && <LeaversSurveyAlert />}
{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