Admin tools: themed dashboard

This commit is contained in:
yu-i-i
2026-03-19 02:34:33 +01:00
parent 51dff8c20a
commit cf84ba650a
19 changed files with 383 additions and 190 deletions
@@ -13,15 +13,15 @@ function ManageProjectsRoot() {
if (!isReady) return null
return (
<SplitTestProvider>
<UserSettingsProvider>
<UserListProvider>
<ProjectListProvider projectsOwnerId={null}>
<UserListProvider>
<ProjectListProvider projectsOwnerId={null}>
<SplitTestProvider>
<UserSettingsProvider>
<ProjectListRoot />
</ProjectListProvider>
</UserListProvider>
</UserSettingsProvider>
</SplitTestProvider>
</UserSettingsProvider>
</SplitTestProvider>
</ProjectListProvider>
</UserListProvider>
)
}
@@ -28,15 +28,15 @@ function ManageUsersRoot() {
if (!isReady) return null
return (
<SplitTestProvider>
<UserSettingsProvider>
<UsersPageProvider>
<UserListProvider>
<UsersPageProvider>
<UserListProvider>
<SplitTestProvider>
<UserSettingsProvider>
<UsersPageSelector />
</UserListProvider>
</UsersPageProvider>
</UserSettingsProvider>
</SplitTestProvider>
</UserSettingsProvider>
</SplitTestProvider>
</UserListProvider>
</UsersPageProvider>
)
}
@@ -15,11 +15,13 @@ import DefaultNavbar from '@/shared/components/navbar/default-navbar'
import Footer from '@/shared/components/footer/footer'
import SidebarDsNav from './sidebar/sidebar-ds-nav'
import overleafLogo from '@/shared/svgs/overleaf-a-ds-solution-mallard.svg'
import overleafLogoDark from '@/shared/svgs/overleaf-a-ds-solution-mallard-dark.svg'
import { getUserName } from '../util/user'
import { useProjectListContext } from '../context/project-list-context'
import { useUserIdentityContext } from '../../user-list/context/user-identity-context'
import Pagination from '@/shared/components/pagination-cep'
import ProjectListSummary from './project-list-summary'
import { useActiveOverallTheme } from '@/shared/hooks/use-active-overall-theme'
export function ProjectListDsNav() {
@@ -39,6 +41,7 @@ export function ProjectListDsNav() {
totalPages,
} = useProjectListContext()
const { getUserNameById } = useUserIdentityContext()
const activeOverallTheme = useActiveOverallTheme('themed-project-dashboard')
const userName = projectsOwnerId ? getUserNameById(projectsOwnerId) : t('all_users')
const tableTopArea = (
@@ -59,7 +62,9 @@ export function ProjectListDsNav() {
<div className="manage-projects-page">
<DefaultNavbar
{...navbarProps}
overleafLogo={overleafLogo}
overleafLogo={
activeOverallTheme === 'dark' ? overleafLogoDark : overleafLogo
}
showCloseIcon
/>
</div>
@@ -1,7 +1,14 @@
import { useTranslation } from 'react-i18next'
import OLFormSelect from '@/shared/components/ol/ol-form-select'
import {
Dropdown,
DropdownMenu,
DropdownItem,
DropdownToggle,
} from '@/shared/components/dropdown/dropdown-menu'
import { useProjectListContext } from '../context/project-list-context'
const OPTIONS = [20, 40, 80]
export default function ProjectListSummary() {
const {
visibleProjects,
@@ -14,37 +21,40 @@ export default function ProjectListSummary() {
return (
<div className="text-center">
<p>
<span aria-live="polite">
{t('showing_x_out_of_n_projects', {
x: visibleProjects.length,
n: visibleProjects.length + hiddenProjectsCount,
})}
</span>
<span className="mx-2">·</span>
<span className="d-inline-flex gap-1">
<OLFormSelect
name="projects_per_page"
value={projectsPerPage}
onChange={(e) => setProjectsPerPage(Number(e.target.value))}
style={{
width: 'auto',
border: '1px solid #ccc',
background: 'var(--green-10)',
padding: '0 0.2rem',
boxShadow: 'none',
cursor: 'pointer',
}}
>
<option value={20}>20</option>
<option value={40}>40</option>
<option value={80}>80</option>
</OLFormSelect>
<span>
{t('per_page')}
</span>
<Dropdown>
<DropdownToggle
as="span"
className="entries-per-page-toggle"
>
{projectsPerPage}
</DropdownToggle>
<DropdownMenu>
{OPTIONS.map((value) => (
<DropdownItem
key={value}
active={value === projectsPerPage}
onClick={() => setProjectsPerPage(value)}
>
{value}
</DropdownItem>
))}
</DropdownMenu>
</Dropdown>
<span>{t('per_page')}</span>
</span>
</p>
</div>
)
}
@@ -27,7 +27,7 @@ function SidebarDsNav() {
const { sessionUser } = getMeta('ol-navbar')
const { containerRef, scrolledUp } = useScrolled()
const themedDsNav = useFeatureFlag('themed-project-dashboard')
const { getUserNameById } = useUserIdentityContext()
const { projectsOwnerId } = useProjectListContext()
@@ -17,7 +17,9 @@ import DefaultNavbar from '@/shared/components/navbar/default-navbar'
import Footer from '@/shared/components/footer/footer'
import SidebarDsNav from './sidebar/sidebar-ds-nav'
import overleafLogo from '@/shared/svgs/overleaf-a-ds-solution-mallard.svg'
import overleafLogoDark from '@/shared/svgs/overleaf-a-ds-solution-mallard-dark.svg'
import CookieBanner from '@/shared/components/cookie-banner'
import { useActiveOverallTheme } from '@/shared/hooks/use-active-overall-theme'
import Pagination from '@/shared/components/pagination-cep'
import UserListSummary from './user-list-summary'
@@ -36,6 +38,7 @@ export function UserListDsNav() {
setCurrentPage,
totalPages,
} = useUserListContext()
const activeOverallTheme = useActiveOverallTheme('themed-project-dashboard')
const tableTopArea = (
<div className="pt-2 pb-3 d-md-none d-flex gap-2">
@@ -54,7 +57,9 @@ export function UserListDsNav() {
<div className="user-ds-nav-page website-redesign">
<DefaultNavbar
{...navbarProps}
overleafLogo={overleafLogo}
overleafLogo={
activeOverallTheme === 'dark' ? overleafLogoDark : overleafLogo
}
showCloseIcon
/>
<div className="user-list-wrapper">
@@ -1,7 +1,14 @@
import { useTranslation } from 'react-i18next'
import OLFormSelect from '@/shared/components/ol/ol-form-select'
import {
Dropdown,
DropdownMenu,
DropdownItem,
DropdownToggle,
} from '@/shared/components/dropdown/dropdown-menu'
import { useUserListContext } from '../context/user-list-context'
const OPTIONS = [20, 40, 80]
export default function UserListSummary() {
const {
visibleUsers,
@@ -9,42 +16,45 @@ export default function UserListSummary() {
usersPerPage,
setUsersPerPage,
} = useUserListContext()
const { t } = useTranslation()
return (
<div className="text-center">
<p>
<span aria-live="polite">
{t('showing_x_out_of_n_users', {
x: visibleUsers.length,
n: visibleUsers.length + hiddenUsersCount,
})}
</span>
<span className="mx-2">·</span>
<span className="d-inline-flex gap-1">
<OLFormSelect
name="users_per_page"
value={usersPerPage}
onChange={(e) => setUsersPerPage(Number(e.target.value))}
style={{
width: 'auto',
border: '1px solid #ccc',
background: 'var(--green-10)',
padding: '0 0.2rem',
boxShadow: 'none',
cursor: 'pointer',
}}
>
<option value={20}>20</option>
<option value={40}>40</option>
<option value={80}>80</option>
</OLFormSelect>
<span>
{t('per_page')}
</span>
<Dropdown>
<DropdownToggle
as="span"
className="entries-per-page-toggle"
>
{usersPerPage}
</DropdownToggle>
<DropdownMenu>
{OPTIONS.map((value) => (
<DropdownItem
key={value}
active={value === usersPerPage}
onClick={() => setUsersPerPage(value)}
>
{value}
</DropdownItem>
))}
</DropdownMenu>
</Dropdown>
<span>{t('per_page')}</span>
</span>
</p>
</div>
)
}