Skip to content

Commit 44c7320

Browse files
authoredJan 14, 2025··
feat(material/schematics): Add option to customize colors for neutral variant and error palettes (#30321)
1 parent 5f238ab commit 44c7320

File tree

4 files changed

+268
-16
lines changed

4 files changed

+268
-16
lines changed
 

‎src/material/schematics/ng-generate/theme-color/README.md

+6-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@ optimized to have enough contrast to be more accessible. See [Science of Color D
1313
for more information about Material's color design.
1414

1515
For more customization, custom colors can be also be provided for the
16-
secondary, tertiary, and neutral palette colors. It is recommended to choose colors that
17-
are contrastful. Material has more detailed guidance for [accessible design](https://m3.material.io/foundations/accessible-design/patterns).
16+
secondary, tertiary, neutral, neutral variant, and error palette colors. It is recommended to choose
17+
colors that are contrastful. Material has more detailed guidance for [accessible design](https://m3.material.io/foundations/accessible-design/patterns).
1818

1919
## Options
2020

@@ -30,6 +30,10 @@ secondary color generated from Material based on the primary.
3030
tertiary color generated from Material based on the primary.
3131
* `neutralColor` - Color to use for app's neutral color palette. Defaults to
3232
neutral color generated from Material based on the primary.
33+
* `neutralVariantColor` - Color to use for app's neutral variant color palette. Defaults to
34+
neutral variant color generated from Material based on the primary.
35+
* `errorColor` - Color to use for app's error color palette. Defaults to
36+
error color generated from Material based on the other palettes.
3337
* `includeHighContrast` - Whether to define high contrast values for the custom colors in the
3438
generated file. For Sass files a mixin is defined, see the [high contrast override mixins section](#high-contrast-override-mixins)
3539
for more information. Defaults to false.

‎src/material/schematics/ng-generate/theme-color/index.spec.ts

+202
Original file line numberDiff line numberDiff line change
@@ -201,6 +201,64 @@ describe('material-theme-color-schematic', () => {
201201
expect(transpileTheme(generatedSCSS)).toBe(transpileTheme(testSCSS));
202202
});
203203

204+
it('should generate themes when provided a primary, secondary, tertiary, neutral, and neutral variant colors', async () => {
205+
const tree = await runM3ThemeSchematic(runner, {
206+
primaryColor: '#984061',
207+
secondaryColor: '#984061',
208+
tertiaryColor: '#984061',
209+
neutralColor: '#984061',
210+
neutralVariantColor: '#984061',
211+
});
212+
213+
const generatedSCSS = tree.readText('_theme-colors.scss');
214+
215+
// Change test theme palette so that secondary, tertiary, and neutral are
216+
// the same source color as primary to match schematic inputs
217+
let testPalettes = testM3ColorPalettes;
218+
testPalettes.secondary = testPalettes.primary;
219+
testPalettes.tertiary = testPalettes.primary;
220+
testPalettes.neutral = testPalettes.primary;
221+
testPalettes.neutralVariant = testPalettes.primary;
222+
223+
const testSCSS = generateSCSSTheme(
224+
testPalettes,
225+
'Color palettes are generated from primary: #984061, secondary: #984061, tertiary: #984061, neutral: #984061, neutral variant: #984061',
226+
);
227+
228+
expect(generatedSCSS).toBe(testSCSS);
229+
expect(transpileTheme(generatedSCSS)).toBe(transpileTheme(testSCSS));
230+
});
231+
232+
it('should generate themes when provided a primary, secondary, tertiary, neutral, neutral variant, and error colors', async () => {
233+
const tree = await runM3ThemeSchematic(runner, {
234+
primaryColor: '#984061',
235+
secondaryColor: '#984061',
236+
tertiaryColor: '#984061',
237+
neutralColor: '#984061',
238+
neutralVariantColor: '#984061',
239+
errorColor: '#984061',
240+
});
241+
242+
const generatedSCSS = tree.readText('_theme-colors.scss');
243+
244+
// Change test theme palette so that secondary, tertiary, and neutral are
245+
// the same source color as primary to match schematic inputs
246+
let testPalettes = testM3ColorPalettes;
247+
testPalettes.secondary = testPalettes.primary;
248+
testPalettes.tertiary = testPalettes.primary;
249+
testPalettes.neutral = testPalettes.primary;
250+
testPalettes.neutralVariant = testPalettes.primary;
251+
testPalettes.error = testPalettes.primary;
252+
253+
const testSCSS = generateSCSSTheme(
254+
testPalettes,
255+
'Color palettes are generated from primary: #984061, secondary: #984061, tertiary: #984061, neutral: #984061, neutral variant: #984061, error: #984061',
256+
);
257+
258+
expect(generatedSCSS).toBe(testSCSS);
259+
expect(transpileTheme(generatedSCSS)).toBe(transpileTheme(testSCSS));
260+
});
261+
204262
describe('and with high contrast overrides', () => {
205263
it('should be able to generate high contrast overrides mixin', async () => {
206264
const tree = await runM3ThemeSchematic(runner, {
@@ -300,6 +358,63 @@ describe('material-theme-color-schematic', () => {
300358
expect(generatedCSS).toContain(`--mat-sys-tertiary: #ffebef`);
301359
expect(generatedCSS).toContain(`--mat-sys-surface-bright: #4f5051`);
302360
});
361+
362+
it('should be able to generate high contrast themes overrides when provided primary, secondary, tertiary, neutral, and neutral variant color', async () => {
363+
const tree = await runM3ThemeSchematic(runner, {
364+
primaryColor: '#984061',
365+
secondaryColor: '#984061',
366+
tertiaryColor: '#984061',
367+
neutralColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
368+
neutralVariantColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
369+
includeHighContrast: true,
370+
});
371+
372+
const generatedCSS = transpileTheme(tree.readText('_theme-colors.scss'));
373+
374+
// Check a system variable from each color palette for their high contrast light theme value
375+
expect(generatedCSS).toContain(`--mat-sys-primary: #580b2f`);
376+
expect(generatedCSS).toContain(`--mat-sys-secondary: #580b2f`);
377+
expect(generatedCSS).toContain(`--mat-sys-tertiary: #580b2f`);
378+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: #f9f9f9`);
379+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: #e2e2e2`);
380+
381+
// Check a system variable from each color palette for their high contrast dark theme value
382+
expect(generatedCSS).toContain(`--mat-sys-primary: #ffebef`);
383+
expect(generatedCSS).toContain(`--mat-sys-secondary: #ffebef`);
384+
expect(generatedCSS).toContain(`--mat-sys-tertiary: #ffebef`);
385+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: #4f5051`);
386+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: #454747`);
387+
});
388+
389+
it('should be able to generate high contrast themes overrides when provided primary, secondary, tertiary, neutral, neutral variant, and error color', async () => {
390+
const tree = await runM3ThemeSchematic(runner, {
391+
primaryColor: '#984061',
392+
secondaryColor: '#984061',
393+
tertiaryColor: '#984061',
394+
neutralColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
395+
neutralVariantColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
396+
errorColor: '#984061',
397+
includeHighContrast: true,
398+
});
399+
400+
const generatedCSS = transpileTheme(tree.readText('_theme-colors.scss'));
401+
402+
// Check a system variable from each color palette for their high contrast light theme value
403+
expect(generatedCSS).toContain(`--mat-sys-primary: #580b2f`);
404+
expect(generatedCSS).toContain(`--mat-sys-secondary: #580b2f`);
405+
expect(generatedCSS).toContain(`--mat-sys-tertiary: #580b2f`);
406+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: #f9f9f9`);
407+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: #e2e2e2`);
408+
expect(generatedCSS).toContain(`--mat-sys-error: #580b2f`);
409+
410+
// Check a system variable from each color palette for their high contrast dark theme value
411+
expect(generatedCSS).toContain(`--mat-sys-primary: #ffebef`);
412+
expect(generatedCSS).toContain(`--mat-sys-secondary: #ffebef`);
413+
expect(generatedCSS).toContain(`--mat-sys-tertiary: #ffebef`);
414+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: #4f5051`);
415+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: #454747`);
416+
expect(generatedCSS).toContain(`--mat-sys-error: #ffebef`);
417+
});
303418
});
304419
});
305420

@@ -405,6 +520,49 @@ describe('material-theme-color-schematic', () => {
405520
expect(generatedCSS).toContain(`--mat-sys-surface-variant: light-dark(#f6dce2, #534247)`);
406521
});
407522

523+
it('should generate CSS system variables when provided a primary, secondary, tertiary, neutral, and neutral variant colors', async () => {
524+
const tree = await runM3ThemeSchematic(runner, {
525+
primaryColor: '#984061',
526+
secondaryColor: '#984061',
527+
tertiaryColor: '#984061',
528+
neutralColor: '#984061',
529+
neutralVariantColor: '#984061',
530+
isScss: false,
531+
});
532+
533+
const generatedCSS = tree.readText('theme.css');
534+
535+
// Check a system variable from each color palette for their light dark value
536+
expect(generatedCSS).toContain(`--mat-sys-primary: light-dark(#984061, #ffb0c8)`);
537+
expect(generatedCSS).toContain(`--mat-sys-secondary: light-dark(#984061, #ffb0c8)`);
538+
expect(generatedCSS).toContain(`--mat-sys-tertiary: light-dark(#984061, #ffb0c8)`);
539+
expect(generatedCSS).toContain(`--mat-sys-error: light-dark(#ba1a1a, #ffb4ab)`);
540+
expect(generatedCSS).toContain(`--mat-sys-surface: light-dark(#fff8f8, #2f0015);`);
541+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: light-dark(#ffd9e2, #7b2949)`);
542+
});
543+
544+
it('should generate CSS system variables when provided a primary, secondary, tertiary, neutral, neutral variant, and error colors', async () => {
545+
const tree = await runM3ThemeSchematic(runner, {
546+
primaryColor: '#984061',
547+
secondaryColor: '#984061',
548+
tertiaryColor: '#984061',
549+
neutralColor: '#984061',
550+
neutralVariantColor: '#984061',
551+
errorColor: '#984061',
552+
isScss: false,
553+
});
554+
555+
const generatedCSS = tree.readText('theme.css');
556+
557+
// Check a system variable from each color palette for their light dark value
558+
expect(generatedCSS).toContain(`--mat-sys-primary: light-dark(#984061, #ffb0c8)`);
559+
expect(generatedCSS).toContain(`--mat-sys-secondary: light-dark(#984061, #ffb0c8)`);
560+
expect(generatedCSS).toContain(`--mat-sys-tertiary: light-dark(#984061, #ffb0c8)`);
561+
expect(generatedCSS).toContain(`--mat-sys-error: light-dark(#984061, #ffb0c8)`);
562+
expect(generatedCSS).toContain(`--mat-sys-surface: light-dark(#fff8f8, #2f0015);`);
563+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: light-dark(#ffd9e2, #7b2949)`);
564+
});
565+
408566
describe('and with high contrast overrides', () => {
409567
it('should generate high contrast system variables', async () => {
410568
const tree = await runM3ThemeSchematic(runner, {
@@ -485,6 +643,50 @@ describe('material-theme-color-schematic', () => {
485643
expect(generatedCSS).toContain(`--mat-sys-tertiary: light-dark(#580b2f, #ffebef)`);
486644
expect(generatedCSS).toContain(`--mat-sys-surface-bright: light-dark(#f9f9f9, #4f5051)`);
487645
});
646+
647+
it('should generate high contrast system variables when provided primary, secondary, tertiary, neutral, and neutral variant color', async () => {
648+
const tree = await runM3ThemeSchematic(runner, {
649+
primaryColor: '#984061',
650+
secondaryColor: '#984061',
651+
tertiaryColor: '#984061',
652+
neutralColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
653+
neutralVariantColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
654+
isScss: false,
655+
includeHighContrast: true,
656+
});
657+
658+
const generatedCSS = tree.readText('theme.css');
659+
660+
// Check a system variable from each color palette for their high contrast light dark value
661+
expect(generatedCSS).toContain(`--mat-sys-primary: light-dark(#580b2f, #ffebef)`);
662+
expect(generatedCSS).toContain(`--mat-sys-secondary: light-dark(#580b2f, #ffebef)`);
663+
expect(generatedCSS).toContain(`--mat-sys-tertiary: light-dark(#580b2f, #ffebef)`);
664+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: light-dark(#f9f9f9, #4f5051)`);
665+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: light-dark(#e2e2e2, #454747);`);
666+
});
667+
668+
it('should generate high contrast system variables when provided primary, secondary, tertiary, neutral, neutral variant, and error color', async () => {
669+
const tree = await runM3ThemeSchematic(runner, {
670+
primaryColor: '#984061',
671+
secondaryColor: '#984061',
672+
tertiaryColor: '#984061',
673+
neutralColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
674+
neutralVariantColor: '#dfdfdf', // Different color since #984061 does not change the tonal palette
675+
errorColor: '#984061',
676+
isScss: false,
677+
includeHighContrast: true,
678+
});
679+
680+
const generatedCSS = tree.readText('theme.css');
681+
682+
// Check a system variable from each color palette for their high contrast light dark value
683+
expect(generatedCSS).toContain(`--mat-sys-primary: light-dark(#580b2f, #ffebef)`);
684+
expect(generatedCSS).toContain(`--mat-sys-secondary: light-dark(#580b2f, #ffebef)`);
685+
expect(generatedCSS).toContain(`--mat-sys-tertiary: light-dark(#580b2f, #ffebef)`);
686+
expect(generatedCSS).toContain(`--mat-sys-surface-bright: light-dark(#f9f9f9, #4f5051)`);
687+
expect(generatedCSS).toContain(`--mat-sys-surface-variant: light-dark(#e2e2e2, #454747);`);
688+
expect(generatedCSS).toContain(`--mat-sys-error: light-dark(#580b2f, #ffebef)`);
689+
});
488690
});
489691
});
490692
});

‎src/material/schematics/ng-generate/theme-color/index.ts

+52-14
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,8 @@ export function getColorPalettes(
116116
secondaryColor?: string,
117117
tertiaryColor?: string,
118118
neutralColor?: string,
119+
neutralVariantColor?: string,
120+
errorColor?: string,
119121
): ColorPalettes {
120122
// Create tonal palettes for each color and custom color overrides if applicable. Used for both
121123
// standard contrast and high contrast schemes since they share the same tonal palettes.
@@ -157,21 +159,31 @@ export function getColorPalettes(
157159
);
158160
}
159161

160-
const neutralVariantPalette = TonalPalette.fromHueAndChroma(
161-
primaryColorHct.hue,
162-
primaryColorHct.chroma / 8.0 + 4.0,
163-
);
162+
let neutralVariantPalette;
163+
if (neutralVariantColor) {
164+
neutralVariantPalette = TonalPalette.fromHct(getHctFromHex(neutralVariantColor));
165+
} else {
166+
neutralVariantPalette = TonalPalette.fromHueAndChroma(
167+
primaryColorHct.hue,
168+
primaryColorHct.chroma / 8.0 + 4.0,
169+
);
170+
}
164171

165-
// Need to create color scheme to get generated error tonal palette.
166-
const errorPalette = getMaterialDynamicScheme(
167-
primaryPalette,
168-
secondaryPalette,
169-
tertiaryPalette,
170-
neutralPalette,
171-
neutralVariantPalette,
172-
/* isDark */ false,
173-
/* contrastLevel */ 0,
174-
).errorPalette;
172+
let errorPalette;
173+
if (errorColor) {
174+
errorPalette = TonalPalette.fromHct(getHctFromHex(errorColor));
175+
} else {
176+
// Need to create color scheme to get generated error tonal palette.
177+
errorPalette = getMaterialDynamicScheme(
178+
primaryPalette,
179+
secondaryPalette,
180+
tertiaryPalette,
181+
neutralPalette,
182+
neutralVariantPalette,
183+
/* isDark */ false,
184+
/* contrastLevel */ 0,
185+
).errorPalette;
186+
}
175187

176188
return {
177189
primary: primaryPalette,
@@ -1007,6 +1019,8 @@ function getColorComment(
10071019
secondaryColor?: string,
10081020
tertiaryColor?: string,
10091021
neutralColor?: string,
1022+
neutralVariantColor?: string,
1023+
errorColor?: string,
10101024
) {
10111025
let colorComment = 'Color palettes are generated from primary: ' + primaryColor;
10121026
if (secondaryColor) {
@@ -1018,6 +1032,12 @@ function getColorComment(
10181032
if (neutralColor) {
10191033
colorComment += ', neutral: ' + neutralColor;
10201034
}
1035+
if (neutralVariantColor) {
1036+
colorComment += ', neutral variant: ' + neutralVariantColor;
1037+
}
1038+
if (errorColor) {
1039+
colorComment += ', error: ' + errorColor;
1040+
}
10211041
return colorComment;
10221042
}
10231043

@@ -1028,13 +1048,17 @@ export default function (options: Schema): Rule {
10281048
options.secondaryColor,
10291049
options.tertiaryColor,
10301050
options.neutralColor,
1051+
options.neutralVariantColor,
1052+
options.errorColor,
10311053
);
10321054

10331055
const colorPalettes = getColorPalettes(
10341056
options.primaryColor,
10351057
options.secondaryColor,
10361058
options.tertiaryColor,
10371059
options.neutralColor,
1060+
options.neutralVariantColor,
1061+
options.errorColor,
10381062
);
10391063

10401064
let lightHighContrastColorScheme: DynamicScheme;
@@ -1059,6 +1083,13 @@ export default function (options: Schema): Rule {
10591083
/* isDark */ true,
10601084
/* contrastLevel */ 1.0,
10611085
);
1086+
1087+
// Error palettes get generated by the color scheme's other palettes. Override the generated
1088+
// error palette with the custom one if applicable.
1089+
if (options.errorColor) {
1090+
lightHighContrastColorScheme.errorPalette = colorPalettes.error;
1091+
darkHighContrastColorScheme.errorPalette = colorPalettes.error;
1092+
}
10621093
}
10631094

10641095
if (options.isScss) {
@@ -1098,6 +1129,13 @@ export default function (options: Schema): Rule {
10981129
/* contrastLevel */ 0,
10991130
);
11001131

1132+
// Error palettes get generated by the color scheme's other palettes. Override the generated
1133+
// error palette with the custom one if applicable.
1134+
if (options.errorColor) {
1135+
lightColorScheme.errorPalette = colorPalettes.error;
1136+
darkColorScheme.errorPalette = colorPalettes.error;
1137+
}
1138+
11011139
themeCss += getAllSysVariablesCSS(lightColorScheme, darkColorScheme);
11021140

11031141
// Add high contrast media query to overwrite the color values when the user specifies

‎src/material/schematics/ng-generate/theme-color/schema.d.ts

+8
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,14 @@ export interface Schema {
2323
* Color to override the neutral color palette.
2424
*/
2525
neutralColor?: string;
26+
/**
27+
* Color to override the neutral variant color palette.
28+
*/
29+
neutralVariantColor?: string;
30+
/**
31+
* Color to override the error color palette.
32+
*/
33+
errorColor?: string;
2634
/**
2735
* Whether to create high contrast override theme mixins.
2836
*/

0 commit comments

Comments
 (0)
Please sign in to comment.