diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-dark.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-dark.png new file mode 100755 index 0000000000..321346fee4 Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-dark.png differ diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-light.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-light.png new file mode 100755 index 0000000000..6ba5984f34 Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-inherit-light.png differ diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-components-within-ic-theme.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-components-within-ic-theme.png new file mode 100755 index 0000000000..846d35d0a5 Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-components-within-ic-theme.png differ diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-dark-mode-component-overrides.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-dark-mode-component-overrides.png new file mode 100755 index 0000000000..18018781e0 Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-light-with-dark-mode-component-overrides.png differ diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-system-light.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-system-light.png new file mode 100755 index 0000000000..18018781e0 Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-ic-theme-system-light.png differ diff --git a/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-sunrise-brand.png b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-sunrise-brand.png new file mode 100755 index 0000000000..76481137ac Binary files /dev/null and b/packages/react/cypress-image-diff-screenshots/baseline/IcTheme.cy.tsx-sunrise-brand.png differ diff --git a/packages/react/src/component-tests/IcTheme/IcTheme.cy.tsx b/packages/react/src/component-tests/IcTheme/IcTheme.cy.tsx new file mode 100644 index 0000000000..14ea952a07 --- /dev/null +++ b/packages/react/src/component-tests/IcTheme/IcTheme.cy.tsx @@ -0,0 +1,199 @@ +/* eslint-disable react/jsx-no-bind */ +/// + +import { mount } from "cypress/react"; +import React from "react"; +import { + ComponentsWithInheritThemeProp, + SwitchBrand, + ThemeWithComponentThemeProps, + ThemeWithComponentWithinSeparateIcTheme, +} from "./IcThemeTestData"; +import { + HAVE_ATTR, + HAVE_BEEN_CALLED_ONCE, + HAVE_BEEN_CALLED_WITH, +} from "../utils/constants"; +import { setThresholdBasedOnEnv } from "../../../cypress/utils/helpers"; + +const THEME_SELECTOR = "ic-theme"; +const DEFAULT_TEST_THRESHOLD = 0.048; + +describe("IcTheme end-to-end and visual regression tests", () => { + beforeEach(() => { + cy.injectAxe(); + cy.viewport(1440, 900); + }); + + afterEach(() => { + cy.task("generateReport"); + }); + + it("should render in light mode when IcTheme theme is set to light with components with theme prop set to dark rendered in dark mode", () => { + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-light-with-dark-mode-component-overrides", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.031), + }); + }); + + it("should render in light mode when IcTheme theme is set to light with components within another IcTheme component with theme set to dark rendered in dark mode", () => { + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-light-with-components-within-ic-theme", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.03), + }); + }); + + it.skip("should render according to dark mode system settings when IcTheme theme is set to system", () => { + cy.wrap( + Cypress.automation("remote:debugger:protocol", { + command: "Emulation.setEmulatedMedia", + params: { + features: [ + { + name: "prefers-color-scheme", + value: "dark", + }, + ], + }, + }) + ); + + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-system-dark", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.03), + }); + }); + + it("should render according to light mode system settings when IcTheme theme is set to system", () => { + cy.wrap( + Cypress.automation("remote:debugger:protocol", { + command: "Emulation.setEmulatedMedia", + params: { + features: [ + { + name: "prefers-color-scheme", + value: "light", + }, + ], + }, + }) + ); + + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-system-light", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.031), + }); + }); + + it("should render components with IcTheme colour when theme prop is set to inherit - dark theme", () => { + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-inherit-dark", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD), + }); + }); + + it("should render components with IcTheme colour when theme prop is set to inherit - light theme", () => { + mount( + + ); + + cy.checkHydrated(THEME_SELECTOR); + + cy.checkA11yWithWait(); + cy.compareSnapshot({ + name: "ic-theme-inherit-light", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.004), + }); + }); + + it("should switch brand colour", () => { + mount(); + + cy.checkHydrated(THEME_SELECTOR); + cy.spy(window.console, "log").as("spyWinConsoleLog"); + cy.get(THEME_SELECTOR).invoke( + "on", + "brandChange", + cy.stub().as("brandChange") + ); + + cy.get(THEME_SELECTOR).should(HAVE_ATTR, "brand-color", "rgb(255, 201, 60)"); + + cy.compareSnapshot({ + name: "sunrise-brand", + testThreshold: setThresholdBasedOnEnv(DEFAULT_TEST_THRESHOLD + 0.022), + }); + + cy.get("ic-button").contains("Default theme").click(); + cy.get(THEME_SELECTOR).should(HAVE_ATTR, "brand-color", "rgb(27, 60, 121)"); + cy.get("@brandChange").should(HAVE_BEEN_CALLED_ONCE); + cy.get("@spyWinConsoleLog").should(HAVE_BEEN_CALLED_WITH, { + mode: "light", + color: { + r: 27, + g: 60, + b: 121, + a: 1, + }, + }); + }); +}); diff --git a/packages/react/src/component-tests/IcTheme/IcThemeTestData.tsx b/packages/react/src/component-tests/IcTheme/IcThemeTestData.tsx new file mode 100644 index 0000000000..2b3f508897 --- /dev/null +++ b/packages/react/src/component-tests/IcTheme/IcThemeTestData.tsx @@ -0,0 +1,595 @@ +import React, { ReactElement, useState } from "react"; +import { + IcTheme, + IcTopNavigation, + IcSearchBar, + IcNavigationButton, + IcNavigationItem, + IcHero, + IcTextField, + IcButton, + IcLink, + IcSectionContainer, + IcCheckbox, + IcAlert, + IcTypography, + IcCardVertical, + IcFooter, + IcFooterLink, + IcSwitch, +} from "../../components"; +import { SlottedSVG } from "../../react-component-lib/slottedSVG"; + +export const ThemeWithComponentThemeProps = (args: { + color: "rgba(27, 60, 121, 1)"; + theme: "light" | "dark" | "system"; + backgroundColor: "white" | "black"; +}): ReactElement => { + return ( + <> +
+ + + + + + + + alert("test")} + > + + + + + + + + + + + + +
+ + + Filter + +
+ + See all + + + Help + +
+ + + + + + Button + + +

+ This is some text and + this is an inline link IcLink + within the text +

+
+
+ +
+ + + + + + The component theme prop should override the IcTheme theme prop + +
+
+ + + Get Started + + + Accessibility + + + Styles + + + Components + + + Patterns + + + Design toolkit + +
+ Slotted Button +
+
+
+
+ + ); +}; + +export const ThemeWithComponentWithinSeparateIcTheme = (args: { + color: "rgba(27, 60, 121, 1)"; + theme: "light" | "dark" | "system"; + backgroundColor: "white" | "black"; +}): ReactElement => { + return ( + <> +
+ + + + + + + + alert("test")} + > + + + + + + + + + + + + +
+ + + Filter + +
+ + See all + + + Help + +
+ + + + + + Button + + +

+ This is some text and + this is an inline link IcLink + within the text +

+
+
+ +
+ + This content is in an IcTheme component with dark mode set + + + + + +
+
+ + + Get Started + + + Accessibility + + + Styles + + + Components + + + Patterns + + + Design toolkit + +
+ Slotted Button +
+
+
+
+ + ); +}; + +export const ComponentsWithInheritThemeProp = (args: { + color: "rgba(27, 60, 121, 1)"; + theme: "light" | "dark" | "system"; + backgroundColor: "white" | "black"; +}): ReactElement => { + return ( + <> +
+ + + + + + + + alert("test")} + > + + + + + + + + + + + + + + +
+ + + + + + Components will inherit their theme colour based on IcTheme when + theme="inherit" + +
+
+
+
+ + ); +}; + +export const SwitchBrand = () => { + const [color, setColour] = useState("rgb(255, 201, 60)"); + const defaultButtonClickHandler = () => { + setColour("rgb(27, 60, 121)"); + }; + const differentButtonClickHandler = () => { + setColour("rgb(255, 201, 60)"); + }; + const handleBrandChange = (e: CustomEvent) => { + console.log(e.detail); + }; + return ( + <> + + + Default theme + + + Sunset theme + +
+ + + + + + + + + + + + + + + + + + + +
+ + + Filter + +
+ + See all + + + Help + +
+ + + Get Started + + + Accessibility + + + Styles + + + Components + + + Patterns + + + Design toolkit + +
+ Slotted Button + + + + + + +
+
+
+ + ); +}; diff --git a/packages/react/src/stories/ic-theme.stories.mdx b/packages/react/src/stories/ic-theme.stories.mdx index dbdf539117..4a0274010e 100644 --- a/packages/react/src/stories/ic-theme.stories.mdx +++ b/packages/react/src/stories/ic-theme.stories.mdx @@ -105,11 +105,11 @@ export const Controlled = () => { full-width >
- + > Filter @@ -117,13 +117,13 @@ export const Controlled = () => { See all - Help - + {

This is some text and - this is an inline link ic-link + this is an inline link ic-link within the text

{ style={{ display: "flex", alignItems: "center", - gap: "var(--ic-space-sm);", + gap: "var(--ic-space-sm)", }} > Slotted Button @@ -269,11 +269,11 @@ export const defaultArgs = { full-width >
- + > Filter @@ -281,13 +281,13 @@ export const defaultArgs = { See all - Help - +

This is some text and - this is an inline link ic-link + this is an inline link ic-link within the text

@@ -361,7 +361,7 @@ export const defaultArgs = { style={{ display: "flex", alignItems: "center", - gap: "var(--ic-space-sm);", + gap: "var(--ic-space-sm)", }} > Slotted Button