1+ <?php
2+
3+ namespace ProgrammatorDev \Validator \Rule ;
4+
5+ use ProgrammatorDev \Validator \Exception \CssColorException ;
6+ use ProgrammatorDev \Validator \Exception \UnexpectedOptionException ;
7+ use ProgrammatorDev \Validator \Exception \UnexpectedTypeException ;
8+
9+ class CssColor extends AbstractRule implements RuleInterface
10+ {
11+ public const FORMAT_HEX_LONG = 'hex-long ' ;
12+ public const FORMAT_HEX_LONG_WITH_ALPHA = 'hex-long-with-alpha ' ;
13+ public const FORMAT_HEX_SHORT = 'hex-short ' ;
14+ public const FORMAT_HEX_SHORT_WITH_ALPHA = 'hex-short-with-alpha ' ;
15+ public const FORMAT_BASIC_NAMED_COLORS = 'basic-named-colors ' ;
16+ public const FORMAT_EXTENDED_NAMED_COLORS = 'extended-named-colors ' ;
17+ public const FORMAT_SYSTEM_COLORS = 'system-colors ' ;
18+ public const FORMAT_KEYWORDS = 'keywords ' ;
19+ public const FORMAT_RGB = 'rgb ' ;
20+ public const FORMAT_RGBA = 'rgba ' ;
21+ public const FORMAT_HSL = 'hsl ' ;
22+ public const FORMAT_HSLA = 'hsla ' ;
23+
24+ private const COLOR_FORMATS = [
25+ self ::FORMAT_HEX_LONG ,
26+ self ::FORMAT_HEX_LONG_WITH_ALPHA ,
27+ self ::FORMAT_HEX_SHORT ,
28+ self ::FORMAT_HEX_SHORT_WITH_ALPHA ,
29+ self ::FORMAT_BASIC_NAMED_COLORS ,
30+ self ::FORMAT_EXTENDED_NAMED_COLORS ,
31+ self ::FORMAT_SYSTEM_COLORS ,
32+ self ::FORMAT_KEYWORDS ,
33+ self ::FORMAT_RGB ,
34+ self ::FORMAT_RGBA ,
35+ self ::FORMAT_HSL ,
36+ self ::FORMAT_HSLA
37+ ];
38+
39+ private const PATTERN_HEX_LONG = '/^#[0-9a-f]{6}$/i ' ;
40+ private const PATTERN_HEX_LONG_WITH_ALPHA = '/^#[0-9a-f]{8}$/i ' ;
41+ private const PATTERN_HEX_SHORT = '/^#[0-9a-f]{3}$/i ' ;
42+ private const PATTERN_HEX_SHORT_WITH_ALPHA = '/^#[0-9a-f]{4}$/i ' ;
43+ // https://www.w3.org/wiki/CSS/Properties/color/keywords#Basic_Colors
44+ private const PATTERN_BASIC_NAMED_COLORS = '/^(black|silver|gray|white|maroon|red|purple|fuchsia|green|lime|olive|yellow|navy|blue|teal|aqua)$/i ' ;
45+ // https://www.w3.org/wiki/CSS/Properties/color/keywords#Extended_colors
46+ private const PATTERN_EXTENDED_NAMED_COLORS = '/^(aliceblue|antiquewhite|aqua|aquamarine|azure|beige|bisque|black|blanchedalmond|blue|blueviolet|brown|burlywood|cadetblue|chartreuse|chocolate|coral|cornflowerblue|cornsilk|crimson|cyan|darkblue|darkcyan|darkgoldenrod|darkgray|darkgreen|darkgrey|darkkhaki|darkmagenta|darkolivegreen|darkorange|darkorchid|darkred|darksalmon|darkseagreen|darkslateblue|darkslategray|darkslategrey|darkturquoise|darkviolet|deeppink|deepskyblue|dimgray|dimgrey|dodgerblue|firebrick|floralwhite|forestgreen|fuchsia|gainsboro|ghostwhite|gold|goldenrod|gray|green|greenyellow|grey|honeydew|hotpink|indianred|indigo|ivory|khaki|lavender|lavenderblush|lawngreen|lemonchiffon|lightblue|lightcoral|lightcyan|lightgoldenrodyellow|lightgray|lightgreen|lightgrey|lightpink|lightsalmon|lightseagreen|lightskyblue|lightslategray|lightslategrey|lightsteelblue|lightyellow|lime|limegreen|linen|magenta|maroon|mediumaquamarine|mediumblue|mediumorchid|mediumpurple|mediumseagreen|mediumslateblue|mediumspringgreen|mediumturquoise|mediumvioletred|midnightblue|mintcream|mistyrose|moccasin|navajowhite|navy|oldlace|olive|olivedrab|orange|orangered|orchid|palegoldenrod|palegreen|paleturquoise|palevioletred|papayawhip|peachpuff|peru|pink|plum|powderblue|purple|red|rosybrown|royalblue|saddlebrown|salmon|sandybrown|seagreen|seashell|sienna|silver|skyblue|slateblue|slategray|slategrey|snow|springgreen|steelblue|tan|teal|thistle|tomato|turquoise|violet|wheat|white|whitesmoke|yellow|yellowgreen)$/i ' ;
47+ // https://drafts.csswg.org/css-color/#css-system-colors
48+ private const PATTERN_SYSTEM_COLORS = '/^(AccentColor|AccentColorText|ActiveText|ButtonBorder|ButtonFace|ButtonText|Canvas|CanvasText|Field|FieldText|GrayText|Highlight|HighlightText|LinkText|Mark|MarkText|SelectedItem|SelectedItemText|VisitedText)$/i ' ;
49+ // https://drafts.csswg.org/css-color/#transparent-color
50+ // https://drafts.csswg.org/css-color/#currentcolor-color
51+ private const PATTERN_KEYWORDS = '/^(transparent|currentColor)$/i ' ;
52+ private const PATTERN_RGB = '/^rgb\(\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d)\s*\)$/i ' ;
53+ private const PATTERN_RGBA = '/^rgba\(\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|0?\.\d+|1(\.0)?)\s*\)$/i ' ;
54+ private const PATTERN_HSL = '/^hsl\(\s*(0|360|35\d|3[0-4]\d|[12]\d\d|0?\d?\d),\s*(0|100|\d{1,2})%,\s*(0|100|\d{1,2})%\s*\)$/i ' ;
55+ private const PATTERN_HSLA = '/^hsla\(\s*(0|360|35\d|3[0-4]\d|[12]\d\d|0?\d?\d),\s*(0|100|\d{1,2})%,\s*(0|100|\d{1,2})%,\s*(0|0?\.\d+|1(\.0)?)\s*\)$/i ' ;
56+
57+ private const COLOR_PATTERNS = [
58+ self ::FORMAT_HEX_LONG => self ::PATTERN_HEX_LONG ,
59+ self ::FORMAT_HEX_LONG_WITH_ALPHA => self ::PATTERN_HEX_LONG_WITH_ALPHA ,
60+ self ::FORMAT_HEX_SHORT => self ::PATTERN_HEX_SHORT ,
61+ self ::FORMAT_HEX_SHORT_WITH_ALPHA => self ::PATTERN_HEX_SHORT_WITH_ALPHA ,
62+ self ::FORMAT_BASIC_NAMED_COLORS => self ::PATTERN_BASIC_NAMED_COLORS ,
63+ self ::FORMAT_EXTENDED_NAMED_COLORS => self ::PATTERN_EXTENDED_NAMED_COLORS ,
64+ self ::FORMAT_SYSTEM_COLORS => self ::PATTERN_SYSTEM_COLORS ,
65+ self ::FORMAT_KEYWORDS => self ::PATTERN_KEYWORDS ,
66+ self ::FORMAT_RGB => self ::PATTERN_RGB ,
67+ self ::FORMAT_RGBA => self ::PATTERN_RGBA ,
68+ self ::FORMAT_HSL => self ::PATTERN_HSL ,
69+ self ::FORMAT_HSLA => self ::PATTERN_HSLA
70+ ];
71+
72+ private array $ formats = self ::COLOR_FORMATS ;
73+ private string $ message = 'The {{ name }} value is not a valid CSS color. ' ;
74+
75+ public function __construct (
76+ ?array $ formats = null ,
77+ ?string $ message = null
78+ )
79+ {
80+ $ this ->formats = $ formats ?? $ this ->formats ;
81+ $ this ->message = $ message ?? $ this ->message ;
82+ }
83+
84+ public function assert (mixed $ value , ?string $ name = null ): void
85+ {
86+ foreach ($ this ->formats as $ format ) {
87+ if (!\in_array ($ format , self ::COLOR_FORMATS , true )) {
88+ throw new UnexpectedOptionException ('format ' , self ::COLOR_FORMATS , $ format );
89+ }
90+ }
91+
92+ if (!\is_string ($ value )) {
93+ throw new UnexpectedTypeException ('string ' , get_debug_type ($ value ));
94+ }
95+
96+ foreach ($ this ->formats as $ format ) {
97+ $ pattern = self ::COLOR_PATTERNS [$ format ];
98+
99+ // it is valid if at least one pattern matches
100+ if (\preg_match ($ pattern , $ value )) {
101+ return ;
102+ }
103+ }
104+
105+ throw new CssColorException (
106+ message: $ this ->message ,
107+ parameters: [
108+ 'value ' => $ value ,
109+ 'name ' => $ name ,
110+ 'formats ' => $ this ->formats
111+ ]
112+ );
113+ }
114+ }
0 commit comments