Files
api/web/__test__/store/theme.test.ts
Eli Bosley 36c104915e fix: missing translations for expiring trials (#1800)
- Removed translation function calls from the UI components for reboot
type text, replacing them with direct references to the computed
properties.
- Enhanced ineligible update messages by integrating localization for
various conditions, ensuring clearer user feedback regarding update
eligibility.
- Added new localization strings for ineligible update scenarios in the
English locale file.

<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit

* **New Features**
* Added new localization keys for OS update eligibility, reboot labels,
changelog link, and expanded uptime/trial expiry messages.

* **Bug Fixes**
* Restored translated strings and added locale-aware release date
formatting for update/ineligible messaging and badges.

* **Theme & UI**
* Streamlined theme initialization and server-driven theme application;
removed legacy CSS-variable persistence and adjusted dark/banner
behavior.

* **Tests**
* Added i18n and date/locale formatting tests and improved
local-storage-like test mocks.

* **Chores**
* Removed an auto-registered global component and strengthened
script/theme initialization and CSS-variable validation.

<sub>✏️ Tip: You can customize this high-level summary in your review
settings.</sub>
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2025-11-20 19:30:39 -05:00

241 lines
6.8 KiB
TypeScript

/**
* Theme store test coverage
*/
import { createApp, nextTick, ref } from 'vue';
import { setActivePinia } from 'pinia';
import { defaultColors } from '~/themes/default';
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
import { globalPinia } from '~/store/globalPinia';
import { useThemeStore } from '~/store/theme';
vi.mock('@vue/apollo-composable', () => ({
useQuery: () => ({
result: ref(null),
loading: ref(false),
onResult: vi.fn(),
onError: vi.fn(),
}),
}));
describe('Theme Store', () => {
const originalAddClassFn = document.body.classList.add;
const originalRemoveClassFn = document.body.classList.remove;
const originalStyleCssText = document.body.style.cssText;
const originalDocumentElementAddClass = document.documentElement.classList.add;
const originalDocumentElementRemoveClass = document.documentElement.classList.remove;
let store: ReturnType<typeof useThemeStore> | undefined;
let app: ReturnType<typeof createApp> | undefined;
beforeEach(() => {
app = createApp({ render: () => null });
app.use(globalPinia);
setActivePinia(globalPinia);
store = undefined;
window.localStorage.clear();
delete (globalPinia.state.value as Record<string, unknown>).theme;
document.body.classList.add = vi.fn();
document.body.classList.remove = vi.fn();
document.body.style.cssText = '';
document.documentElement.classList.add = vi.fn();
document.documentElement.classList.remove = vi.fn();
vi.spyOn(window, 'requestAnimationFrame').mockImplementation((cb) => {
cb(0);
return 0;
});
vi.clearAllMocks();
});
afterEach(() => {
store?.$dispose();
store = undefined;
app?.unmount();
app = undefined;
document.body.classList.add = originalAddClassFn;
document.body.classList.remove = originalRemoveClassFn;
document.body.style.cssText = originalStyleCssText;
document.documentElement.classList.add = originalDocumentElementAddClass;
document.documentElement.classList.remove = originalDocumentElementRemoveClass;
vi.restoreAllMocks();
});
const createStore = () => {
if (!store) {
store = useThemeStore();
}
return store;
};
describe('State and Initialization', () => {
it('should initialize with default theme', () => {
const store = createStore();
expect(store.theme).toEqual({
name: 'white',
banner: false,
bannerGradient: false,
bgColor: '',
descriptionShow: false,
metaColor: '',
textColor: '',
});
expect(store.activeColorVariables).toEqual(defaultColors.white);
});
it('should compute darkMode correctly', () => {
const store = createStore();
expect(store.darkMode).toBe(false);
store.setTheme({ ...store.theme, name: 'black' });
expect(store.darkMode).toBe(true);
store.setTheme({ ...store.theme, name: 'gray' });
expect(store.darkMode).toBe(true);
store.setTheme({ ...store.theme, name: 'white' });
expect(store.darkMode).toBe(false);
});
it('should compute bannerGradient correctly', () => {
const store = createStore();
expect(store.bannerGradient).toBeUndefined();
store.setTheme({
...store.theme,
banner: true,
bannerGradient: true,
});
expect(store.bannerGradient).toMatchInlineSnapshot(
`"background-image: linear-gradient(90deg, rgba(0, 0, 0, 0) 0, var(--header-background-color) 90%);"`
);
store.setTheme({
...store.theme,
banner: true,
bannerGradient: true,
bgColor: '#123456',
});
expect(store.bannerGradient).toMatchInlineSnapshot(
`"background-image: linear-gradient(90deg, var(--header-gradient-start) 0, var(--header-gradient-end) 90%);"`
);
});
});
describe('Actions', () => {
it('should set theme correctly', () => {
const store = createStore();
const newTheme = {
name: 'black',
banner: true,
bannerGradient: true,
bgColor: '#123456',
descriptionShow: true,
metaColor: '#abcdef',
textColor: '#ffffff',
};
store.setTheme(newTheme);
expect(store.theme).toEqual(newTheme);
});
it('should update body classes for dark mode', async () => {
const store = createStore();
store.setTheme({ ...store.theme, name: 'black' });
await nextTick();
expect(document.body.classList.add).toHaveBeenCalledWith('dark');
store.setTheme({ ...store.theme, name: 'white' });
await nextTick();
expect(document.body.classList.remove).toHaveBeenCalledWith('dark');
});
it('should update activeColorVariables when theme changes', async () => {
const store = createStore();
store.setTheme({
...store.theme,
name: 'white',
textColor: '#333333',
metaColor: '#666666',
bgColor: '#ffffff',
});
await nextTick();
// activeColorVariables now contains the theme defaults from defaultColors
// The white theme's --color-beta is a reference to var(--header-text-primary)
expect(store.activeColorVariables['--color-beta']).toBe('var(--header-text-primary)');
});
it('should apply dark mode classes when theme changes', async () => {
const store = createStore();
store.setTheme({
...store.theme,
name: 'black',
});
await nextTick();
expect(document.documentElement.classList.add).toHaveBeenCalledWith('dark');
expect(document.body.classList.add).toHaveBeenCalledWith('dark');
});
it('should apply dark mode classes to all .unapi elements', async () => {
const store = createStore();
const unapiElement1 = document.createElement('div');
unapiElement1.classList.add('unapi');
document.body.appendChild(unapiElement1);
const unapiElement2 = document.createElement('div');
unapiElement2.classList.add('unapi');
document.body.appendChild(unapiElement2);
const addSpy1 = vi.spyOn(unapiElement1.classList, 'add');
const addSpy2 = vi.spyOn(unapiElement2.classList, 'add');
const removeSpy1 = vi.spyOn(unapiElement1.classList, 'remove');
const removeSpy2 = vi.spyOn(unapiElement2.classList, 'remove');
store.setTheme({
...store.theme,
name: 'black',
});
await nextTick();
expect(addSpy1).toHaveBeenCalledWith('dark');
expect(addSpy2).toHaveBeenCalledWith('dark');
store.setTheme({
...store.theme,
name: 'white',
});
await nextTick();
expect(removeSpy1).toHaveBeenCalledWith('dark');
expect(removeSpy2).toHaveBeenCalledWith('dark');
document.body.removeChild(unapiElement1);
document.body.removeChild(unapiElement2);
});
});
});