diff options
author | John Molakvoæ <skjnldsv@protonmail.com> | 2022-12-08 09:46:59 +0100 |
---|---|---|
committer | John Molakvoæ <skjnldsv@protonmail.com> | 2022-12-08 17:52:50 +0100 |
commit | 1acd42e10ce13d253516efb722f7d05e74fa46e9 (patch) | |
tree | fa04592316f60f454794d761567e25038fc80ce0 | |
parent | fabcd68a5e2c4909dc58d35e756acb0675e61311 (diff) | |
download | nextcloud-server-1acd42e10ce13d253516efb722f7d05e74fa46e9.tar.gz nextcloud-server-1acd42e10ce13d253516efb722f7d05e74fa46e9.zip |
Invert header if primary is bright and background disabled
Signed-off-by: John Molakvoæ <skjnldsv@protonmail.com>
-rw-r--r-- | apps/theming/lib/Controller/UserThemeController.php | 12 | ||||
-rw-r--r-- | apps/theming/lib/Themes/CommonThemeTrait.php | 15 | ||||
-rw-r--r-- | cypress/e2e/theming/admin-settings.cy.ts | 63 | ||||
-rw-r--r-- | cypress/e2e/theming/user-background.cy.ts | 100 | ||||
-rw-r--r-- | cypress/support/commands.ts | 52 |
5 files changed, 196 insertions, 46 deletions
diff --git a/apps/theming/lib/Controller/UserThemeController.php b/apps/theming/lib/Controller/UserThemeController.php index a227e72d998..6a58366c4f6 100644 --- a/apps/theming/lib/Controller/UserThemeController.php +++ b/apps/theming/lib/Controller/UserThemeController.php @@ -50,16 +50,14 @@ use OCP\PreConditionNotMetException; class UserThemeController extends OCSController { - protected string $userId; + protected ?string $userId = null; + private IConfig $config; private IUserSession $userSession; private ThemesService $themesService; private ThemingDefaults $themingDefaults; private BackgroundService $backgroundService; - /** - * Config constructor. - */ public function __construct(string $appName, IRequest $request, IConfig $config, @@ -73,7 +71,11 @@ class UserThemeController extends OCSController { $this->themesService = $themesService; $this->themingDefaults = $themingDefaults; $this->backgroundService = $backgroundService; - $this->userId = $userSession->getUser()->getUID(); + + $user = $userSession->getUser(); + if ($user !== null) { + $this->userId = $user->getUID(); + } } /** diff --git a/apps/theming/lib/Themes/CommonThemeTrait.php b/apps/theming/lib/Themes/CommonThemeTrait.php index 0305a95b3d5..17591c232bb 100644 --- a/apps/theming/lib/Themes/CommonThemeTrait.php +++ b/apps/theming/lib/Themes/CommonThemeTrait.php @@ -24,10 +24,10 @@ declare(strict_types=1); */ namespace OCA\Theming\Themes; -use OCA\Theming\AppInfo\Application; +use OCA\Theming\Util; use OCA\Theming\ImageManager; +use OCA\Theming\AppInfo\Application; use OCA\Theming\Service\BackgroundService; -use OCA\Theming\Util; trait CommonThemeTrait { public Util $util; @@ -82,9 +82,9 @@ trait CommonThemeTrait { * Generate admin theming background-related variables */ protected function generateGlobalBackgroundVariables(): array { - $user = $this->userSession->getUser(); $backgroundDeleted = $this->config->getAppValue(Application::APP_ID, 'backgroundMime', '') === 'backgroundColor'; $hasCustomLogoHeader = $this->util->isLogoThemed(); + $isDefaultPrimaryBright = $this->util->invertTextColor($this->defaultPrimaryColor); $variables = []; @@ -95,8 +95,10 @@ trait CommonThemeTrait { // If primary as background has been request or if we have a custom primary colour // let's not define the background image if ($backgroundDeleted) { - $variables['--color-background-plain'] = $this->themingDefaults->getColorPrimary(); + $variables['--color-background-plain'] = $this->defaultPrimaryColor; $variables['--image-background-plain'] = 'yes'; + // If no background image is set, we need to check against the shown primary colour + $variables['--background-image-invert-if-bright'] = $isDefaultPrimaryBright ? 'invert(100%)' : 'no'; } // Register image variables only if custom-defined @@ -125,12 +127,15 @@ trait CommonThemeTrait { && $this->appManager->isEnabledForUser(Application::APP_ID)) { $backgroundImage = $this->config->getUserValue($user->getUID(), Application::APP_ID, 'background_image', BackgroundService::BACKGROUND_DEFAULT); $currentVersion = (int)$this->config->getUserValue($user->getUID(), Application::APP_ID, 'userCacheBuster', '0'); + $isPrimaryBright = $this->util->invertTextColor($this->primaryColor); // The user removed the background if ($backgroundImage === BackgroundService::BACKGROUND_DISABLED) { return [ '--image-background' => 'no', - '--color-background-plain' => $this->themingDefaults->getColorPrimary(), + '--color-background-plain' => $this->primaryColor, + // If no background image is set, we need to check against the shown primary colour + '--background-image-invert-if-bright' => $isPrimaryBright ? 'invert(100%)' : 'no', ]; } diff --git a/cypress/e2e/theming/admin-settings.cy.ts b/cypress/e2e/theming/admin-settings.cy.ts index 23e7ab9858f..97f3b66c36e 100644 --- a/cypress/e2e/theming/admin-settings.cy.ts +++ b/cypress/e2e/theming/admin-settings.cy.ts @@ -32,7 +32,7 @@ const defaultBackground = 'kamil-porembinski-clouds.jpg' describe('Admin theming settings', function() { before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) @@ -53,7 +53,7 @@ describe('Change the primary colour and reset it', function() { let selectedColor = '' before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) @@ -79,7 +79,7 @@ describe('Change the primary colour and reset it', function() { }) it('Undo theming settings', function() { - cy.resetTheming() + cy.resetAdminTheming() }) it('Screenshot the login page', function() { @@ -92,7 +92,7 @@ describe('Change the primary colour and reset it', function() { describe('Remove the default background and restore it', function() { before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) @@ -122,7 +122,7 @@ describe('Remove the default background and restore it', function() { }) it('Undo theming settings', function() { - cy.resetTheming() + cy.resetAdminTheming() }) it('Screenshot the login page', function() { @@ -132,6 +132,49 @@ describe('Remove the default background and restore it', function() { }) }) +describe.only('Remove the default background with a bright color', function() { + before(function() { + // Just in case previous test failed + cy.resetAdminTheming() + cy.resetUserTheming(admin) + cy.login(admin) + }) + + it('See the admin theming section', function() { + cy.visit('/settings/admin/theming') + cy.get('[data-admin-theming-settings]').scrollIntoView().should('be.visible') + }) + + it('Remove the default background', function() { + cy.intercept('*/apps/theming/ajax/updateStylesheet').as('removeBackground') + + cy.get('[data-admin-theming-setting-file-remove]').click() + + cy.wait('@removeBackground') + }) + + it('Change the primary colour', function() { + cy.intercept('*/apps/theming/ajax/updateStylesheet').as('setColor') + + // Pick one of the bright color preset + cy.get('[data-admin-theming-setting-primary-color-picker]').click() + cy.get('.color-picker__simple-color-circle:eq(4)').click() + + cy.wait('@setColor') + cy.waitUntil(() => validateBodyThemingCss('#ddcb55', '')) + }) + + it('See the header being inverted', function() { + cy.waitUntil(() => cy.window().then((win) => { + const firstEntry = win.document.querySelector('.app-menu-main li') + if (!firstEntry) { + return false + } + return getComputedStyle(firstEntry).filter === 'invert(1)' + })) + }) +}) + describe('Change the login fields then reset them', function() { const name = 'ABCdef123' const url = 'https://example.com' @@ -139,7 +182,7 @@ describe('Change the login fields then reset them', function() { before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) @@ -196,7 +239,7 @@ describe('Change the login fields then reset them', function() { }) it('Undo theming settings', function() { - cy.resetTheming() + cy.resetAdminTheming() }) it('Check login screen changes', function() { @@ -212,7 +255,7 @@ describe('Change the login fields then reset them', function() { describe('Disable user theming and enable it back', function() { before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) @@ -250,12 +293,12 @@ describe('User default option matches admin theming', function() { before(function() { // Just in case previous test failed - cy.resetTheming() + cy.resetAdminTheming() cy.login(admin) }) after(function() { - cy.resetTheming() + cy.resetAdminTheming() }) it('See the admin theming section', function() { diff --git a/cypress/e2e/theming/user-background.cy.ts b/cypress/e2e/theming/user-background.cy.ts index ded0d89efec..f2fde122ce4 100644 --- a/cypress/e2e/theming/user-background.cy.ts +++ b/cypress/e2e/theming/user-background.cy.ts @@ -23,20 +23,8 @@ import type { User } from '@nextcloud/cypress' const defaultPrimary = '#006aa3' const defaultBackground = 'kamil-porembinski-clouds.jpg' -import { colord } from 'colord' -const validateThemingCss = function(expectedPrimary = '#0082c9', expectedBackground = 'kamil-porembinski-clouds.jpg', bright = false) { - return cy.window().then((win) => { - const backgroundColor = getComputedStyle(win.document.body).backgroundColor - const backgroundImage = getComputedStyle(win.document.body).backgroundImage - const invertIfBright = getComputedStyle(win.document.body).getPropertyValue('--background-image-invert-if-bright') - - // Returning boolean for cy.waitUntil usage - return colord(backgroundColor).isEqual(expectedPrimary) - && backgroundImage.includes(expectedBackground) - && invertIfBright === (bright ? 'invert(100%)' : 'no') - }) -} +import { pickRandomColor, validateBodyThemingCss } from './themingUtils' describe('User default background settings', function() { before(function() { @@ -61,7 +49,7 @@ describe('User default background settings', function() { }) }) -describe('User select shipped backgrounds', function() { +describe('User select shipped backgrounds and remove background', function() { before(function() { cy.createRandomUser().then((user: User) => { cy.login(user) @@ -82,7 +70,7 @@ describe('User select shipped backgrounds', function() { // Validate changed background and primary cy.wait('@setBackground') - cy.waitUntil(() => validateThemingCss('#a53c17', background)) + cy.waitUntil(() => validateBodyThemingCss('#a53c17', background)) }) it('Select a bright shipped background', function() { @@ -94,7 +82,7 @@ describe('User select shipped backgrounds', function() { // Validate changed background and primary cy.wait('@setBackground') - cy.waitUntil(() => validateThemingCss('#56633d', background, true)) + cy.waitUntil(() => validateBodyThemingCss('#56633d', background, true)) }) it('Remove background', function() { @@ -105,7 +93,7 @@ describe('User select shipped backgrounds', function() { // Validate clear background cy.wait('@clearBackground') - cy.waitUntil(() => validateThemingCss('#56633d', '')) + cy.waitUntil(() => validateBodyThemingCss('#56633d', '')) }) }) @@ -124,10 +112,9 @@ describe('User select a custom color', function() { it('Select a custom color', function() { cy.intercept('*/apps/theming/background/color').as('setColor') - cy.get('[data-user-theming-background-color]').click() - cy.get('.color-picker__simple-color-circle:eq(3)').click() + pickRandomColor('[data-user-theming-background-color]') - // Validate clear background + // Validate custom colour change cy.wait('@setColor') cy.waitUntil(() => cy.window().then((win) => { const primary = getComputedStyle(win.document.body).getPropertyValue('--color-primary') @@ -136,6 +123,73 @@ describe('User select a custom color', function() { }) }) +describe('User select a bright custom color and remove background', function() { + before(function() { + cy.createRandomUser().then((user: User) => { + cy.login(user) + }) + }) + + it('See the user background settings', function() { + cy.visit('/settings/user/theming') + cy.get('[data-user-theming-background-settings]').scrollIntoView().should('be.visible') + }) + + it('Remove background', function() { + cy.intercept('*/apps/theming/background/custom').as('clearBackground') + + // Clear background + cy.get('[data-user-theming-background-clear]').click() + + // Validate clear background + cy.wait('@clearBackground') + cy.waitUntil(() => validateBodyThemingCss(undefined, '')) + }) + + it('Select a custom color', function() { + cy.intercept('*/apps/theming/background/color').as('setColor') + + // Pick one of the bright color preset + cy.get('[data-user-theming-background-color]').click() + cy.get('.color-picker__simple-color-circle:eq(4)').click() + + // Validate custom colour change + cy.wait('@setColor') + }) + + it('See the header being inverted', function() { + cy.waitUntil(() => cy.window().then((win) => { + const firstEntry = win.document.querySelector('.app-menu-main li') + if (!firstEntry) { + return false + } + return getComputedStyle(firstEntry).filter === 'invert(1)' + })) + }) + + it('Select a shipped background', function() { + const background = 'anatoly-mikhaltsov-butterfly-wing-scale.jpg' + cy.intercept('*/apps/theming/background/shipped').as('setBackground') + + // Select background + cy.get(`[data-user-theming-background-shipped="${background}"]`).click() + + // Validate changed background and primary + cy.wait('@setBackground') + cy.waitUntil(() => validateBodyThemingCss('#a53c17', background)) + }) + + it('See the header NOT being inverted', function() { + cy.waitUntil(() => cy.window().then((win) => { + const firstEntry = win.document.querySelector('.app-menu-main li') + if (!firstEntry) { + return false + } + return getComputedStyle(firstEntry).filter === 'none' + })) + }) +}) + describe('User select a custom background', function() { const image = 'image.jpg' before(function() { @@ -160,7 +214,7 @@ describe('User select a custom background', function() { // Wait for background to be set cy.wait('@setBackground') - cy.waitUntil(() => validateThemingCss('#4c0c04', 'apps/theming/background?v=')) + cy.waitUntil(() => validateBodyThemingCss('#4c0c04', 'apps/theming/background?v=')) }) }) @@ -192,7 +246,7 @@ describe('User changes settings and reload the page', function() { // Wait for background to be set cy.wait('@setBackground') - cy.waitUntil(() => validateThemingCss(primaryFromImage, 'apps/theming/background?v=')) + cy.waitUntil(() => validateBodyThemingCss(primaryFromImage, 'apps/theming/background?v=')) }) it('Select a custom color', function() { @@ -211,6 +265,6 @@ describe('User changes settings and reload the page', function() { it('Reload the page and validate persistent changes', function() { cy.reload() - cy.waitUntil(() => validateThemingCss(selectedColor, 'apps/theming/background?v=')) + cy.waitUntil(() => validateBodyThemingCss(selectedColor, 'apps/theming/background?v=')) }) }) diff --git a/cypress/support/commands.ts b/cypress/support/commands.ts index eab26beda79..ad3f665cade 100644 --- a/cypress/support/commands.ts +++ b/cypress/support/commands.ts @@ -33,8 +33,24 @@ declare global { // eslint-disable-next-line @typescript-eslint/no-namespace namespace Cypress { interface Chainable<Subject = any> { - uploadFile(user: User, fixture?: string, mimeType?: string, target ?: string): Cypress.Chainable<void>, - resetTheming(): Cypress.Chainable<void>, + /** + * Upload a file from the fixtures folder to a given user storage. + * **Warning**: Using this function will reset the previous session + */ + uploadFile(user: User, fixture?: string, mimeType?: string, target?: string): Cypress.Chainable<void>, + + /** + * Reset the admin theming entirely. + * **Warning**: Using this function will reset the previous session + */ + resetAdminTheming(): Cypress.Chainable<void>, + + /** + * Reset the user theming settings. + * If provided, will clear session and login as the given user. + * **Warning**: Providing a user will reset the previous session. + */ + resetUserTheming(user?: User): Cypress.Chainable<void>, } } } @@ -89,7 +105,7 @@ Cypress.Commands.add('uploadFile', (user, fixture = 'image.jpg', mimeType = 'ima /** * Reset the admin theming entirely */ - Cypress.Commands.add('resetTheming', () => { + Cypress.Commands.add('resetAdminTheming', () => { const admin = new User('admin', 'admin') cy.clearCookies() @@ -111,3 +127,33 @@ Cypress.Commands.add('uploadFile', (user, fixture = 'image.jpg', mimeType = 'ima // Clear admin session cy.clearCookies() }) + +/** + * Reset the current or provided user theming settings + * It does not reset the theme config as it is enforced in the + * server config for cypress testing. + */ +Cypress.Commands.add('resetUserTheming', (user?: User) => { + if (user) { + cy.clearCookies() + cy.login(user) + } + + // Reset background config + cy.request('/csrftoken').then(({ body }) => { + const requestToken = body.token + + cy.request({ + method: 'POST', + url: '/apps/theming/background/default', + headers: { + 'requesttoken': requestToken, + }, + }) + }) + + if (user) { + // Clear current session + cy.clearCookies() + } +}) |