Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add the UI toggle for changing theme #4840

Merged
merged 21 commits into from
Sep 5, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
58cb461
Add color mode to ui store
zackkrida Aug 26, 2024
027b0b3
Merge branch 'main' into feat-4308-frontend-color-mode-ui-state
zackkrida Aug 29, 2024
71c6dd2
Merge branch 'main' into feat-4308-frontend-color-mode-ui-state
zackkrida Aug 29, 2024
c378847
Discard changes to frontend/src/components/VFooter/VFooter.vue
zackkrida Aug 29, 2024
6d85e59
Apply suggestions from code review
zackkrida Aug 29, 2024
103c3e4
Fix linting issues
zackkrida Aug 29, 2024
c34ac7a
Add assets for the sun and moon icon
dhruvkb Aug 29, 2024
f93183a
Add strings pertaining to theme switching
dhruvkb Aug 29, 2024
a8ac4da
Add prop to hide the selected choice
dhruvkb Aug 29, 2024
655e59e
Create a theme selector
dhruvkb Aug 29, 2024
9991034
Add theme selector to footer
dhruvkb Aug 29, 2024
17207df
Hook up the theme selector to the UI store functionality
dhruvkb Aug 30, 2024
da176fd
Remove the "force_dark_mode" feature flag as it is redundant
dhruvkb Aug 30, 2024
a3f775f
Avoid snapshot changes
dhruvkb Aug 30, 2024
d04a8ab
Merge branch 'main' of https://github.com/WordPress/openverse into da…
dhruvkb Aug 30, 2024
a4efc3a
Make the theme selector client-only
dhruvkb Aug 30, 2024
147b039
Fix overly large select field in Safari
dhruvkb Aug 30, 2024
e9ea67e
Remove "overflow-hidden" class from `VSelectField`
dhruvkb Aug 31, 2024
4eafee9
Add handling for OS color mode to the composable
dhruvkb Sep 3, 2024
6dc18b5
Use computed props from `useDarkMode`
dhruvkb Sep 3, 2024
735ac71
Fix types
dhruvkb Sep 3, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 0 additions & 9 deletions frontend/feat/feature-flags.json
Original file line number Diff line number Diff line change
Expand Up @@ -31,15 +31,6 @@
"supportsQuery": false,
"storage": "session"
},
"force_dark_mode": {
"status": {
"staging": "switchable",
"production": "disabled"
},
"defaultState": "off",
"description": "Force the site to render in dark mode.",
"storage": "session"
},
"dark_mode_ui_toggle": {
"status": {
"staging": "switchable",
Expand Down
8 changes: 8 additions & 0 deletions frontend/src/assets/svg/raw/moon.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
13 changes: 13 additions & 0 deletions frontend/src/assets/svg/raw/sun.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
2 changes: 2 additions & 0 deletions frontend/src/assets/svg/sprite/icons.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
31 changes: 26 additions & 5 deletions frontend/src/components/VFooter/VFooter.vue
Original file line number Diff line number Diff line change
Expand Up @@ -11,11 +11,13 @@ import useResizeObserver from "~/composables/use-resize-observer"
import { SCREEN_SIZES } from "~/constants/screens"

import { useUiStore } from "~/stores/ui"
import { useFeatureFlagStore } from "~/stores/feature-flag"

import type { SelectFieldProps } from "~/components/VSelectField/VSelectField.vue"
import VLink from "~/components/VLink.vue"
import VBrand from "~/components/VBrand/VBrand.vue"
import VLanguageSelect from "~/components/VLanguageSelect/VLanguageSelect.vue"
import VThemeSelect from "~/components/VThemeSelect/VThemeSelect.vue"
import VPageLinks from "~/components/VHeader/VPageLinks.vue"
import VWordPressLink from "~/components/VHeader/VWordPressLink.vue"

Expand All @@ -42,6 +44,11 @@ const { all: allPages } = usePages()

const isContentMode = computed(() => props.mode === "content")

const featureFlagStore = useFeatureFlagStore()
const showThemeSwitcher = computed(() =>
featureFlagStore.isOn("dark_mode_ui_toggle")
)

/** JS-based responsiveness */
const footerEl = ref<HTMLElement | null>(null)
const initialWidth = SCREEN_SIZES[uiStore.breakpoint]
Expand Down Expand Up @@ -90,11 +97,25 @@ const linkColumnHeight = computed(() => ({

<!-- Locale chooser and WordPress affiliation graphic -->
<div class="locale-and-wp flex flex-col justify-between">
<VLanguageSelect
v-bind="languageProps"
class="language max-w-full border-secondary"
/>
<VWordPressLink mode="light" />
<template v-if="showThemeSwitcher">
<VWordPressLink mode="light" />
<div class="flex flex-row items-center gap-6">
<VLanguageSelect
v-bind="languageProps"
class="language max-w-full border-secondary"
/>
<ClientOnly>
<VThemeSelect class="border-secondary" />
</ClientOnly>
</div>
</template>
<template v-else>
<VLanguageSelect
v-bind="languageProps"
class="language max-w-full border-secondary"
/>
<VWordPressLink mode="light" />
</template>
</div>
</footer>
</template>
Expand Down
11 changes: 8 additions & 3 deletions frontend/src/components/VSelectField/VSelectField.vue
Original file line number Diff line number Diff line change
Expand Up @@ -33,10 +33,12 @@ const props = withDefaults(
fieldId: string
labelText: string
choices: Choice[]
showSelected?: boolean
}>(),
{
modelValue: "",
blankText: "",
showSelected: true,
}
)

Expand Down Expand Up @@ -70,7 +72,7 @@ const splitAttrs = computed(() => {

<template>
<div
class="relative m-0.5px box-content block w-fit overflow-hidden rounded-sm border border-default text-sm focus-within:m-0 focus-within:border-1.5 focus-within:border-focus hover:border-hover focus-within:hover:border-focus"
class="relative m-0.5px box-content block w-fit rounded-sm border border-default text-sm focus-within:m-0 focus-within:border-1.5 focus-within:border-focus hover:border-hover focus-within:hover:border-focus"
:class="splitAttrs.classAttrs"
>
<div class="pointer-events-none absolute inset-y-0 start-2 my-auto h-fit">
Expand All @@ -82,8 +84,11 @@ const splitAttrs = computed(() => {
<select
:id="fieldId"
v-model="selectValue"
class="flex h-[calc(theme(spacing.10)_-_2_*_theme(borderWidth.DEFAULT))] w-full appearance-none truncate bg-tx pe-10"
:class="hasStartContent ? 'ps-10' : 'ps-2'"
class="flex h-[calc(theme(spacing.10)_-_2_*_theme(borderWidth.DEFAULT))] appearance-none truncate bg-tx pe-10"
:class="[
showSelected ? 'w-full' : 'w-0 max-w-0',
hasStartContent ? 'ps-10' : 'ps-2',
]"
:name="fieldName"
v-bind="splitAttrs.nonClassAttrs"
:aria-label="labelText"
Expand Down
100 changes: 100 additions & 0 deletions frontend/src/components/VThemeSelect/VThemeSelect.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
<!--
Present users with a way to change the app theme between three options:
light, dark and system.
-->

<script setup lang="ts">
import { useI18n } from "#imports"

import { computed, type ComputedRef } from "vue"
import { usePreferredColorScheme } from "@vueuse/core"

import { useUiStore } from "~/stores/ui"

import VIcon from "~/components/VIcon/VIcon.vue"
import VSelectField, {
type Choice,
} from "~/components/VSelectField/VSelectField.vue"

/**
* `ActualColorMode` is the evaluated form of `ColorMode`.
*
* It's value is the same as `ColorMode` for light and dark color modes
* but for the system color mode, it evaluates to either depending on
* the user's system theme.
*/
type ActualColorMode = "light" | "dark"

const i18n = useI18n({ useScope: "global" })
const uiStore = useUiStore()

const THEME_ICON_NAME = Object.freeze({
light: "sun",
dark: "moon",
})

const THEME_TEXT = {
light: i18n.t(`theme.choices.light`),
dark: i18n.t(`theme.choices.dark`),
}

const colorMode = computed({
get: () => uiStore.colorMode,
set: (value) => {
uiStore.setColorMode(value)
},
})

/**
* Get the user's preferred color scheme at the OS level. If the user
* has not set an OS level preference, we fall back to light mode.
*/
const preferredColorScheme: ComputedRef<ActualColorMode> = computed(() => {
const pref = usePreferredColorScheme()
return pref.value === "no-preference" ? "light" : pref.value
})

/**
* Get the user's preferred color scheme at the app level. If the user
* has set the color mode to "system", we fall back to the OS level
* preference.
*/
const actualColorMode: ComputedRef<ActualColorMode> = computed(() =>
colorMode.value === "system" ? preferredColorScheme.value : colorMode.value
)

/**
* The icon always reflects the actual theme applied to the site.
* Therefore, it must be based on the value of `actualColorMode`.
*/
const currentThemeIcon = computed(() => THEME_ICON_NAME[actualColorMode.value])

/**
* The choices are computed because the text for the color mode choice
* "system" is dynamic and reflects the user's preferred color scheme at
* the OS-level.
*/
const choices: ComputedRef<Choice[]> = computed(() => {
const systemText = `${i18n.t(`theme.choices.system`)} (${THEME_TEXT[preferredColorScheme.value]})`
return [
{ key: "light", text: THEME_TEXT.light },
{ key: "dark", text: THEME_TEXT.dark },
{ key: "system", text: systemText },
]
})
</script>

<template>
<VSelectField
v-model="colorMode"
field-id="theme"
:choices="choices"
:blank-text="$t('theme.theme')"
:label-text="$t('theme.theme')"
:show-selected="false"
>
<template #start>
<VIcon :name="currentThemeIcon" />
</template>
</VSelectField>
</template>
8 changes: 3 additions & 5 deletions frontend/src/composables/use-dark-mode.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,7 @@ export const LIGHT_MODE_CLASS = "light-mode"
/**
* Determines the dark mode setting based on user preference or feature flag.
*
* When dark mode toggling is disabled, the site is in "light mode" unless
* the `force_dark_mode` feature flag is on.
* When dark mode toggling is disabled, the site is in "light mode".
*
* When the "dark_mode_ui_toggle" flag is enabled, the site will respect
* the user system preference by default.
Expand All @@ -22,13 +21,12 @@ export function useDarkMode() {
const darkModeToggleable = computed(() =>
featureFlagStore.isOn("dark_mode_ui_toggle")
)
const forceDarkMode = computed(() => featureFlagStore.isOn("force_dark_mode"))

const colorMode = computed(() => {
if (darkModeToggleable.value && !forceDarkMode.value) {
if (darkModeToggleable.value) {
return uiStore.colorMode
}
return forceDarkMode.value ? "dark" : "light"
return "light"
})

const cssClass = computed(() => {
Expand Down
8 changes: 8 additions & 0 deletions frontend/src/locales/scripts/en.json5
Original file line number Diff line number Diff line change
Expand Up @@ -1085,6 +1085,14 @@
language: {
language: "Language",
},
theme: {
theme: "Theme",
choices: {
dark: "Dark",
light: "Light",
system: "System",
},
},
recentSearches: {
heading: "Recent searches",
clear: {
Expand Down
17 changes: 5 additions & 12 deletions frontend/test/unit/specs/composables/use-dark-mode.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,23 +11,16 @@ import { useUiStore } from "~/stores/ui"

describe("useDarkMode", () => {
test.each`
description | featureFlags | uiColorMode | expectedColorMode | expectedCssClass
${"Force dark mode and disable toggling"} | ${{ force_dark_mode: ON, dark_mode_ui_toggle: OFF }} | ${"light"} | ${"dark"} | ${DARK_MODE_CLASS}
${"Don't force dark mode and disable toggling"} | ${{ force_dark_mode: OFF, dark_mode_ui_toggle: OFF }} | ${"dark"} | ${"light"} | ${LIGHT_MODE_CLASS}
${"Force dark mode and enable toggling"} | ${{ force_dark_mode: ON, dark_mode_ui_toggle: ON }} | ${"light"} | ${"dark"} | ${DARK_MODE_CLASS}
${"Enable toggling, User preference: light"} | ${{ force_dark_mode: OFF, dark_mode_ui_toggle: ON }} | ${"light"} | ${"light"} | ${LIGHT_MODE_CLASS}
${"Enable toggling, User preference: dark"} | ${{ force_dark_mode: OFF, dark_mode_ui_toggle: ON }} | ${"dark"} | ${"dark"} | ${DARK_MODE_CLASS}
${"Enable toggling, User preference: system"} | ${{ force_dark_mode: OFF, dark_mode_ui_toggle: ON }} | ${"system"} | ${"system"} | ${""}
description | featureFlags | uiColorMode | expectedColorMode | expectedCssClass
${"Disable toggling"} | ${{ dark_mode_ui_toggle: OFF }} | ${"dark"} | ${"light"} | ${LIGHT_MODE_CLASS}
${"Enable toggling, User preference: light"} | ${{ dark_mode_ui_toggle: ON }} | ${"light"} | ${"light"} | ${LIGHT_MODE_CLASS}
${"Enable toggling, User preference: dark"} | ${{ dark_mode_ui_toggle: ON }} | ${"dark"} | ${"dark"} | ${DARK_MODE_CLASS}
${"Enable toggling, User preference: system"} | ${{ dark_mode_ui_toggle: ON }} | ${"system"} | ${"system"} | ${""}
`(
"$description: should report colorMode as $expectedColorMode and cssClass as $expectedCssClass",
({ featureFlags, uiColorMode, expectedColorMode, expectedCssClass }) => {
const featureFlagStore = useFeatureFlagStore()

// Set the feature flags
featureFlagStore.toggleFeature(
"force_dark_mode",
featureFlags.force_dark_mode
)
featureFlagStore.toggleFeature(
"dark_mode_ui_toggle",
featureFlags.dark_mode_ui_toggle
Expand Down