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

View File

@@ -1169,7 +1169,9 @@ module.exports = {
}, },
splitTestOverrides: { splitTestOverrides: {
// new-fancy-feature': 'enabled', ...(process.env.OVERLEAF_THEMED_DASHBOARD?.toLowerCase() === 'true' ? {
'themed-project-dashboard': 'enabled',
} : {}),
...(process.env.OVERLEAF_HISTORY_RESTORE?.toLowerCase() === 'true' ? { ...(process.env.OVERLEAF_HISTORY_RESTORE?.toLowerCase() === 'true' ? {
'history-ranges-support': 'enabled', 'history-ranges-support': 'enabled',
'revert-file': 'enabled', 'revert-file': 'enabled',

View File

@@ -0,0 +1,26 @@
.entries-per-page-toggle {
cursor: pointer;
background-color: var(--bg-light-tertiary);
padding: 0 var(--spacing-02);
border-radius: var(--border-radius-medium);
&:hover {
background-color: var(--bg-accent-03);
}
& + .dropdown-menu {
min-width: auto;
width: auto;
}
}
.entries-per-page-toggle::after {
display: none;
}
@include theme('default') {
.entries-per-page-toggle {
background-color: var(--bg-dark-tertiary);
&:hover {
background-color: var(--green-70);
}
}
}

View File

@@ -5,3 +5,12 @@
} }
} }
} }
@include theme('default') {
#manage-projects-root {
.website-redesign-navbar,
.website-redesign .navbar-default {
@include navbar-dark;
}
}
}

View File

@@ -0,0 +1,15 @@
@include theme('default') {
#manage-users-root {
.website-redesign-navbar,
.website-redesign .navbar-default {
@include navbar-dark;
}
}
.user-ds-nav-page {
@include dark-dropdown-menu;
.table,
.table-container {
@include dark-table;
}
}
}

View File

@@ -0,0 +1,33 @@
@include theme('default') {
.project-ds-nav-page,
.user-ds-nav-page {
.pagination {
> li {
> a,
> span,
> button {
color: var(--content-primary-inverse);
background-color: var(--bg-secondary);
border-color: var(--border-primary);
&:hover,
&:focus {
color: var(--content-primary-inverse);
background-color: var(--bg-tertiary);
}
}
}
> .active > a,
> .active > span,
> .active > button {
&,
&:hover,
&:focus {
color: var(--content-primary-inverse);
background-color: var(--green-70);
}
}
}
}
}

View File

@@ -1,9 +1,42 @@
:root {
--ds-nav-active-bg: var(--bg-accent-03);
--ds-nav-active-color: var(--green-60);
--theme-toggle-selected-background: var(--green-20);
--ds-nav-content-bg-secondary: var(--bg-light-secondary);
--table-icon-bg-hover: 27 34 44;
--themed-dashboard-popover-bg: var(--bg-dark-primary);
--themed-dashboard-popover-color: var(--content-primary-dark);
--themed-dashboard-popover-link-color: var(--link-ui-dark);
--themed-dashboard-popover-link-hover-color: var(--link-ui-hover-dark);
--themed-dashboard-popover-link-visited-color: var(--link-ui-visited-dark);
--ds-nav-color-scheme: light;
@include theme('default') {
--ds-nav-active-bg: var(--green-70);
--ds-nav-active-color: var(--green-10);
--theme-toggle-selected-background: var(--green-70);
--ds-nav-content-bg-secondary: var(--bg-dark-secondary);
--table-icon-bg-hover: 255 255 255;
--themed-dashboard-popover-bg: var(--bg-light-primary);
--themed-dashboard-popover-color: var(--content-primary);
--themed-dashboard-popover-link-color: var(--link-ui);
--themed-dashboard-popover-link-hover-color: var(--link-ui-hover);
--themed-dashboard-popover-link-visited-color: var(--link-ui-visited);
--ds-nav-color-scheme: dark;
}
}
body {
--ds-nav-bg-color: var(--bg-primary-themed);
--ds-nav-hover-bg: var(--bg-secondary-themed);
--ds-nav-color: var(--content-secondary-themed);
}
.user-ds-nav-page { .user-ds-nav-page {
display: flex; @include full-height-stacked-page;
flex-direction: column;
height: 100vh; color-scheme: var(--ds-nav-color-scheme);
height: 100dvh; color: var(--content-secondary-themed);
color: var(--content-secondary);
// NOTE-AC: This code can be eliminated when we remove sidebar-navigation-ui-update // NOTE-AC: This code can be eliminated when we remove sidebar-navigation-ui-update
--navbar-btn-padding-h: var(--spacing-06); --navbar-btn-padding-h: var(--spacing-06);
@@ -15,9 +48,6 @@
.navbar-default { .navbar-default {
position: relative; position: relative;
--navbar-toggler-expanded-bg: none;
--navbar-toggler-expanded-color: var(--content-secondary);
.navbar-header .navbar-logo { .navbar-header .navbar-logo {
@include media-breakpoint-up(md) { @include media-breakpoint-up(md) {
position: relative; position: relative;
@@ -48,7 +78,7 @@
.nav-item-help::before { .nav-item-help::before {
content: ''; content: '';
display: block; display: block;
border-top: 1px solid var(--border-divider); border-top: 1px solid var(--border-divider-themed);
margin: var(--spacing-07) var(--spacing-06); margin: var(--spacing-07) var(--spacing-06);
} }
@@ -63,8 +93,8 @@
border-radius: var(--border-radius-medium); border-radius: var(--border-radius-medium);
&.show { &.show {
background-color: var(--bg-accent-03); background-color: var(--ds-nav-active-bg);
color: var(--green-60); color: var(--ds-nav-active-color);
} }
} }
} }
@@ -78,6 +108,7 @@
} }
.user-list-wrapper { .user-list-wrapper {
background-color: var(--ds-nav-bg-color);
flex-grow: 1; flex-grow: 1;
display: flex; display: flex;
overflow-y: hidden; overflow-y: hidden;
@@ -95,6 +126,10 @@
.create-account-button-wrapper { .create-account-button-wrapper {
padding: 0 var(--spacing-08) var(--spacing-05) var(--spacing-05); padding: 0 var(--spacing-08) var(--spacing-05) var(--spacing-05);
border-bottom: solid 1px transparent; border-bottom: solid 1px transparent;
&.show-shadow {
border-bottom-color: var(--border-divider-themed);
}
} }
nav { nav {
@@ -117,39 +152,13 @@
border-top: solid 1px transparent; border-top: solid 1px transparent;
&.show-shadow { &.show-shadow {
border-top-color: var(--border-divider); border-top-color: var(--border-divider-themed);
} }
} }
}
.user-list-sidebar-survey-link { hr {
color: var(--content-secondary) !important; border-top: 1px solid var(--border-divider-themed);
}
.survey-notification {
background-color: var(--bg-light-secondary);
color: var(--content-secondary);
box-shadow: none;
border-radius: var(--border-radius-large);
position: relative;
.user-notification-close {
border: none;
padding: 0;
background: none;
position: absolute;
top: var(--spacing-07);
right: var(--spacing-07);
color: inherit;
}
p {
margin-bottom: var(--spacing-03);
}
}
.user-list-sidebar-survey-wrapper .user-notifications {
margin-bottom: var(--spacing-05);
}
} }
ul.user-list-filters { ul.user-list-filters {
@@ -163,7 +172,7 @@
> button { > button {
width: 100%; width: 100%;
text-align: left; text-align: left;
color: var(--content-secondary); color: var(--ds-nav-color);
background: none; background: none;
border-radius: var(--border-radius-medium); border-radius: var(--border-radius-medium);
border: none; border: none;
@@ -171,18 +180,19 @@
} }
&:hover button { &:hover button {
background-color: var(--bg-light-secondary); background-color: var(--ds-nav-hover-bg);
} }
&.active button { &.active button {
background-color: var(--bg-accent-03); background-color: var(--ds-nav-active-bg);
color: var(--green-60); color: var(--ds-nav-active-color);
font-weight: bold; font-weight: bold;
} }
} }
.dropdown-header { .dropdown-header {
font-weight: bold; font-weight: bold;
color: var(--ds-nav-color);
} }
> li.tag { > li.tag {
@@ -195,7 +205,7 @@
button.dropdown-toggle { button.dropdown-toggle {
border-radius: var(--border-radius-full); border-radius: var(--border-radius-full);
border: none; border: none;
color: var(--content-secondary); color: var(--ds-nav-color);
height: 20px; height: 20px;
width: 20px; width: 20px;
padding: 0; padding: 0;
@@ -224,7 +234,7 @@
.user-dash-table { .user-dash-table {
.btn-link { .btn-link {
color: var(--content-secondary); color: var(--content-secondary-themed);
height: var(--spacing-08); height: var(--spacing-08);
width: var(--spacing-08); width: var(--spacing-08);
border-radius: 100%; border-radius: 100%;
@@ -233,12 +243,12 @@
&:hover, &:hover,
&:focus { &:focus {
background-color: #d9d9d9 !important; background-color: rgb(var(--table-icon-bg-hover) / 8%) !important;
} }
} }
.dash-cell-name a { .dash-cell-name a {
color: var(--content-secondary) !important; color: var(--content-primary-themed) !important;
} }
} }
@@ -249,7 +259,7 @@
> * { > * {
@include media-breakpoint-up(md) { @include media-breakpoint-up(md) {
border-left: 1px solid var(--border-divider); border-left: 1px solid var(--border-divider-themed);
} }
} }
@@ -257,17 +267,16 @@
flex-grow: 1; flex-grow: 1;
overflow-y: auto; overflow-y: auto;
position: relative; position: relative;
background-color: var(--bg-light-secondary); background-color: var(--ds-nav-content-bg-secondary);
@include media-breakpoint-up(md) { @include media-breakpoint-up(md) {
border-top-left-radius: var(--border-radius-large); border-top-left-radius: var(--border-radius-large);
border-top: 1px solid var(--border-divider); border-top: 1px solid var(--border-divider-themed);
} }
} }
.cookie-banner { .cookie-banner {
position: static; position: static;
background-color: var(--bg-light-primary);
// Remove the parts of the shadow that stick out of the sides // Remove the parts of the shadow that stick out of the sides
clip-path: inset(-13px 0 0 0); clip-path: inset(-13px 0 0 0);
@@ -276,11 +285,16 @@
z-index: auto; z-index: auto;
} }
} }
.btn-link {
--link-color: var(--link-web-themed);
--link-hover-color: var(--link-web-hover-themed);
}
} }
.ds-nav-icon-dropdown { .ds-nav-icon-dropdown {
.dropdown-toggle { .dropdown-toggle {
color: var(--content-secondary); color: var(--ds-nav-color);
background: none; background: none;
height: 44px; height: 44px;
width: 44px; width: 44px;
@@ -288,12 +302,12 @@
overflow: hidden; overflow: hidden;
&:hover { &:hover {
background-color: var(--bg-light-secondary); background-color: var(--ds-nav-hover-bg);
} }
&.show { &.show {
background-color: var(--bg-accent-03); background-color: var(--ds-nav-active-bg);
color: var(--green-60); color: var(--ds-nav-active-color);
} }
&::after { &::after {
@@ -314,6 +328,7 @@
} }
.ds-nav-ds-name { .ds-nav-ds-name {
color: var(--ds-nav-color);
margin-bottom: var(--spacing-05); margin-bottom: var(--spacing-05);
span { span {
@@ -323,4 +338,59 @@
@include body-xs; @include body-xs;
} }
} }
.add-affiliation {
color: var(--ds-nav-color);
}
}
.theme-toggle {
display: flex;
justify-content: space-between;
cursor: default !important;
align-items: center;
&:hover {
background-color: transparent !important;
color: var(--content-primary-themed);
}
legend {
font-size: var(--font-size-02);
line-height: var(--line-height-02);
margin-bottom: 0;
}
}
.theme-toggle-radios {
display: flex;
border-radius: var(--border-radius-full);
background-color: var(--bg-secondary-themed);
padding: var(--spacing-01);
gap: var(--spacing-01);
}
.theme-toggle-radio {
display: flex;
input {
all: unset;
}
label {
display: flex;
padding: var(--spacing-03);
margin-bottom: 0;
border-radius: var(--border-radius-full);
cursor: pointer;
color: var(--content-primary-themed);
}
.material-symbols {
font-size: var(--font-size-03);
}
input:checked + label {
background-color: var(--theme-toggle-selected-background);
}
} }

View File

@@ -17,7 +17,7 @@
padding: 0 var(--spacing-02); padding: 0 var(--spacing-02);
} }
#user-list-root .user-notifications ul { #manage-users-root .user-notifications ul {
margin-bottom: 0; margin-bottom: 0;
} }
@@ -50,7 +50,7 @@
.user-list-title { .user-list-title {
@include heading-sm; @include heading-sm;
color: $content-secondary; color: var(--content-secondary-themed);
font-weight: bold; font-weight: bold;
min-width: 0; min-width: 0;
} }
@@ -241,7 +241,7 @@ ul.user-list-filters {
@include media-breakpoint-down(md) { @include media-breakpoint-down(md) {
tr:not(:last-child) { tr:not(:last-child) {
border-bottom: 1px solid $table-border-color; border-bottom: 1px solid var(--border-divider-themed);
} }
td { td {
@@ -260,7 +260,7 @@ ul.user-list-filters {
.table-header-sort-btn { .table-header-sort-btn {
border: 0; border: 0;
text-align: left; text-align: left;
color: var(--content-secondary); color: var(--content-secondary-themed);
background-color: transparent; background-color: transparent;
padding: 0; padding: 0;
font-weight: bold; font-weight: bold;
@@ -271,7 +271,7 @@ ul.user-list-filters {
&:hover, &:hover,
&:focus { &:focus {
color: var(--content-secondary); color: var(--content-secondary-themed);
text-decoration: none; text-decoration: none;
} }
@@ -283,11 +283,8 @@ ul.user-list-filters {
.dash-row-admin { .dash-row-admin {
font-weight: bold; font-weight: bold;
td { a {
a { color: var(--content-danger) !important;
color: darkred !important;
--bs-link-color: darkred !important;
}
} }
} }
@@ -474,7 +471,7 @@ ul.user-list-filters {
} }
.dash-cell-actions { .dash-cell-actions {
width: 12; width: 12%;
} }
} }
@@ -532,57 +529,18 @@ ul.user-list-filters {
} }
} }
.survey-notification {
display: flex;
flex-wrap: wrap;
padding: var(--spacing-06);
background-color: var(--bg-dark-tertiary);
border-color: transparent;
color: var(--neutral-20);
box-shadow: 2px 4px 6px rgb(0 0 0 / 25%);
border-radius: var(--border-radius-base);
@include media-breakpoint-up(md) {
flex-wrap: nowrap;
}
button.close {
@extend .text-white;
padding: 0;
}
}
.user-list-sidebar-survey-wrapper {
.survey-notification {
font-size: var(--font-size-02);
a {
text-decoration: none;
}
}
@include media-breakpoint-down(md) {
.survey-notification {
font-size: unset;
.user-list-sidebar-survey-link {
display: block;
align-items: center;
min-width: 48px;
min-height: 48px;
padding-top: var(--spacing-07);
}
}
}
}
.user-list-load-more-button {
margin-bottom: var(--spacing-05);
}
form.user-search { form.user-search {
.form-group { .form-group {
margin-bottom: 0; margin-bottom: 0;
} }
} }
@include theme('default') {
.dash-cell-email a,
.dash-cell-email-date a {
color: var(--green-40);
}
.dash-row-admin a {
color: var(--content-danger-dark) !important;
}
}

View File

@@ -9,6 +9,9 @@
@import 'symbol-palette'; @import 'symbol-palette';
@import 'writefull'; @import 'writefull';
@import 'labs'; @import 'labs';
@import 'admin-tools/entries-per-page';
@import 'admin-tools/user-list'; @import 'admin-tools/user-list';
@import 'admin-tools/user-list-ds-nav'; @import 'admin-tools/user-list-ds-nav';
@import 'admin-tools/manage-projects-page'; @import 'admin-tools/manage-projects-page';
@import 'admin-tools/manage-users-page';
@import 'admin-tools/pagination-dark';

View File

@@ -4,21 +4,32 @@ import { fileURLToPath } from 'node:url'
import { expressify } from '@overleaf/promise-utils' import { expressify } from '@overleaf/promise-utils'
import logger from '@overleaf/logger' import logger from '@overleaf/logger'
import Metrics from '@overleaf/metrics' import Metrics from '@overleaf/metrics'
import SessionManager from '../../../../app/src/Features/Authentication/SessionManager.mjs'
import PrivilegeLevels from '../../../../app/src/Features/Authorization/PrivilegeLevels.mjs'
import ProjectHelper from '../../../../app/src/Features/Project/ProjectHelper.mjs' import ProjectHelper from '../../../../app/src/Features/Project/ProjectHelper.mjs'
import ProjectGetter from '../../../../app/src/Features/Project/ProjectGetter.mjs' import ProjectGetter from '../../../../app/src/Features/Project/ProjectGetter.mjs'
import PrivilegeLevels from '../../../../app/src/Features/Authorization/PrivilegeLevels.mjs' import ProjectDeleter from '../../../../app/src/Features/Project/ProjectDeleter.mjs'
import SessionManager from '../../../../app/src/Features/Authentication/SessionManager.mjs' import UserSettingsHelper from '../../../../app/src/Features/Project/UserSettingsHelper.mjs'
import UserGetter from '../../../../app/src/Features/User/UserGetter.mjs' import UserGetter from '../../../../app/src/Features/User/UserGetter.mjs'
import { OError } from '../../../../app/src/Features/Errors/Errors.js'
import { User } from '../../../../app/src/models/User.mjs' import { User } from '../../../../app/src/models/User.mjs'
import { Project } from '../../../../app/src/models/Project.mjs' import { Project } from '../../../../app/src/models/Project.mjs'
import { DeletedProject } from '../../../../app/src/models/DeletedProject.mjs' import { DeletedProject } from '../../../../app/src/models/DeletedProject.mjs'
import ProjectDeleter from '../../../../app/src/Features/Project/ProjectDeleter.mjs' import { OError } from '../../../../app/src/Features/Errors/Errors.js'
import HttpErrorHandler from '../../../../app/src/Features/Errors/HttpErrorHandler.mjs' import HttpErrorHandler from '../../../../app/src/Features/Errors/HttpErrorHandler.mjs'
import SplitTestHandler from '../../../../app/src/Features/SplitTests/SplitTestHandler.mjs'
const __dirname = Path.dirname(fileURLToPath(import.meta.url)) const __dirname = Path.dirname(fileURLToPath(import.meta.url))
function cleanupSession(req) {
// cleanup redirects at the end of the redirect chain
delete req.session.postCheckoutRedirect
delete req.session.postLoginRedirect
delete req.session.postOnboardingRedirect
}
async function manageProjectsPage(req, res, next) { async function manageProjectsPage(req, res, next) {
cleanupSession(req)
const projectsBlobPending = _getProjects().catch(err => { const projectsBlobPending = _getProjects().catch(err => {
logger.err({ err }, 'projects listing in background failed') logger.err({ err }, 'projects listing in background failed')
return undefined return undefined
@@ -30,8 +41,24 @@ async function manageProjectsPage(req, res, next) {
status: prefetchedProjectsBlob ? 'success' : 'error', status: prefetchedProjectsBlob ? 'success' : 'error',
}) })
const userId = SessionManager.getLoggedInUserId(req.session)
const user = await User.findById(userId, 'ace')
const userSettings = await UserSettingsHelper.buildUserSettings(
req,
res,
user
)
await SplitTestHandler.promises.getAssignment(
req,
res,
'themed-project-dashboard'
)
res.render(Path.resolve(__dirname, '../views/manage-projects-react'), { res.render(Path.resolve(__dirname, '../views/manage-projects-react'), {
title: 'Manage Projects', title: 'Manage Projects',
userSettings,
prefetchedProjectsBlob, prefetchedProjectsBlob,
}) })
} }

View File

@@ -16,12 +16,14 @@ import OneTimeTokenHandler from '../../../../app/src/Features/Security/OneTimeTo
import UserGetter from '../../../../app/src/Features/User/UserGetter.mjs' import UserGetter from '../../../../app/src/Features/User/UserGetter.mjs'
import UserUpdater from '../../../../app/src/Features/User/UserUpdater.mjs' import UserUpdater from '../../../../app/src/Features/User/UserUpdater.mjs'
import UserDeleter from '../../../../app/src/Features/User/UserDeleter.mjs' import UserDeleter from '../../../../app/src/Features/User/UserDeleter.mjs'
import UserSettingsHelper from '../../../../app/src/Features/Project/UserSettingsHelper.mjs'
import ProjectDeleter from '../../../../app/src/Features/Project/ProjectDeleter.mjs' import ProjectDeleter from '../../../../app/src/Features/Project/ProjectDeleter.mjs'
import OwnershipTransferHandler from '../../../../app/src/Features/Collaborators/OwnershipTransferHandler.mjs' import OwnershipTransferHandler from '../../../../app/src/Features/Collaborators/OwnershipTransferHandler.mjs'
import HttpErrorHandler from '../../../../app/src/Features/Errors/HttpErrorHandler.mjs' import HttpErrorHandler from '../../../../app/src/Features/Errors/HttpErrorHandler.mjs'
import ErrorController from '../../../../app/src/Features/Errors/ErrorController.mjs' import ErrorController from '../../../../app/src/Features/Errors/ErrorController.mjs'
import Errors, { OError } from '../../../../app/src/Features/Errors/Errors.js' import Errors, { OError } from '../../../../app/src/Features/Errors/Errors.js'
import { db } from '../../../../app/src/infrastructure/mongodb.mjs' import { db } from '../../../../app/src/infrastructure/mongodb.mjs'
import SplitTestHandler from '../../../../app/src/Features/SplitTests/SplitTestHandler.mjs'
const __dirname = Path.dirname(fileURLToPath(import.meta.url)) const __dirname = Path.dirname(fileURLToPath(import.meta.url))
@@ -71,8 +73,6 @@ function cleanupSession(req) {
async function manageUsersPage(req, res, next) { async function manageUsersPage(req, res, next) {
cleanupSession(req) cleanupSession(req)
const userId = SessionManager.getLoggedInUserId(req.session)
const usersBlobPending = _getUsers().catch(err => { const usersBlobPending = _getUsers().catch(err => {
logger.err({ err }, 'users listing in background failed') logger.err({ err }, 'users listing in background failed')
return undefined return undefined
@@ -84,8 +84,24 @@ async function manageUsersPage(req, res, next) {
status: prefetchedUsersBlob ? 'success' : 'error', status: prefetchedUsersBlob ? 'success' : 'error',
}) })
await SplitTestHandler.promises.getAssignment(
req,
res,
'themed-project-dashboard'
)
const userId = SessionManager.getLoggedInUserId(req.session)
const user = await User.findById(userId, 'ace')
const userSettings = await UserSettingsHelper.buildUserSettings(
req,
res,
user
)
res.render(Path.resolve(__dirname, '../views/manage-users-react'), { res.render(Path.resolve(__dirname, '../views/manage-users-react'), {
title: 'Manage Users', title: 'Manage Users',
userSettings,
prefetchedUsersBlob, prefetchedUsersBlob,
availableAuthMethods, availableAuthMethods,
userDetailsUpdatedOnLogin, userDetailsUpdatedOnLogin,

View File

@@ -15,6 +15,8 @@ block append meta
data-type='json' data-type='json'
content=prefetchedProjectsBlob content=prefetchedProjectsBlob
) )
meta(name='ol-userSettings' data-type='json' content=userSettings)
meta(name='ol-overallThemes' data-type='json' content=overallThemes)
if suggestedLanguageSubdomainConfig if suggestedLanguageSubdomainConfig
meta( meta(
name='ol-suggestedLanguage' name='ol-suggestedLanguage'

View File

@@ -16,6 +16,8 @@ block append meta
data-type='json' data-type='json'
content=prefetchedUsersBlob content=prefetchedUsersBlob
) )
meta(name='ol-userSettings' data-type='json' content=userSettings)
meta(name='ol-overallThemes' data-type='json' content=overallThemes)
if suggestedLanguageSubdomainConfig if suggestedLanguageSubdomainConfig
meta( meta(
name='ol-suggestedLanguage' name='ol-suggestedLanguage'

View File

@@ -13,15 +13,15 @@ function ManageProjectsRoot() {
if (!isReady) return null if (!isReady) return null
return ( return (
<SplitTestProvider> <UserListProvider>
<UserSettingsProvider> <ProjectListProvider projectsOwnerId={null}>
<UserListProvider> <SplitTestProvider>
<ProjectListProvider projectsOwnerId={null}> <UserSettingsProvider>
<ProjectListRoot /> <ProjectListRoot />
</ProjectListProvider> </UserSettingsProvider>
</UserListProvider> </SplitTestProvider>
</UserSettingsProvider> </ProjectListProvider>
</SplitTestProvider> </UserListProvider>
) )
} }

View File

@@ -28,15 +28,15 @@ function ManageUsersRoot() {
if (!isReady) return null if (!isReady) return null
return ( return (
<SplitTestProvider> <UsersPageProvider>
<UserSettingsProvider> <UserListProvider>
<UsersPageProvider> <SplitTestProvider>
<UserListProvider> <UserSettingsProvider>
<UsersPageSelector /> <UsersPageSelector />
</UserListProvider> </UserSettingsProvider>
</UsersPageProvider> </SplitTestProvider>
</UserSettingsProvider> </UserListProvider>
</SplitTestProvider> </UsersPageProvider>
) )
} }

View File

@@ -15,11 +15,13 @@ import DefaultNavbar from '@/shared/components/navbar/default-navbar'
import Footer from '@/shared/components/footer/footer' import Footer from '@/shared/components/footer/footer'
import SidebarDsNav from './sidebar/sidebar-ds-nav' import SidebarDsNav from './sidebar/sidebar-ds-nav'
import overleafLogo from '@/shared/svgs/overleaf-a-ds-solution-mallard.svg' 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 { getUserName } from '../util/user'
import { useProjectListContext } from '../context/project-list-context' import { useProjectListContext } from '../context/project-list-context'
import { useUserIdentityContext } from '../../user-list/context/user-identity-context' import { useUserIdentityContext } from '../../user-list/context/user-identity-context'
import Pagination from '@/shared/components/pagination-cep' import Pagination from '@/shared/components/pagination-cep'
import ProjectListSummary from './project-list-summary' import ProjectListSummary from './project-list-summary'
import { useActiveOverallTheme } from '@/shared/hooks/use-active-overall-theme'
export function ProjectListDsNav() { export function ProjectListDsNav() {
@@ -39,6 +41,7 @@ export function ProjectListDsNav() {
totalPages, totalPages,
} = useProjectListContext() } = useProjectListContext()
const { getUserNameById } = useUserIdentityContext() const { getUserNameById } = useUserIdentityContext()
const activeOverallTheme = useActiveOverallTheme('themed-project-dashboard')
const userName = projectsOwnerId ? getUserNameById(projectsOwnerId) : t('all_users') const userName = projectsOwnerId ? getUserNameById(projectsOwnerId) : t('all_users')
const tableTopArea = ( const tableTopArea = (
@@ -59,7 +62,9 @@ export function ProjectListDsNav() {
<div className="manage-projects-page"> <div className="manage-projects-page">
<DefaultNavbar <DefaultNavbar
{...navbarProps} {...navbarProps}
overleafLogo={overleafLogo} overleafLogo={
activeOverallTheme === 'dark' ? overleafLogoDark : overleafLogo
}
showCloseIcon showCloseIcon
/> />
</div> </div>

View File

@@ -1,7 +1,14 @@
import { useTranslation } from 'react-i18next' 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' import { useProjectListContext } from '../context/project-list-context'
const OPTIONS = [20, 40, 80]
export default function ProjectListSummary() { export default function ProjectListSummary() {
const { const {
visibleProjects, visibleProjects,
@@ -14,37 +21,40 @@ export default function ProjectListSummary() {
return ( return (
<div className="text-center"> <div className="text-center">
<p>
<span aria-live="polite"> <span aria-live="polite">
{t('showing_x_out_of_n_projects', { {t('showing_x_out_of_n_projects', {
x: visibleProjects.length, x: visibleProjects.length,
n: visibleProjects.length + hiddenProjectsCount, n: visibleProjects.length + hiddenProjectsCount,
})} })}
</span> </span>
<span className="mx-2">·</span> <span className="mx-2">·</span>
<span className="d-inline-flex gap-1"> <span className="d-inline-flex gap-1">
<OLFormSelect <Dropdown>
name="projects_per_page"
value={projectsPerPage} <DropdownToggle
onChange={(e) => setProjectsPerPage(Number(e.target.value))} as="span"
style={{ className="entries-per-page-toggle"
width: 'auto', >
border: '1px solid #ccc', {projectsPerPage}
background: 'var(--green-10)', </DropdownToggle>
padding: '0 0.2rem',
boxShadow: 'none', <DropdownMenu>
cursor: 'pointer', {OPTIONS.map((value) => (
}} <DropdownItem
> key={value}
<option value={20}>20</option> active={value === projectsPerPage}
<option value={40}>40</option> onClick={() => setProjectsPerPage(value)}
<option value={80}>80</option> >
</OLFormSelect> {value}
<span> </DropdownItem>
{t('per_page')} ))}
</span> </DropdownMenu>
</Dropdown>
<span>{t('per_page')}</span>
</span> </span>
</p>
</div> </div>
) )
} }

View File

@@ -27,7 +27,7 @@ function SidebarDsNav() {
const { sessionUser } = getMeta('ol-navbar') const { sessionUser } = getMeta('ol-navbar')
const { containerRef, scrolledUp } = useScrolled() const { containerRef, scrolledUp } = useScrolled()
const themedDsNav = useFeatureFlag('themed-project-dashboard') const themedDsNav = useFeatureFlag('themed-project-dashboard')
const { getUserNameById } = useUserIdentityContext() const { getUserNameById } = useUserIdentityContext()
const { projectsOwnerId } = useProjectListContext() const { projectsOwnerId } = useProjectListContext()

View File

@@ -17,7 +17,9 @@ import DefaultNavbar from '@/shared/components/navbar/default-navbar'
import Footer from '@/shared/components/footer/footer' import Footer from '@/shared/components/footer/footer'
import SidebarDsNav from './sidebar/sidebar-ds-nav' import SidebarDsNav from './sidebar/sidebar-ds-nav'
import overleafLogo from '@/shared/svgs/overleaf-a-ds-solution-mallard.svg' 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 CookieBanner from '@/shared/components/cookie-banner'
import { useActiveOverallTheme } from '@/shared/hooks/use-active-overall-theme'
import Pagination from '@/shared/components/pagination-cep' import Pagination from '@/shared/components/pagination-cep'
import UserListSummary from './user-list-summary' import UserListSummary from './user-list-summary'
@@ -36,6 +38,7 @@ export function UserListDsNav() {
setCurrentPage, setCurrentPage,
totalPages, totalPages,
} = useUserListContext() } = useUserListContext()
const activeOverallTheme = useActiveOverallTheme('themed-project-dashboard')
const tableTopArea = ( const tableTopArea = (
<div className="pt-2 pb-3 d-md-none d-flex gap-2"> <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"> <div className="user-ds-nav-page website-redesign">
<DefaultNavbar <DefaultNavbar
{...navbarProps} {...navbarProps}
overleafLogo={overleafLogo} overleafLogo={
activeOverallTheme === 'dark' ? overleafLogoDark : overleafLogo
}
showCloseIcon showCloseIcon
/> />
<div className="user-list-wrapper"> <div className="user-list-wrapper">

View File

@@ -1,7 +1,14 @@
import { useTranslation } from 'react-i18next' 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' import { useUserListContext } from '../context/user-list-context'
const OPTIONS = [20, 40, 80]
export default function UserListSummary() { export default function UserListSummary() {
const { const {
visibleUsers, visibleUsers,
@@ -9,42 +16,45 @@ export default function UserListSummary() {
usersPerPage, usersPerPage,
setUsersPerPage, setUsersPerPage,
} = useUserListContext() } = useUserListContext()
const { t } = useTranslation() const { t } = useTranslation()
return ( return (
<div className="text-center"> <div className="text-center">
<p>
<span aria-live="polite"> <span aria-live="polite">
{t('showing_x_out_of_n_users', { {t('showing_x_out_of_n_users', {
x: visibleUsers.length, x: visibleUsers.length,
n: visibleUsers.length + hiddenUsersCount, n: visibleUsers.length + hiddenUsersCount,
})} })}
</span> </span>
<span className="mx-2">·</span> <span className="mx-2">·</span>
<span className="d-inline-flex gap-1"> <span className="d-inline-flex gap-1">
<OLFormSelect <Dropdown>
name="users_per_page"
value={usersPerPage} <DropdownToggle
onChange={(e) => setUsersPerPage(Number(e.target.value))} as="span"
style={{ className="entries-per-page-toggle"
width: 'auto', >
border: '1px solid #ccc', {usersPerPage}
background: 'var(--green-10)', </DropdownToggle>
padding: '0 0.2rem',
boxShadow: 'none', <DropdownMenu>
cursor: 'pointer', {OPTIONS.map((value) => (
}} <DropdownItem
> key={value}
<option value={20}>20</option> active={value === usersPerPage}
<option value={40}>40</option> onClick={() => setUsersPerPage(value)}
<option value={80}>80</option> >
</OLFormSelect> {value}
<span> </DropdownItem>
{t('per_page')} ))}
</span> </DropdownMenu>
</Dropdown>
<span>{t('per_page')}</span>
</span> </span>
</p>
</div> </div>
) )
} }