Parsing support for text-underline-offset and text-decoration-thickness
[WebKit-https.git] / Source / WebCore / css / CSSPrimitiveValueMappings.h
1 /*
2  * Copyright (C) 2007 Alexey Proskuryakov <ap@nypop.com>.
3  * Copyright (C) 2008, 2009, 2010, 2011 Apple Inc. All rights reserved.
4  * Copyright (C) 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/)
5  * Copyright (C) 2009 Jeff Schiller <codedread@gmail.com>
6  * Copyright (C) Research In Motion Limited 2010. All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  *
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29
30 #pragma once
31
32 #include "CSSCalculationValue.h"
33 #include "CSSFontFamily.h"
34 #include "CSSPrimitiveValue.h"
35 #include "CSSReflectionDirection.h"
36 #include "CSSToLengthConversionData.h"
37 #include "CSSValueKeywords.h"
38 #include "GraphicsTypes.h"
39 #include "Length.h"
40 #include "LineClampValue.h"
41 #include "RenderStyleConstants.h"
42 #include "SVGRenderStyleDefs.h"
43 #include "TextFlags.h"
44 #include "ThemeTypes.h"
45 #include "UnicodeBidi.h"
46 #include "WritingMode.h"
47 #include <wtf/MathExtras.h>
48 #include <wtf/OptionSet.h>
49
50 #if ENABLE(CSS_IMAGE_ORIENTATION)
51 #include "ImageOrientation.h"
52 #endif
53
54 namespace WebCore {
55
56 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(short i)
57     : CSSValue(PrimitiveClass)
58 {
59     m_primitiveUnitType = CSS_NUMBER;
60     m_value.num = static_cast<double>(i);
61 }
62
63 template<> inline CSSPrimitiveValue::operator short() const
64 {
65     if (m_primitiveUnitType == CSS_NUMBER)
66         return clampTo<short>(m_value.num);
67
68     ASSERT_NOT_REACHED();
69     return 0;
70 }
71
72 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(unsigned short i)
73     : CSSValue(PrimitiveClass)
74 {
75     m_primitiveUnitType = CSS_NUMBER;
76     m_value.num = static_cast<double>(i);
77 }
78
79 template<> inline CSSPrimitiveValue::operator unsigned short() const
80 {
81     if (primitiveType() == CSS_NUMBER)
82         return value<unsigned short>();
83
84     ASSERT_NOT_REACHED();
85     return 0;
86 }
87
88 template<> inline CSSPrimitiveValue::operator int() const
89 {
90     if (primitiveType() == CSS_NUMBER)
91         return value<int>();
92
93     ASSERT_NOT_REACHED();
94     return 0;
95 }
96
97 template<> inline CSSPrimitiveValue::operator unsigned() const
98 {
99     if (primitiveType() == CSS_NUMBER)
100         return value<unsigned>();
101
102     ASSERT_NOT_REACHED();
103     return 0;
104 }
105
106
107 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(float i)
108     : CSSValue(PrimitiveClass)
109 {
110     m_primitiveUnitType = CSS_NUMBER;
111     m_value.num = static_cast<double>(i);
112 }
113
114 template<> inline CSSPrimitiveValue::operator float() const
115 {
116     if (primitiveType() == CSS_NUMBER)
117         return value<float>();
118
119     ASSERT_NOT_REACHED();
120     return 0.0f;
121 }
122
123 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineClampValue i)
124     : CSSValue(PrimitiveClass)
125 {
126     m_primitiveUnitType = i.isPercentage() ? CSS_PERCENTAGE : CSS_NUMBER;
127     m_value.num = static_cast<double>(i.value());
128 }
129
130 template<> inline CSSPrimitiveValue::operator LineClampValue() const
131 {
132     if (primitiveType() == CSS_NUMBER)
133         return LineClampValue(value<int>(), LineClamp::LineCount);
134
135     if (primitiveType() == CSS_PERCENTAGE)
136         return LineClampValue(value<int>(), LineClamp::Percentage);
137
138     ASSERT_NOT_REACHED();
139     return LineClampValue();
140 }
141
142 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(CSSReflectionDirection e)
143     : CSSValue(PrimitiveClass)
144 {
145     m_primitiveUnitType = CSS_VALUE_ID;
146     switch (e) {
147     case ReflectionAbove:
148         m_value.valueID = CSSValueAbove;
149         break;
150     case ReflectionBelow:
151         m_value.valueID = CSSValueBelow;
152         break;
153     case ReflectionLeft:
154         m_value.valueID = CSSValueLeft;
155         break;
156     case ReflectionRight:
157         m_value.valueID = CSSValueRight;
158     }
159 }
160
161 template<> inline CSSPrimitiveValue::operator CSSReflectionDirection() const
162 {
163     ASSERT(isValueID());
164
165     switch (m_value.valueID) {
166     case CSSValueAbove:
167         return ReflectionAbove;
168     case CSSValueBelow:
169         return ReflectionBelow;
170     case CSSValueLeft:
171         return ReflectionLeft;
172     case CSSValueRight:
173         return ReflectionRight;
174     default:
175         break;
176     }
177
178     ASSERT_NOT_REACHED();
179     return ReflectionBelow;
180 }
181
182 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ColumnFill columnFill)
183     : CSSValue(PrimitiveClass)
184 {
185     m_primitiveUnitType = CSS_VALUE_ID;
186     switch (columnFill) {
187     case ColumnFill::Auto:
188         m_value.valueID = CSSValueAuto;
189         break;
190     case ColumnFill::Balance:
191         m_value.valueID = CSSValueBalance;
192         break;
193     }
194 }
195
196 template<> inline CSSPrimitiveValue::operator ColumnFill() const
197 {
198     if (m_primitiveUnitType == CSS_VALUE_ID) {
199         if (m_value.valueID == CSSValueBalance)
200             return ColumnFill::Balance;
201         if (m_value.valueID == CSSValueAuto)
202             return ColumnFill::Auto;
203     }
204     ASSERT_NOT_REACHED();
205     return ColumnFill::Balance;
206 }
207
208 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ColumnSpan columnSpan)
209     : CSSValue(PrimitiveClass)
210 {
211     m_primitiveUnitType = CSS_VALUE_ID;
212     switch (columnSpan) {
213     case ColumnSpan::All:
214         m_value.valueID = CSSValueAll;
215         break;
216     case ColumnSpan::None:
217         m_value.valueID = CSSValueNone;
218         break;
219     }
220 }
221
222 template<> inline CSSPrimitiveValue::operator ColumnSpan() const
223 {
224     // Map 1 to none for compatibility reasons.
225     if (m_primitiveUnitType == CSS_NUMBER && m_value.num == 1)
226         return ColumnSpan::None;
227
228     ASSERT(isValueID());
229
230     switch (m_value.valueID) {
231     case CSSValueAll:
232         return ColumnSpan::All;
233     case CSSValueNone:
234         return ColumnSpan::None;
235     default:
236         break;
237     }
238
239     ASSERT_NOT_REACHED();
240     return ColumnSpan::None;
241 }
242
243
244 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(PrintColorAdjust value)
245     : CSSValue(PrimitiveClass)
246 {
247     m_primitiveUnitType = CSS_VALUE_ID;
248     switch (value) {
249     case PrintColorAdjust::Exact:
250         m_value.valueID = CSSValueExact;
251         break;
252     case PrintColorAdjust::Economy:
253         m_value.valueID = CSSValueEconomy;
254         break;
255     }
256 }
257
258 template<> inline CSSPrimitiveValue::operator PrintColorAdjust() const
259 {
260     ASSERT(isValueID());
261
262     switch (m_value.valueID) {
263     case CSSValueEconomy:
264         return PrintColorAdjust::Economy;
265     case CSSValueExact:
266         return PrintColorAdjust::Exact;
267     default:
268         break;
269     }
270
271     ASSERT_NOT_REACHED();
272     return PrintColorAdjust::Economy;
273 }
274
275
276 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BorderStyle e)
277     : CSSValue(PrimitiveClass)
278 {
279     m_primitiveUnitType = CSS_VALUE_ID;
280     switch (e) {
281     case BorderStyle::None:
282         m_value.valueID = CSSValueNone;
283         break;
284     case BorderStyle::Hidden:
285         m_value.valueID = CSSValueHidden;
286         break;
287     case BorderStyle::Inset:
288         m_value.valueID = CSSValueInset;
289         break;
290     case BorderStyle::Groove:
291         m_value.valueID = CSSValueGroove;
292         break;
293     case BorderStyle::Ridge:
294         m_value.valueID = CSSValueRidge;
295         break;
296     case BorderStyle::Outset:
297         m_value.valueID = CSSValueOutset;
298         break;
299     case BorderStyle::Dotted:
300         m_value.valueID = CSSValueDotted;
301         break;
302     case BorderStyle::Dashed:
303         m_value.valueID = CSSValueDashed;
304         break;
305     case BorderStyle::Solid:
306         m_value.valueID = CSSValueSolid;
307         break;
308     case BorderStyle::Double:
309         m_value.valueID = CSSValueDouble;
310         break;
311     }
312 }
313
314 template<> inline CSSPrimitiveValue::operator BorderStyle() const
315 {
316     ASSERT(isValueID());
317
318     if (m_value.valueID == CSSValueAuto) // Valid for CSS outline-style
319         return BorderStyle::Dotted;
320     return static_cast<BorderStyle>(m_value.valueID - CSSValueNone);
321 }
322
323 template<> inline CSSPrimitiveValue::operator OutlineIsAuto() const
324 {
325     ASSERT(isValueID());
326
327     if (m_value.valueID == CSSValueAuto)
328         return OutlineIsAuto::On;
329     return OutlineIsAuto::Off;
330 }
331
332 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(CompositeOperator e)
333     : CSSValue(PrimitiveClass)
334 {
335     m_primitiveUnitType = CSS_VALUE_ID;
336     switch (e) {
337     case CompositeClear:
338         m_value.valueID = CSSValueClear;
339         break;
340     case CompositeCopy:
341         m_value.valueID = CSSValueCopy;
342         break;
343     case CompositeSourceOver:
344         m_value.valueID = CSSValueSourceOver;
345         break;
346     case CompositeSourceIn:
347         m_value.valueID = CSSValueSourceIn;
348         break;
349     case CompositeSourceOut:
350         m_value.valueID = CSSValueSourceOut;
351         break;
352     case CompositeSourceAtop:
353         m_value.valueID = CSSValueSourceAtop;
354         break;
355     case CompositeDestinationOver:
356         m_value.valueID = CSSValueDestinationOver;
357         break;
358     case CompositeDestinationIn:
359         m_value.valueID = CSSValueDestinationIn;
360         break;
361     case CompositeDestinationOut:
362         m_value.valueID = CSSValueDestinationOut;
363         break;
364     case CompositeDestinationAtop:
365         m_value.valueID = CSSValueDestinationAtop;
366         break;
367     case CompositeXOR:
368         m_value.valueID = CSSValueXor;
369         break;
370     case CompositePlusDarker:
371         m_value.valueID = CSSValuePlusDarker;
372         break;
373     case CompositePlusLighter:
374         m_value.valueID = CSSValuePlusLighter;
375         break;
376     case CompositeDifference:
377         ASSERT_NOT_REACHED();
378         break;
379     }
380 }
381
382 template<> inline CSSPrimitiveValue::operator CompositeOperator() const
383 {
384     ASSERT(isValueID());
385
386     switch (m_value.valueID) {
387     case CSSValueClear:
388         return CompositeClear;
389     case CSSValueCopy:
390         return CompositeCopy;
391     case CSSValueSourceOver:
392         return CompositeSourceOver;
393     case CSSValueSourceIn:
394         return CompositeSourceIn;
395     case CSSValueSourceOut:
396         return CompositeSourceOut;
397     case CSSValueSourceAtop:
398         return CompositeSourceAtop;
399     case CSSValueDestinationOver:
400         return CompositeDestinationOver;
401     case CSSValueDestinationIn:
402         return CompositeDestinationIn;
403     case CSSValueDestinationOut:
404         return CompositeDestinationOut;
405     case CSSValueDestinationAtop:
406         return CompositeDestinationAtop;
407     case CSSValueXor:
408         return CompositeXOR;
409     case CSSValuePlusDarker:
410         return CompositePlusDarker;
411     case CSSValuePlusLighter:
412         return CompositePlusLighter;
413     default:
414         break;
415     }
416
417     ASSERT_NOT_REACHED();
418     return CompositeClear;
419 }
420
421 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ControlPart e)
422     : CSSValue(PrimitiveClass)
423 {
424     m_primitiveUnitType = CSS_VALUE_ID;
425     switch (e) {
426     case NoControlPart:
427         m_value.valueID = CSSValueNone;
428         break;
429     case CheckboxPart:
430         m_value.valueID = CSSValueCheckbox;
431         break;
432     case RadioPart:
433         m_value.valueID = CSSValueRadio;
434         break;
435     case PushButtonPart:
436         m_value.valueID = CSSValuePushButton;
437         break;
438     case SquareButtonPart:
439         m_value.valueID = CSSValueSquareButton;
440         break;
441     case ButtonPart:
442         m_value.valueID = CSSValueButton;
443         break;
444     case ButtonBevelPart:
445         m_value.valueID = CSSValueButtonBevel;
446         break;
447     case DefaultButtonPart:
448         m_value.valueID = CSSValueDefaultButton;
449         break;
450     case InnerSpinButtonPart:
451         m_value.valueID = CSSValueInnerSpinButton;
452         break;
453     case ListboxPart:
454         m_value.valueID = CSSValueListbox;
455         break;
456     case ListItemPart:
457         m_value.valueID = CSSValueListitem;
458         break;
459     case MediaEnterFullscreenButtonPart:
460         m_value.valueID = CSSValueMediaEnterFullscreenButton;
461         break;
462     case MediaExitFullscreenButtonPart:
463         m_value.valueID = CSSValueMediaExitFullscreenButton;
464         break;
465     case MediaPlayButtonPart:
466         m_value.valueID = CSSValueMediaPlayButton;
467         break;
468     case MediaOverlayPlayButtonPart:
469         m_value.valueID = CSSValueMediaOverlayPlayButton;
470         break;
471     case MediaMuteButtonPart:
472         m_value.valueID = CSSValueMediaMuteButton;
473         break;
474     case MediaSeekBackButtonPart:
475         m_value.valueID = CSSValueMediaSeekBackButton;
476         break;
477     case MediaSeekForwardButtonPart:
478         m_value.valueID = CSSValueMediaSeekForwardButton;
479         break;
480     case MediaRewindButtonPart:
481         m_value.valueID = CSSValueMediaRewindButton;
482         break;
483     case MediaReturnToRealtimeButtonPart:
484         m_value.valueID = CSSValueMediaReturnToRealtimeButton;
485         break;
486     case MediaToggleClosedCaptionsButtonPart:
487         m_value.valueID = CSSValueMediaToggleClosedCaptionsButton;
488         break;
489     case MediaSliderPart:
490         m_value.valueID = CSSValueMediaSlider;
491         break;
492     case MediaSliderThumbPart:
493         m_value.valueID = CSSValueMediaSliderthumb;
494         break;
495     case MediaVolumeSliderContainerPart:
496         m_value.valueID = CSSValueMediaVolumeSliderContainer;
497         break;
498     case MediaVolumeSliderPart:
499         m_value.valueID = CSSValueMediaVolumeSlider;
500         break;
501     case MediaVolumeSliderMuteButtonPart:
502         m_value.valueID = CSSValueMediaVolumeSliderMuteButton;
503         break;
504     case MediaVolumeSliderThumbPart:
505         m_value.valueID = CSSValueMediaVolumeSliderthumb;
506         break;
507     case MediaControlsBackgroundPart:
508         m_value.valueID = CSSValueMediaControlsBackground;
509         break;
510     case MediaControlsFullscreenBackgroundPart:
511         m_value.valueID = CSSValueMediaControlsFullscreenBackground;
512         break;
513     case MediaFullScreenVolumeSliderPart:
514         m_value.valueID = CSSValueMediaFullscreenVolumeSlider;
515         break;
516     case MediaFullScreenVolumeSliderThumbPart:
517         m_value.valueID = CSSValueMediaFullscreenVolumeSliderThumb;
518         break;
519     case MediaCurrentTimePart:
520         m_value.valueID = CSSValueMediaCurrentTimeDisplay;
521         break;
522     case MediaTimeRemainingPart:
523         m_value.valueID = CSSValueMediaTimeRemainingDisplay;
524         break;
525     case MediaControlsLightBarBackgroundPart:
526         m_value.valueID = CSSValueMediaControlsLightBarBackground;
527         break;
528     case MediaControlsDarkBarBackgroundPart:
529         m_value.valueID = CSSValueMediaControlsDarkBarBackground;
530         break;
531     case MenulistPart:
532         m_value.valueID = CSSValueMenulist;
533         break;
534     case MenulistButtonPart:
535         m_value.valueID = CSSValueMenulistButton;
536         break;
537     case MenulistTextPart:
538         m_value.valueID = CSSValueMenulistText;
539         break;
540     case MenulistTextFieldPart:
541         m_value.valueID = CSSValueMenulistTextfield;
542         break;
543     case MeterPart:
544         m_value.valueID = CSSValueMeter;
545         break;
546     case RelevancyLevelIndicatorPart:
547         m_value.valueID = CSSValueRelevancyLevelIndicator;
548         break;
549     case ContinuousCapacityLevelIndicatorPart:
550         m_value.valueID = CSSValueContinuousCapacityLevelIndicator;
551         break;
552     case DiscreteCapacityLevelIndicatorPart:
553         m_value.valueID = CSSValueDiscreteCapacityLevelIndicator;
554         break;
555     case RatingLevelIndicatorPart:
556         m_value.valueID = CSSValueRatingLevelIndicator;
557         break;
558     case ProgressBarPart:
559         m_value.valueID = CSSValueProgressBar;
560         break;
561     case ProgressBarValuePart:
562         m_value.valueID = CSSValueProgressBarValue;
563         break;
564     case SliderHorizontalPart:
565         m_value.valueID = CSSValueSliderHorizontal;
566         break;
567     case SliderVerticalPart:
568         m_value.valueID = CSSValueSliderVertical;
569         break;
570     case SliderThumbHorizontalPart:
571         m_value.valueID = CSSValueSliderthumbHorizontal;
572         break;
573     case SliderThumbVerticalPart:
574         m_value.valueID = CSSValueSliderthumbVertical;
575         break;
576     case CaretPart:
577         m_value.valueID = CSSValueCaret;
578         break;
579     case SearchFieldPart:
580         m_value.valueID = CSSValueSearchfield;
581         break;
582     case SearchFieldDecorationPart:
583         m_value.valueID = CSSValueSearchfieldDecoration;
584         break;
585     case SearchFieldResultsDecorationPart:
586         m_value.valueID = CSSValueSearchfieldResultsDecoration;
587         break;
588     case SearchFieldResultsButtonPart:
589         m_value.valueID = CSSValueSearchfieldResultsButton;
590         break;
591     case SearchFieldCancelButtonPart:
592         m_value.valueID = CSSValueSearchfieldCancelButton;
593         break;
594     case SnapshottedPluginOverlayPart:
595         m_value.valueID = CSSValueSnapshottedPluginOverlay;
596         break;
597     case TextFieldPart:
598         m_value.valueID = CSSValueTextfield;
599         break;
600     case TextAreaPart:
601         m_value.valueID = CSSValueTextarea;
602         break;
603     case CapsLockIndicatorPart:
604         m_value.valueID = CSSValueCapsLockIndicator;
605         break;
606 #if ENABLE(ATTACHMENT_ELEMENT)
607     case AttachmentPart:
608         m_value.valueID = CSSValueAttachment;
609         break;
610     case BorderlessAttachmentPart:
611         m_value.valueID = CSSValueBorderlessAttachment;
612         break;
613 #endif
614 #if ENABLE(SERVICE_CONTROLS)
615     case ImageControlsButtonPart:
616         m_value.valueID = CSSValueImageControlsButton;
617         break;
618 #endif
619 #if ENABLE(APPLE_PAY)
620     case ApplePayButtonPart:
621         m_value.valueID = CSSValueApplePayButton;
622         break;
623 #endif
624 #if ENABLE(INPUT_TYPE_COLOR)
625     case ColorWellPart:
626         m_value.valueID = CSSValueColorWell;
627         break;
628 #endif
629 #if ENABLE(DATALIST_ELEMENT)
630     case ListButtonPart:
631         m_value.valueID = CSSValueListButton;
632         break;
633 #endif
634     }
635 }
636
637 template<> inline CSSPrimitiveValue::operator ControlPart() const
638 {
639     ASSERT(isValueID());
640
641     if (m_value.valueID == CSSValueNone)
642         return NoControlPart;
643     return ControlPart(m_value.valueID - CSSValueCheckbox + 1);
644 }
645
646 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BackfaceVisibility e)
647     : CSSValue(PrimitiveClass)
648 {
649     m_primitiveUnitType = CSS_VALUE_ID;
650     switch (e) {
651     case BackfaceVisibility::Visible:
652         m_value.valueID = CSSValueVisible;
653         break;
654     case BackfaceVisibility::Hidden:
655         m_value.valueID = CSSValueHidden;
656         break;
657     }
658 }
659
660 template<> inline CSSPrimitiveValue::operator BackfaceVisibility() const
661 {
662     ASSERT(isValueID());
663
664     switch (m_value.valueID) {
665     case CSSValueVisible:
666         return BackfaceVisibility::Visible;
667     case CSSValueHidden:
668         return BackfaceVisibility::Hidden;
669     default:
670         break;
671     }
672
673     ASSERT_NOT_REACHED();
674     return BackfaceVisibility::Hidden;
675 }
676
677
678 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FillAttachment e)
679     : CSSValue(PrimitiveClass)
680 {
681     m_primitiveUnitType = CSS_VALUE_ID;
682     switch (e) {
683     case FillAttachment::ScrollBackground:
684         m_value.valueID = CSSValueScroll;
685         break;
686     case FillAttachment::LocalBackground:
687         m_value.valueID = CSSValueLocal;
688         break;
689     case FillAttachment::FixedBackground:
690         m_value.valueID = CSSValueFixed;
691         break;
692     }
693 }
694
695 template<> inline CSSPrimitiveValue::operator FillAttachment() const
696 {
697     ASSERT(isValueID());
698
699     switch (m_value.valueID) {
700     case CSSValueScroll:
701         return FillAttachment::ScrollBackground;
702     case CSSValueLocal:
703         return FillAttachment::LocalBackground;
704     case CSSValueFixed:
705         return FillAttachment::FixedBackground;
706     default:
707         break;
708     }
709
710     ASSERT_NOT_REACHED();
711     return FillAttachment::ScrollBackground;
712 }
713
714 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FillBox e)
715     : CSSValue(PrimitiveClass)
716 {
717     m_primitiveUnitType = CSS_VALUE_ID;
718     switch (e) {
719     case FillBox::Border:
720         m_value.valueID = CSSValueBorderBox;
721         break;
722     case FillBox::Padding:
723         m_value.valueID = CSSValuePaddingBox;
724         break;
725     case FillBox::Content:
726         m_value.valueID = CSSValueContentBox;
727         break;
728     case FillBox::Text:
729         m_value.valueID = CSSValueText;
730         break;
731     }
732 }
733
734 template<> inline CSSPrimitiveValue::operator FillBox() const
735 {
736     ASSERT(isValueID());
737
738     switch (m_value.valueID) {
739     case CSSValueBorder:
740     case CSSValueBorderBox:
741         return FillBox::Border;
742     case CSSValuePadding:
743     case CSSValuePaddingBox:
744         return FillBox::Padding;
745     case CSSValueContent:
746     case CSSValueContentBox:
747         return FillBox::Content;
748     case CSSValueText:
749     case CSSValueWebkitText:
750         return FillBox::Text;
751     default:
752         break;
753     }
754
755     ASSERT_NOT_REACHED();
756     return FillBox::Border;
757 }
758
759 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FillRepeat e)
760     : CSSValue(PrimitiveClass)
761 {
762     m_primitiveUnitType = CSS_VALUE_ID;
763     switch (e) {
764     case FillRepeat::Repeat:
765         m_value.valueID = CSSValueRepeat;
766         break;
767     case FillRepeat::NoRepeat:
768         m_value.valueID = CSSValueNoRepeat;
769         break;
770     case FillRepeat::Round:
771         m_value.valueID = CSSValueRound;
772         break;
773     case FillRepeat::Space:
774         m_value.valueID = CSSValueSpace;
775         break;
776     }
777 }
778
779 template<> inline CSSPrimitiveValue::operator FillRepeat() const
780 {
781     ASSERT(isValueID());
782
783     switch (m_value.valueID) {
784     case CSSValueRepeat:
785         return FillRepeat::Repeat;
786     case CSSValueNoRepeat:
787         return FillRepeat::NoRepeat;
788     case CSSValueRound:
789         return FillRepeat::Round;
790     case CSSValueSpace:
791         return FillRepeat::Space;
792     default:
793         break;
794     }
795
796     ASSERT_NOT_REACHED();
797     return FillRepeat::Repeat;
798 }
799
800 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxPack e)
801     : CSSValue(PrimitiveClass)
802 {
803     m_primitiveUnitType = CSS_VALUE_ID;
804     switch (e) {
805     case BoxPack::Start:
806         m_value.valueID = CSSValueStart;
807         break;
808     case BoxPack::Center:
809         m_value.valueID = CSSValueCenter;
810         break;
811     case BoxPack::End:
812         m_value.valueID = CSSValueEnd;
813         break;
814     case BoxPack::Justify:
815         m_value.valueID = CSSValueJustify;
816         break;
817     }
818 }
819
820 template<> inline CSSPrimitiveValue::operator BoxPack() const
821 {
822     ASSERT(isValueID());
823
824     switch (m_value.valueID) {
825     case CSSValueStart:
826         return BoxPack::Start;
827     case CSSValueEnd:
828         return BoxPack::End;
829     case CSSValueCenter:
830         return BoxPack::Center;
831     case CSSValueJustify:
832         return BoxPack::Justify;
833     default:
834         break;
835     }
836
837     ASSERT_NOT_REACHED();
838     return BoxPack::Justify;
839 }
840
841 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxAlignment e)
842     : CSSValue(PrimitiveClass)
843 {
844     m_primitiveUnitType = CSS_VALUE_ID;
845     switch (e) {
846     case BoxAlignment::Stretch:
847         m_value.valueID = CSSValueStretch;
848         break;
849     case BoxAlignment::Start:
850         m_value.valueID = CSSValueStart;
851         break;
852     case BoxAlignment::Center:
853         m_value.valueID = CSSValueCenter;
854         break;
855     case BoxAlignment::End:
856         m_value.valueID = CSSValueEnd;
857         break;
858     case BoxAlignment::Baseline:
859         m_value.valueID = CSSValueBaseline;
860         break;
861     }
862 }
863
864 template<> inline CSSPrimitiveValue::operator BoxAlignment() const
865 {
866     ASSERT(isValueID());
867
868     switch (m_value.valueID) {
869     case CSSValueStretch:
870         return BoxAlignment::Stretch;
871     case CSSValueStart:
872         return BoxAlignment::Start;
873     case CSSValueEnd:
874         return BoxAlignment::End;
875     case CSSValueCenter:
876         return BoxAlignment::Center;
877     case CSSValueBaseline:
878         return BoxAlignment::Baseline;
879     default:
880         break;
881     }
882
883     ASSERT_NOT_REACHED();
884     return BoxAlignment::Stretch;
885 }
886
887 #if ENABLE(CSS_BOX_DECORATION_BREAK)
888 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxDecorationBreak e)
889     : CSSValue(PrimitiveClass)
890 {
891     m_primitiveUnitType = CSS_VALUE_ID;
892     switch (e) {
893     case BoxDecorationBreak::Slice:
894         m_value.valueID = CSSValueSlice;
895         break;
896     case BoxDecorationBreak::Clone:
897         m_value.valueID = CSSValueClone;
898         break;
899     }
900 }
901
902 template<> inline CSSPrimitiveValue::operator BoxDecorationBreak() const
903 {
904     ASSERT(isValueID());
905
906     switch (m_value.valueID) {
907     case CSSValueSlice:
908         return BoxDecorationBreak::Slice;
909     case CSSValueClone:
910         return BoxDecorationBreak::Clone;
911     default:
912         break;
913     }
914
915     ASSERT_NOT_REACHED();
916     return BoxDecorationBreak::Slice;
917 }
918 #endif
919
920 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Edge e)
921     : CSSValue(PrimitiveClass)
922 {
923     m_primitiveUnitType = CSS_VALUE_ID;
924     switch (e) {
925     case Edge::Top:
926         m_value.valueID = CSSValueTop;
927         break;
928     case Edge::Right:
929         m_value.valueID = CSSValueRight;
930         break;
931     case Edge::Bottom:
932         m_value.valueID = CSSValueBottom;
933         break;
934     case Edge::Left:
935         m_value.valueID = CSSValueLeft;
936         break;
937     }
938 }
939
940 template<> inline CSSPrimitiveValue::operator Edge() const
941 {
942     ASSERT(isValueID());
943
944     switch (m_value.valueID) {
945     case CSSValueTop:
946         return Edge::Top;
947     case CSSValueRight:
948         return Edge::Right;
949     case CSSValueBottom:
950         return Edge::Bottom;
951     case CSSValueLeft:
952         return Edge::Left;
953     default:
954         break;
955     }
956
957     ASSERT_NOT_REACHED();
958     return Edge::Top;
959 }
960
961 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxSizing e)
962     : CSSValue(PrimitiveClass)
963 {
964     m_primitiveUnitType = CSS_VALUE_ID;
965     switch (e) {
966     case BoxSizing::BorderBox:
967         m_value.valueID = CSSValueBorderBox;
968         break;
969     case BoxSizing::ContentBox:
970         m_value.valueID = CSSValueContentBox;
971         break;
972     }
973 }
974
975 template<> inline CSSPrimitiveValue::operator BoxSizing() const
976 {
977     ASSERT(isValueID());
978
979     switch (m_value.valueID) {
980     case CSSValueBorderBox:
981         return BoxSizing::BorderBox;
982     case CSSValueContentBox:
983         return BoxSizing::ContentBox;
984     default:
985         break;
986     }
987
988     ASSERT_NOT_REACHED();
989     return BoxSizing::BorderBox;
990 }
991
992 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxDirection e)
993     : CSSValue(PrimitiveClass)
994 {
995     m_primitiveUnitType = CSS_VALUE_ID;
996     switch (e) {
997     case BoxDirection::Normal:
998         m_value.valueID = CSSValueNormal;
999         break;
1000     case BoxDirection::Reverse:
1001         m_value.valueID = CSSValueReverse;
1002         break;
1003     }
1004 }
1005
1006 template<> inline CSSPrimitiveValue::operator BoxDirection() const
1007 {
1008     ASSERT(isValueID());
1009
1010     switch (m_value.valueID) {
1011     case CSSValueNormal:
1012         return BoxDirection::Normal;
1013     case CSSValueReverse:
1014         return BoxDirection::Reverse;
1015     default:
1016         break;
1017     }
1018
1019     ASSERT_NOT_REACHED();
1020     return BoxDirection::Normal;
1021 }
1022
1023 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxLines e)
1024     : CSSValue(PrimitiveClass)
1025 {
1026     m_primitiveUnitType = CSS_VALUE_ID;
1027     switch (e) {
1028     case BoxLines::Single:
1029         m_value.valueID = CSSValueSingle;
1030         break;
1031     case BoxLines::Multiple:
1032         m_value.valueID = CSSValueMultiple;
1033         break;
1034     }
1035 }
1036
1037 template<> inline CSSPrimitiveValue::operator BoxLines() const
1038 {
1039     ASSERT(isValueID());
1040
1041     switch (m_value.valueID) {
1042     case CSSValueSingle:
1043         return BoxLines::Single;
1044     case CSSValueMultiple:
1045         return BoxLines::Multiple;
1046     default:
1047         break;
1048     }
1049
1050     ASSERT_NOT_REACHED();
1051     return BoxLines::Single;
1052 }
1053
1054 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BoxOrient e)
1055     : CSSValue(PrimitiveClass)
1056 {
1057     m_primitiveUnitType = CSS_VALUE_ID;
1058     switch (e) {
1059     case BoxOrient::Horizontal:
1060         m_value.valueID = CSSValueHorizontal;
1061         break;
1062     case BoxOrient::Vertical:
1063         m_value.valueID = CSSValueVertical;
1064         break;
1065     }
1066 }
1067
1068 template<> inline CSSPrimitiveValue::operator BoxOrient() const
1069 {
1070     ASSERT(isValueID());
1071
1072     switch (m_value.valueID) {
1073     case CSSValueHorizontal:
1074     case CSSValueInlineAxis:
1075         return BoxOrient::Horizontal;
1076     case CSSValueVertical:
1077     case CSSValueBlockAxis:
1078         return BoxOrient::Vertical;
1079     default:
1080         break;
1081     }
1082
1083     ASSERT_NOT_REACHED();
1084     return BoxOrient::Horizontal;
1085 }
1086
1087 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(CaptionSide e)
1088     : CSSValue(PrimitiveClass)
1089 {
1090     m_primitiveUnitType = CSS_VALUE_ID;
1091     switch (e) {
1092     case CaptionSide::Left:
1093         m_value.valueID = CSSValueLeft;
1094         break;
1095     case CaptionSide::Right:
1096         m_value.valueID = CSSValueRight;
1097         break;
1098     case CaptionSide::Top:
1099         m_value.valueID = CSSValueTop;
1100         break;
1101     case CaptionSide::Bottom:
1102         m_value.valueID = CSSValueBottom;
1103         break;
1104     }
1105 }
1106
1107 template<> inline CSSPrimitiveValue::operator CaptionSide() const
1108 {
1109     ASSERT(isValueID());
1110
1111     switch (m_value.valueID) {
1112     case CSSValueLeft:
1113         return CaptionSide::Left;
1114     case CSSValueRight:
1115         return CaptionSide::Right;
1116     case CSSValueTop:
1117         return CaptionSide::Top;
1118     case CSSValueBottom:
1119         return CaptionSide::Bottom;
1120     default:
1121         break;
1122     }
1123
1124     ASSERT_NOT_REACHED();
1125     return CaptionSide::Top;
1126 }
1127
1128 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Clear e)
1129     : CSSValue(PrimitiveClass)
1130 {
1131     m_primitiveUnitType = CSS_VALUE_ID;
1132     switch (e) {
1133     case Clear::None:
1134         m_value.valueID = CSSValueNone;
1135         break;
1136     case Clear::Left:
1137         m_value.valueID = CSSValueLeft;
1138         break;
1139     case Clear::Right:
1140         m_value.valueID = CSSValueRight;
1141         break;
1142     case Clear::Both:
1143         m_value.valueID = CSSValueBoth;
1144         break;
1145     }
1146 }
1147
1148 template<> inline CSSPrimitiveValue::operator Clear() const
1149 {
1150     ASSERT(isValueID());
1151
1152     switch (m_value.valueID) {
1153     case CSSValueNone:
1154         return Clear::None;
1155     case CSSValueLeft:
1156         return Clear::Left;
1157     case CSSValueRight:
1158         return Clear::Right;
1159     case CSSValueBoth:
1160         return Clear::Both;
1161     default:
1162         break;
1163     }
1164
1165     ASSERT_NOT_REACHED();
1166     return Clear::None;
1167 }
1168
1169 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(CursorType e)
1170     : CSSValue(PrimitiveClass)
1171 {
1172     m_primitiveUnitType = CSS_VALUE_ID;
1173     switch (e) {
1174     case CursorType::Auto:
1175         m_value.valueID = CSSValueAuto;
1176         break;
1177     case CursorType::Default:
1178         m_value.valueID = CSSValueDefault;
1179         break;
1180     case CursorType::None:
1181         m_value.valueID = CSSValueNone;
1182         break;
1183     case CursorType::ContextMenu:
1184         m_value.valueID = CSSValueContextMenu;
1185         break;
1186     case CursorType::Help:
1187         m_value.valueID = CSSValueHelp;
1188         break;
1189     case CursorType::Pointer:
1190         m_value.valueID = CSSValuePointer;
1191         break;
1192     case CursorType::Progress:
1193         m_value.valueID = CSSValueProgress;
1194         break;
1195     case CursorType::Wait:
1196         m_value.valueID = CSSValueWait;
1197         break;
1198     case CursorType::Cell:
1199         m_value.valueID = CSSValueCell;
1200         break;
1201     case CursorType::Crosshair:
1202         m_value.valueID = CSSValueCrosshair;
1203         break;
1204     case CursorType::Text:
1205         m_value.valueID = CSSValueText;
1206         break;
1207     case CursorType::VerticalText:
1208         m_value.valueID = CSSValueVerticalText;
1209         break;
1210     case CursorType::Alias:
1211         m_value.valueID = CSSValueAlias;
1212         break;
1213     case CursorType::Copy:
1214         m_value.valueID = CSSValueCopy;
1215         break;
1216     case CursorType::Move:
1217         m_value.valueID = CSSValueMove;
1218         break;
1219     case CursorType::NoDrop:
1220         m_value.valueID = CSSValueNoDrop;
1221         break;
1222     case CursorType::NotAllowed:
1223         m_value.valueID = CSSValueNotAllowed;
1224         break;
1225     case CursorType::Grab:
1226         m_value.valueID = CSSValueGrab;
1227         break;
1228     case CursorType::Grabbing:
1229         m_value.valueID = CSSValueGrabbing;
1230         break;
1231     case CursorType::EResize:
1232         m_value.valueID = CSSValueEResize;
1233         break;
1234     case CursorType::NResize:
1235         m_value.valueID = CSSValueNResize;
1236         break;
1237     case CursorType::NEResize:
1238         m_value.valueID = CSSValueNeResize;
1239         break;
1240     case CursorType::NWResize:
1241         m_value.valueID = CSSValueNwResize;
1242         break;
1243     case CursorType::SResize:
1244         m_value.valueID = CSSValueSResize;
1245         break;
1246     case CursorType::SEResize:
1247         m_value.valueID = CSSValueSeResize;
1248         break;
1249     case CursorType::SWResize:
1250         m_value.valueID = CSSValueSwResize;
1251         break;
1252     case CursorType::WResize:
1253         m_value.valueID = CSSValueWResize;
1254         break;
1255     case CursorType::EWResize:
1256         m_value.valueID = CSSValueEwResize;
1257         break;
1258     case CursorType::NSResize:
1259         m_value.valueID = CSSValueNsResize;
1260         break;
1261     case CursorType::NESWResize:
1262         m_value.valueID = CSSValueNeswResize;
1263         break;
1264     case CursorType::NWSEResize:
1265         m_value.valueID = CSSValueNwseResize;
1266         break;
1267     case CursorType::ColumnResize:
1268         m_value.valueID = CSSValueColResize;
1269         break;
1270     case CursorType::RowResize:
1271         m_value.valueID = CSSValueRowResize;
1272         break;
1273     case CursorType::AllScroll:
1274         m_value.valueID = CSSValueAllScroll;
1275         break;
1276     case CursorType::ZoomIn:
1277         m_value.valueID = CSSValueZoomIn;
1278         break;
1279     case CursorType::ZoomOut:
1280         m_value.valueID = CSSValueZoomOut;
1281         break;
1282     }
1283 }
1284
1285 template<> inline CSSPrimitiveValue::operator CursorType() const
1286 {
1287     ASSERT(isValueID());
1288     switch (m_value.valueID) {
1289     case CSSValueCopy:
1290         return CursorType::Copy;
1291     case CSSValueWebkitGrab:
1292         return CursorType::Grab;
1293     case CSSValueWebkitGrabbing:
1294         return CursorType::Grabbing;
1295     case CSSValueWebkitZoomIn:
1296         return CursorType::ZoomIn;
1297     case CSSValueWebkitZoomOut:
1298         return CursorType::ZoomOut;
1299     case CSSValueNone:
1300         return CursorType::None;
1301     default:
1302         return static_cast<CursorType>(m_value.valueID - CSSValueAuto);
1303     }
1304 }
1305
1306 #if ENABLE(CURSOR_VISIBILITY)
1307 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(CursorVisibility e)
1308     : CSSValue(PrimitiveClass)
1309 {
1310     m_primitiveUnitType = CSS_VALUE_ID;
1311     switch (e) {
1312     case CursorVisibility::Auto:
1313         m_value.valueID = CSSValueAuto;
1314         break;
1315     case CursorVisibility::AutoHide:
1316         m_value.valueID = CSSValueAutoHide;
1317         break;
1318     }
1319 }
1320
1321 template<> inline CSSPrimitiveValue::operator CursorVisibility() const
1322 {
1323     ASSERT(isValueID());
1324
1325     if (m_value.valueID == CSSValueAuto)
1326         return CursorVisibility::Auto;
1327     if (m_value.valueID == CSSValueAutoHide)
1328         return CursorVisibility::AutoHide;
1329
1330     ASSERT_NOT_REACHED();
1331     return CursorVisibility::Auto;
1332 }
1333 #endif
1334
1335 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(DisplayType e)
1336     : CSSValue(PrimitiveClass)
1337 {
1338     m_primitiveUnitType = CSS_VALUE_ID;
1339     switch (e) {
1340     case DisplayType::Inline:
1341         m_value.valueID = CSSValueInline;
1342         break;
1343     case DisplayType::Block:
1344         m_value.valueID = CSSValueBlock;
1345         break;
1346     case DisplayType::ListItem:
1347         m_value.valueID = CSSValueListItem;
1348         break;
1349     case DisplayType::Compact:
1350         m_value.valueID = CSSValueCompact;
1351         break;
1352     case DisplayType::InlineBlock:
1353         m_value.valueID = CSSValueInlineBlock;
1354         break;
1355     case DisplayType::Table:
1356         m_value.valueID = CSSValueTable;
1357         break;
1358     case DisplayType::InlineTable:
1359         m_value.valueID = CSSValueInlineTable;
1360         break;
1361     case DisplayType::TableRowGroup:
1362         m_value.valueID = CSSValueTableRowGroup;
1363         break;
1364     case DisplayType::TableHeaderGroup:
1365         m_value.valueID = CSSValueTableHeaderGroup;
1366         break;
1367     case DisplayType::TableFooterGroup:
1368         m_value.valueID = CSSValueTableFooterGroup;
1369         break;
1370     case DisplayType::TableRow:
1371         m_value.valueID = CSSValueTableRow;
1372         break;
1373     case DisplayType::TableColumnGroup:
1374         m_value.valueID = CSSValueTableColumnGroup;
1375         break;
1376     case DisplayType::TableColumn:
1377         m_value.valueID = CSSValueTableColumn;
1378         break;
1379     case DisplayType::TableCell:
1380         m_value.valueID = CSSValueTableCell;
1381         break;
1382     case DisplayType::TableCaption:
1383         m_value.valueID = CSSValueTableCaption;
1384         break;
1385     case DisplayType::Box:
1386         m_value.valueID = CSSValueWebkitBox;
1387         break;
1388     case DisplayType::InlineBox:
1389         m_value.valueID = CSSValueWebkitInlineBox;
1390         break;
1391     case DisplayType::Flex:
1392     case DisplayType::WebKitFlex:
1393         m_value.valueID = CSSValueFlex;
1394         break;
1395     case DisplayType::InlineFlex:
1396     case DisplayType::WebKitInlineFlex:
1397         m_value.valueID = CSSValueInlineFlex;
1398         break;
1399     case DisplayType::Grid:
1400         m_value.valueID = CSSValueGrid;
1401         break;
1402     case DisplayType::InlineGrid:
1403         m_value.valueID = CSSValueInlineGrid;
1404         break;
1405     case DisplayType::None:
1406         m_value.valueID = CSSValueNone;
1407         break;
1408     case DisplayType::Contents:
1409         m_value.valueID = CSSValueContents;
1410         break;
1411     }
1412 }
1413
1414 template<> inline CSSPrimitiveValue::operator DisplayType() const
1415 {
1416     ASSERT(isValueID());
1417
1418     if (m_value.valueID == CSSValueNone)
1419         return DisplayType::None;
1420
1421     DisplayType display = static_cast<DisplayType>(m_value.valueID - CSSValueInline);
1422     ASSERT(display >= DisplayType::Inline && display <= DisplayType::None);
1423     if (display == DisplayType::WebKitFlex)
1424         return DisplayType::Flex;
1425     if (display == DisplayType::WebKitInlineFlex)
1426         return DisplayType::InlineFlex;
1427     return display;
1428 }
1429
1430 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(EmptyCell e)
1431     : CSSValue(PrimitiveClass)
1432 {
1433     m_primitiveUnitType = CSS_VALUE_ID;
1434     switch (e) {
1435     case EmptyCell::Show:
1436         m_value.valueID = CSSValueShow;
1437         break;
1438     case EmptyCell::Hide:
1439         m_value.valueID = CSSValueHide;
1440         break;
1441     }
1442 }
1443
1444 template<> inline CSSPrimitiveValue::operator EmptyCell() const
1445 {
1446     ASSERT(isValueID());
1447
1448     switch (m_value.valueID) {
1449     case CSSValueShow:
1450         return EmptyCell::Show;
1451     case CSSValueHide:
1452         return EmptyCell::Hide;
1453     default:
1454         break;
1455     }
1456
1457     ASSERT_NOT_REACHED();
1458     return EmptyCell::Show;
1459 }
1460
1461 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FlexDirection e)
1462     : CSSValue(PrimitiveClass)
1463 {
1464     m_primitiveUnitType = CSS_VALUE_ID;
1465     switch (e) {
1466     case FlexDirection::Row:
1467         m_value.valueID = CSSValueRow;
1468         break;
1469     case FlexDirection::RowReverse:
1470         m_value.valueID = CSSValueRowReverse;
1471         break;
1472     case FlexDirection::Column:
1473         m_value.valueID = CSSValueColumn;
1474         break;
1475     case FlexDirection::ColumnReverse:
1476         m_value.valueID = CSSValueColumnReverse;
1477         break;
1478     }
1479 }
1480
1481 template<> inline CSSPrimitiveValue::operator FlexDirection() const
1482 {
1483     ASSERT(isValueID());
1484
1485     switch (m_value.valueID) {
1486     case CSSValueRow:
1487         return FlexDirection::Row;
1488     case CSSValueRowReverse:
1489         return FlexDirection::RowReverse;
1490     case CSSValueColumn:
1491         return FlexDirection::Column;
1492     case CSSValueColumnReverse:
1493         return FlexDirection::ColumnReverse;
1494     default:
1495         break;
1496     }
1497
1498     ASSERT_NOT_REACHED();
1499     return FlexDirection::Row;
1500 }
1501
1502 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(AlignContent e)
1503     : CSSValue(PrimitiveClass)
1504 {
1505     m_primitiveUnitType = CSS_VALUE_ID;
1506     switch (e) {
1507     case AlignContent::FlexStart:
1508         m_value.valueID = CSSValueFlexStart;
1509         break;
1510     case AlignContent::FlexEnd:
1511         m_value.valueID = CSSValueFlexEnd;
1512         break;
1513     case AlignContent::Center:
1514         m_value.valueID = CSSValueCenter;
1515         break;
1516     case AlignContent::SpaceBetween:
1517         m_value.valueID = CSSValueSpaceBetween;
1518         break;
1519     case AlignContent::SpaceAround:
1520         m_value.valueID = CSSValueSpaceAround;
1521         break;
1522     case AlignContent::Stretch:
1523         m_value.valueID = CSSValueStretch;
1524         break;
1525     }
1526 }
1527
1528 template<> inline CSSPrimitiveValue::operator AlignContent() const
1529 {
1530     ASSERT(isValueID());
1531
1532     switch (m_value.valueID) {
1533     case CSSValueFlexStart:
1534         return AlignContent::FlexStart;
1535     case CSSValueFlexEnd:
1536         return AlignContent::FlexEnd;
1537     case CSSValueCenter:
1538         return AlignContent::Center;
1539     case CSSValueSpaceBetween:
1540         return AlignContent::SpaceBetween;
1541     case CSSValueSpaceAround:
1542         return AlignContent::SpaceAround;
1543     case CSSValueStretch:
1544         return AlignContent::Stretch;
1545     default:
1546         break;
1547     }
1548
1549     ASSERT_NOT_REACHED();
1550     return AlignContent::Stretch;
1551 }
1552
1553 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FlexWrap e)
1554     : CSSValue(PrimitiveClass)
1555 {
1556     m_primitiveUnitType = CSS_VALUE_ID;
1557     switch (e) {
1558     case FlexWrap::NoWrap:
1559         m_value.valueID = CSSValueNowrap;
1560         break;
1561     case FlexWrap::Wrap:
1562         m_value.valueID = CSSValueWrap;
1563         break;
1564     case FlexWrap::Reverse:
1565         m_value.valueID = CSSValueWrapReverse;
1566         break;
1567     }
1568 }
1569
1570 template<> inline CSSPrimitiveValue::operator FlexWrap() const
1571 {
1572     ASSERT(isValueID());
1573
1574     switch (m_value.valueID) {
1575     case CSSValueNowrap:
1576         return FlexWrap::NoWrap;
1577     case CSSValueWrap:
1578         return FlexWrap::Wrap;
1579     case CSSValueWrapReverse:
1580         return FlexWrap::Reverse;
1581     default:
1582         break;
1583     }
1584
1585     ASSERT_NOT_REACHED();
1586     return FlexWrap::NoWrap;
1587 }
1588
1589 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Float e)
1590     : CSSValue(PrimitiveClass)
1591 {
1592     m_primitiveUnitType = CSS_VALUE_ID;
1593     switch (e) {
1594     case Float::No:
1595         m_value.valueID = CSSValueNone;
1596         break;
1597     case Float::Left:
1598         m_value.valueID = CSSValueLeft;
1599         break;
1600     case Float::Right:
1601         m_value.valueID = CSSValueRight;
1602         break;
1603     }
1604 }
1605
1606 template<> inline CSSPrimitiveValue::operator Float() const
1607 {
1608     ASSERT(isValueID());
1609
1610     switch (m_value.valueID) {
1611     case CSSValueLeft:
1612         return Float::Left;
1613     case CSSValueRight:
1614         return Float::Right;
1615     case CSSValueNone:
1616     case CSSValueCenter: // Non-standard CSS value.
1617         return Float::No;
1618     default:
1619         break;
1620     }
1621
1622     ASSERT_NOT_REACHED();
1623     return Float::No;
1624 }
1625
1626 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineBreak e)
1627     : CSSValue(PrimitiveClass)
1628 {
1629     m_primitiveUnitType = CSS_VALUE_ID;
1630     switch (e) {
1631     case LineBreak::Auto:
1632         m_value.valueID = CSSValueAuto;
1633         break;
1634     case LineBreak::Loose:
1635         m_value.valueID = CSSValueLoose;
1636         break;
1637     case LineBreak::Normal:
1638         m_value.valueID = CSSValueNormal;
1639         break;
1640     case LineBreak::Strict:
1641         m_value.valueID = CSSValueStrict;
1642         break;
1643     case LineBreak::AfterWhiteSpace:
1644         m_value.valueID = CSSValueAfterWhiteSpace;
1645         break;
1646     }
1647 }
1648
1649 template<> inline CSSPrimitiveValue::operator OptionSet<HangingPunctuation>() const
1650 {
1651     ASSERT(isValueID());
1652     
1653     switch (m_value.valueID) {
1654     case CSSValueNone:
1655         return OptionSet<HangingPunctuation> { };
1656     case CSSValueFirst:
1657         return HangingPunctuation::First;
1658     case CSSValueLast:
1659         return HangingPunctuation::Last;
1660     case CSSValueAllowEnd:
1661         return HangingPunctuation::AllowEnd;
1662     case CSSValueForceEnd:
1663         return HangingPunctuation::ForceEnd;
1664     default:
1665         break;
1666     }
1667     
1668     ASSERT_NOT_REACHED();
1669     return OptionSet<HangingPunctuation> { };
1670 }
1671
1672 template<> inline CSSPrimitiveValue::operator LineBreak() const
1673 {
1674     ASSERT(isValueID());
1675
1676     switch (m_value.valueID) {
1677     case CSSValueAuto:
1678         return LineBreak::Auto;
1679     case CSSValueLoose:
1680         return LineBreak::Loose;
1681     case CSSValueNormal:
1682         return LineBreak::Normal;
1683     case CSSValueStrict:
1684         return LineBreak::Strict;
1685     case CSSValueAfterWhiteSpace:
1686         return LineBreak::AfterWhiteSpace;
1687     default:
1688         break;
1689     }
1690
1691     ASSERT_NOT_REACHED();
1692     return LineBreak::Auto;
1693 }
1694
1695 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ListStylePosition e)
1696     : CSSValue(PrimitiveClass)
1697 {
1698     m_primitiveUnitType = CSS_VALUE_ID;
1699     switch (e) {
1700     case ListStylePosition::Outside:
1701         m_value.valueID = CSSValueOutside;
1702         break;
1703     case ListStylePosition::Inside:
1704         m_value.valueID = CSSValueInside;
1705         break;
1706     }
1707 }
1708
1709 template<> inline CSSPrimitiveValue::operator ListStylePosition() const
1710 {
1711     ASSERT(isValueID());
1712
1713     switch (m_value.valueID) {
1714     case CSSValueOutside:
1715         return ListStylePosition::Outside;
1716     case CSSValueInside:
1717         return ListStylePosition::Inside;
1718     default:
1719         break;
1720     }
1721
1722     ASSERT_NOT_REACHED();
1723     return ListStylePosition::Outside;
1724 }
1725
1726 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ListStyleType e)
1727     : CSSValue(PrimitiveClass)
1728 {
1729     m_primitiveUnitType = CSS_VALUE_ID;
1730     switch (e) {
1731     case ListStyleType::Afar:
1732         m_value.valueID = CSSValueAfar;
1733         break;
1734     case ListStyleType::Amharic:
1735         m_value.valueID = CSSValueAmharic;
1736         break;
1737     case ListStyleType::AmharicAbegede:
1738         m_value.valueID = CSSValueAmharicAbegede;
1739         break;
1740     case ListStyleType::ArabicIndic:
1741         m_value.valueID = CSSValueArabicIndic;
1742         break;
1743     case ListStyleType::Armenian:
1744         m_value.valueID = CSSValueArmenian;
1745         break;
1746     case ListStyleType::Asterisks:
1747         m_value.valueID = CSSValueAsterisks;
1748         break;
1749     case ListStyleType::Binary:
1750         m_value.valueID = CSSValueBinary;
1751         break;
1752     case ListStyleType::Bengali:
1753         m_value.valueID = CSSValueBengali;
1754         break;
1755     case ListStyleType::Cambodian:
1756         m_value.valueID = CSSValueCambodian;
1757         break;
1758     case ListStyleType::Circle:
1759         m_value.valueID = CSSValueCircle;
1760         break;
1761     case ListStyleType::CjkEarthlyBranch:
1762         m_value.valueID = CSSValueCjkEarthlyBranch;
1763         break;
1764     case ListStyleType::CjkHeavenlyStem:
1765         m_value.valueID = CSSValueCjkHeavenlyStem;
1766         break;
1767     case ListStyleType::CJKIdeographic:
1768         m_value.valueID = CSSValueCjkIdeographic;
1769         break;
1770     case ListStyleType::DecimalLeadingZero:
1771         m_value.valueID = CSSValueDecimalLeadingZero;
1772         break;
1773     case ListStyleType::Decimal:
1774         m_value.valueID = CSSValueDecimal;
1775         break;
1776     case ListStyleType::Devanagari:
1777         m_value.valueID = CSSValueDevanagari;
1778         break;
1779     case ListStyleType::Disc:
1780         m_value.valueID = CSSValueDisc;
1781         break;
1782     case ListStyleType::Ethiopic:
1783         m_value.valueID = CSSValueEthiopic;
1784         break;
1785     case ListStyleType::EthiopicAbegede:
1786         m_value.valueID = CSSValueEthiopicAbegede;
1787         break;
1788     case ListStyleType::EthiopicAbegedeAmEt:
1789         m_value.valueID = CSSValueEthiopicAbegedeAmEt;
1790         break;
1791     case ListStyleType::EthiopicAbegedeGez:
1792         m_value.valueID = CSSValueEthiopicAbegedeGez;
1793         break;
1794     case ListStyleType::EthiopicAbegedeTiEr:
1795         m_value.valueID = CSSValueEthiopicAbegedeTiEr;
1796         break;
1797     case ListStyleType::EthiopicAbegedeTiEt:
1798         m_value.valueID = CSSValueEthiopicAbegedeTiEt;
1799         break;
1800     case ListStyleType::EthiopicHalehameAaEr:
1801         m_value.valueID = CSSValueEthiopicHalehameAaEr;
1802         break;
1803     case ListStyleType::EthiopicHalehameAaEt:
1804         m_value.valueID = CSSValueEthiopicHalehameAaEt;
1805         break;
1806     case ListStyleType::EthiopicHalehameAmEt:
1807         m_value.valueID = CSSValueEthiopicHalehameAmEt;
1808         break;
1809     case ListStyleType::EthiopicHalehameGez:
1810         m_value.valueID = CSSValueEthiopicHalehameGez;
1811         break;
1812     case ListStyleType::EthiopicHalehameOmEt:
1813         m_value.valueID = CSSValueEthiopicHalehameOmEt;
1814         break;
1815     case ListStyleType::EthiopicHalehameSidEt:
1816         m_value.valueID = CSSValueEthiopicHalehameSidEt;
1817         break;
1818     case ListStyleType::EthiopicHalehameSoEt:
1819         m_value.valueID = CSSValueEthiopicHalehameSoEt;
1820         break;
1821     case ListStyleType::EthiopicHalehameTiEr:
1822         m_value.valueID = CSSValueEthiopicHalehameTiEr;
1823         break;
1824     case ListStyleType::EthiopicHalehameTiEt:
1825         m_value.valueID = CSSValueEthiopicHalehameTiEt;
1826         break;
1827     case ListStyleType::EthiopicHalehameTig:
1828         m_value.valueID = CSSValueEthiopicHalehameTig;
1829         break;
1830     case ListStyleType::Footnotes:
1831         m_value.valueID = CSSValueFootnotes;
1832         break;
1833     case ListStyleType::Georgian:
1834         m_value.valueID = CSSValueGeorgian;
1835         break;
1836     case ListStyleType::Gujarati:
1837         m_value.valueID = CSSValueGujarati;
1838         break;
1839     case ListStyleType::Gurmukhi:
1840         m_value.valueID = CSSValueGurmukhi;
1841         break;
1842     case ListStyleType::Hangul:
1843         m_value.valueID = CSSValueHangul;
1844         break;
1845     case ListStyleType::HangulConsonant:
1846         m_value.valueID = CSSValueHangulConsonant;
1847         break;
1848     case ListStyleType::Hebrew:
1849         m_value.valueID = CSSValueHebrew;
1850         break;
1851     case ListStyleType::Hiragana:
1852         m_value.valueID = CSSValueHiragana;
1853         break;
1854     case ListStyleType::HiraganaIroha:
1855         m_value.valueID = CSSValueHiraganaIroha;
1856         break;
1857     case ListStyleType::Kannada:
1858         m_value.valueID = CSSValueKannada;
1859         break;
1860     case ListStyleType::Katakana:
1861         m_value.valueID = CSSValueKatakana;
1862         break;
1863     case ListStyleType::KatakanaIroha:
1864         m_value.valueID = CSSValueKatakanaIroha;
1865         break;
1866     case ListStyleType::Khmer:
1867         m_value.valueID = CSSValueKhmer;
1868         break;
1869     case ListStyleType::Lao:
1870         m_value.valueID = CSSValueLao;
1871         break;
1872     case ListStyleType::LowerAlpha:
1873         m_value.valueID = CSSValueLowerAlpha;
1874         break;
1875     case ListStyleType::LowerArmenian:
1876         m_value.valueID = CSSValueLowerArmenian;
1877         break;
1878     case ListStyleType::LowerGreek:
1879         m_value.valueID = CSSValueLowerGreek;
1880         break;
1881     case ListStyleType::LowerHexadecimal:
1882         m_value.valueID = CSSValueLowerHexadecimal;
1883         break;
1884     case ListStyleType::LowerLatin:
1885         m_value.valueID = CSSValueLowerLatin;
1886         break;
1887     case ListStyleType::LowerNorwegian:
1888         m_value.valueID = CSSValueLowerNorwegian;
1889         break;
1890     case ListStyleType::LowerRoman:
1891         m_value.valueID = CSSValueLowerRoman;
1892         break;
1893     case ListStyleType::Malayalam:
1894         m_value.valueID = CSSValueMalayalam;
1895         break;
1896     case ListStyleType::Mongolian:
1897         m_value.valueID = CSSValueMongolian;
1898         break;
1899     case ListStyleType::Myanmar:
1900         m_value.valueID = CSSValueMyanmar;
1901         break;
1902     case ListStyleType::None:
1903         m_value.valueID = CSSValueNone;
1904         break;
1905     case ListStyleType::Octal:
1906         m_value.valueID = CSSValueOctal;
1907         break;
1908     case ListStyleType::Oriya:
1909         m_value.valueID = CSSValueOriya;
1910         break;
1911     case ListStyleType::Oromo:
1912         m_value.valueID = CSSValueOromo;
1913         break;
1914     case ListStyleType::Persian:
1915         m_value.valueID = CSSValuePersian;
1916         break;
1917     case ListStyleType::Sidama:
1918         m_value.valueID = CSSValueSidama;
1919         break;
1920     case ListStyleType::Somali:
1921         m_value.valueID = CSSValueSomali;
1922         break;
1923     case ListStyleType::Square:
1924         m_value.valueID = CSSValueSquare;
1925         break;
1926     case ListStyleType::Telugu:
1927         m_value.valueID = CSSValueTelugu;
1928         break;
1929     case ListStyleType::Thai:
1930         m_value.valueID = CSSValueThai;
1931         break;
1932     case ListStyleType::Tibetan:
1933         m_value.valueID = CSSValueTibetan;
1934         break;
1935     case ListStyleType::Tigre:
1936         m_value.valueID = CSSValueTigre;
1937         break;
1938     case ListStyleType::TigrinyaEr:
1939         m_value.valueID = CSSValueTigrinyaEr;
1940         break;
1941     case ListStyleType::TigrinyaErAbegede:
1942         m_value.valueID = CSSValueTigrinyaErAbegede;
1943         break;
1944     case ListStyleType::TigrinyaEt:
1945         m_value.valueID = CSSValueTigrinyaEt;
1946         break;
1947     case ListStyleType::TigrinyaEtAbegede:
1948         m_value.valueID = CSSValueTigrinyaEtAbegede;
1949         break;
1950     case ListStyleType::UpperAlpha:
1951         m_value.valueID = CSSValueUpperAlpha;
1952         break;
1953     case ListStyleType::UpperArmenian:
1954         m_value.valueID = CSSValueUpperArmenian;
1955         break;
1956     case ListStyleType::UpperGreek:
1957         m_value.valueID = CSSValueUpperGreek;
1958         break;
1959     case ListStyleType::UpperHexadecimal:
1960         m_value.valueID = CSSValueUpperHexadecimal;
1961         break;
1962     case ListStyleType::UpperLatin:
1963         m_value.valueID = CSSValueUpperLatin;
1964         break;
1965     case ListStyleType::UpperNorwegian:
1966         m_value.valueID = CSSValueUpperNorwegian;
1967         break;
1968     case ListStyleType::UpperRoman:
1969         m_value.valueID = CSSValueUpperRoman;
1970         break;
1971     case ListStyleType::Urdu:
1972         m_value.valueID = CSSValueUrdu;
1973         break;
1974     }
1975 }
1976
1977 template<> inline CSSPrimitiveValue::operator ListStyleType() const
1978 {
1979     ASSERT(isValueID());
1980
1981     switch (m_value.valueID) {
1982     case CSSValueNone:
1983         return ListStyleType::None;
1984     default:
1985         return static_cast<ListStyleType>(m_value.valueID - CSSValueDisc);
1986     }
1987 }
1988
1989 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(MarginCollapse e)
1990     : CSSValue(PrimitiveClass)
1991 {
1992     m_primitiveUnitType = CSS_VALUE_ID;
1993     switch (e) {
1994     case MarginCollapse::Collapse:
1995         m_value.valueID = CSSValueCollapse;
1996         break;
1997     case MarginCollapse::Separate:
1998         m_value.valueID = CSSValueSeparate;
1999         break;
2000     case MarginCollapse::Discard:
2001         m_value.valueID = CSSValueDiscard;
2002         break;
2003     }
2004 }
2005
2006 template<> inline CSSPrimitiveValue::operator MarginCollapse() const
2007 {
2008     ASSERT(isValueID());
2009
2010     switch (m_value.valueID) {
2011     case CSSValueCollapse:
2012         return MarginCollapse::Collapse;
2013     case CSSValueSeparate:
2014         return MarginCollapse::Separate;
2015     case CSSValueDiscard:
2016         return MarginCollapse::Discard;
2017     default:
2018         break;
2019     }
2020
2021     ASSERT_NOT_REACHED();
2022     return MarginCollapse::Collapse;
2023 }
2024
2025 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(MarqueeBehavior e)
2026     : CSSValue(PrimitiveClass)
2027 {
2028     m_primitiveUnitType = CSS_VALUE_ID;
2029     switch (e) {
2030     case MarqueeBehavior::None:
2031         m_value.valueID = CSSValueNone;
2032         break;
2033     case MarqueeBehavior::Scroll:
2034         m_value.valueID = CSSValueScroll;
2035         break;
2036     case MarqueeBehavior::Slide:
2037         m_value.valueID = CSSValueSlide;
2038         break;
2039     case MarqueeBehavior::Alternate:
2040         m_value.valueID = CSSValueAlternate;
2041         break;
2042     }
2043 }
2044
2045 template<> inline CSSPrimitiveValue::operator MarqueeBehavior() const
2046 {
2047     ASSERT(isValueID());
2048
2049     switch (m_value.valueID) {
2050     case CSSValueNone:
2051         return MarqueeBehavior::None;
2052     case CSSValueScroll:
2053         return MarqueeBehavior::Scroll;
2054     case CSSValueSlide:
2055         return MarqueeBehavior::Slide;
2056     case CSSValueAlternate:
2057         return MarqueeBehavior::Alternate;
2058     default:
2059         break;
2060     }
2061
2062     ASSERT_NOT_REACHED();
2063     return MarqueeBehavior::None;
2064 }
2065
2066 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(MarqueeDirection direction)
2067     : CSSValue(PrimitiveClass)
2068 {
2069     m_primitiveUnitType = CSS_VALUE_ID;
2070     switch (direction) {
2071     case MarqueeDirection::Forward:
2072         m_value.valueID = CSSValueForwards;
2073         break;
2074     case MarqueeDirection::Backward:
2075         m_value.valueID = CSSValueBackwards;
2076         break;
2077     case MarqueeDirection::Auto:
2078         m_value.valueID = CSSValueAuto;
2079         break;
2080     case MarqueeDirection::Up:
2081         m_value.valueID = CSSValueUp;
2082         break;
2083     case MarqueeDirection::Down:
2084         m_value.valueID = CSSValueDown;
2085         break;
2086     case MarqueeDirection::Left:
2087         m_value.valueID = CSSValueLeft;
2088         break;
2089     case MarqueeDirection::Right:
2090         m_value.valueID = CSSValueRight;
2091         break;
2092     }
2093 }
2094
2095 template<> inline CSSPrimitiveValue::operator MarqueeDirection() const
2096 {
2097     ASSERT(isValueID());
2098
2099     switch (m_value.valueID) {
2100     case CSSValueForwards:
2101         return MarqueeDirection::Forward;
2102     case CSSValueBackwards:
2103         return MarqueeDirection::Backward;
2104     case CSSValueAuto:
2105         return MarqueeDirection::Auto;
2106     case CSSValueAhead:
2107     case CSSValueUp: // We don't support vertical languages, so AHEAD just maps to UP.
2108         return MarqueeDirection::Up;
2109     case CSSValueReverse:
2110     case CSSValueDown: // REVERSE just maps to DOWN, since we don't do vertical text.
2111         return MarqueeDirection::Down;
2112     case CSSValueLeft:
2113         return MarqueeDirection::Left;
2114     case CSSValueRight:
2115         return MarqueeDirection::Right;
2116     default:
2117         break;
2118     }
2119
2120     ASSERT_NOT_REACHED();
2121     return MarqueeDirection::Auto;
2122 }
2123
2124 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(NBSPMode e)
2125     : CSSValue(PrimitiveClass)
2126 {
2127     m_primitiveUnitType = CSS_VALUE_ID;
2128     switch (e) {
2129     case NBSPMode::Normal:
2130         m_value.valueID = CSSValueNormal;
2131         break;
2132     case NBSPMode::Space:
2133         m_value.valueID = CSSValueSpace;
2134         break;
2135     }
2136 }
2137
2138 template<> inline CSSPrimitiveValue::operator NBSPMode() const
2139 {
2140     ASSERT(isValueID());
2141
2142     switch (m_value.valueID) {
2143     case CSSValueSpace:
2144         return NBSPMode::Space;
2145     case CSSValueNormal:
2146         return NBSPMode::Normal;
2147     default:
2148         break;
2149     }
2150
2151     ASSERT_NOT_REACHED();
2152     return NBSPMode::Normal;
2153 }
2154
2155 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Overflow e)
2156     : CSSValue(PrimitiveClass)
2157 {
2158     m_primitiveUnitType = CSS_VALUE_ID;
2159     switch (e) {
2160     case Overflow::Visible:
2161         m_value.valueID = CSSValueVisible;
2162         break;
2163     case Overflow::Hidden:
2164         m_value.valueID = CSSValueHidden;
2165         break;
2166     case Overflow::Scroll:
2167         m_value.valueID = CSSValueScroll;
2168         break;
2169     case Overflow::Auto:
2170         m_value.valueID = CSSValueAuto;
2171         break;
2172     case Overflow::PagedX:
2173         m_value.valueID = CSSValueWebkitPagedX;
2174         break;
2175     case Overflow::PagedY:
2176         m_value.valueID = CSSValueWebkitPagedY;
2177         break;
2178     }
2179 }
2180
2181 template<> inline CSSPrimitiveValue::operator Overflow() const
2182 {
2183     ASSERT(isValueID());
2184
2185     switch (m_value.valueID) {
2186     case CSSValueVisible:
2187         return Overflow::Visible;
2188     case CSSValueHidden:
2189         return Overflow::Hidden;
2190     case CSSValueScroll:
2191         return Overflow::Scroll;
2192     case CSSValueOverlay:
2193     case CSSValueAuto:
2194         return Overflow::Auto;
2195     case CSSValueWebkitPagedX:
2196         return Overflow::PagedX;
2197     case CSSValueWebkitPagedY:
2198         return Overflow::PagedY;
2199     default:
2200         break;
2201     }
2202
2203     ASSERT_NOT_REACHED();
2204     return Overflow::Visible;
2205 }
2206
2207 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BreakBetween e)
2208     : CSSValue(PrimitiveClass)
2209 {
2210     m_primitiveUnitType = CSS_VALUE_ID;
2211     switch (e) {
2212     case BreakBetween::Auto:
2213         m_value.valueID = CSSValueAuto;
2214         break;
2215     case BreakBetween::Avoid:
2216         m_value.valueID = CSSValueAvoid;
2217         break;
2218     case BreakBetween::AvoidColumn:
2219         m_value.valueID = CSSValueAvoidColumn;
2220         break;
2221     case BreakBetween::AvoidPage:
2222         m_value.valueID = CSSValueAvoidPage;
2223         break;
2224     case BreakBetween::Column:
2225         m_value.valueID = CSSValueColumn;
2226         break;
2227     case BreakBetween::Page:
2228         m_value.valueID = CSSValuePage;
2229         break;
2230     case BreakBetween::LeftPage:
2231         m_value.valueID = CSSValueLeft;
2232         break;
2233     case BreakBetween::RightPage:
2234         m_value.valueID = CSSValueRight;
2235         break;
2236     case BreakBetween::RectoPage:
2237         m_value.valueID = CSSValueRecto;
2238         break;
2239     case BreakBetween::VersoPage:
2240         m_value.valueID = CSSValueVerso;
2241         break;
2242     }
2243 }
2244
2245 template<> inline CSSPrimitiveValue::operator BreakBetween() const
2246 {
2247     ASSERT(isValueID());
2248
2249     switch (m_value.valueID) {
2250     case CSSValueAuto:
2251         return BreakBetween::Auto;
2252     case CSSValueAvoid:
2253         return BreakBetween::Avoid;
2254     case CSSValueAvoidColumn:
2255         return BreakBetween::AvoidColumn;
2256     case CSSValueAvoidPage:
2257         return BreakBetween::AvoidPage;
2258     case CSSValueColumn:
2259         return BreakBetween::Column;
2260     case CSSValuePage:
2261         return BreakBetween::Page;
2262     case CSSValueLeft:
2263         return BreakBetween::LeftPage;
2264     case CSSValueRight:
2265         return BreakBetween::RightPage;
2266     case CSSValueRecto:
2267         return BreakBetween::RectoPage;
2268     case CSSValueVerso:
2269         return BreakBetween::VersoPage;
2270     default:
2271         break;
2272     }
2273
2274     ASSERT_NOT_REACHED();
2275     return BreakBetween::Auto;
2276 }
2277
2278 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BreakInside e)
2279     : CSSValue(PrimitiveClass)
2280 {
2281     m_primitiveUnitType = CSS_VALUE_ID;
2282     switch (e) {
2283     case BreakInside::Auto:
2284         m_value.valueID = CSSValueAuto;
2285         break;
2286     case BreakInside::Avoid:
2287         m_value.valueID = CSSValueAvoid;
2288         break;
2289     case BreakInside::AvoidColumn:
2290         m_value.valueID = CSSValueAvoidColumn;
2291         break;
2292     case BreakInside::AvoidPage:
2293         m_value.valueID = CSSValueAvoidPage;
2294         break;
2295     }
2296 }
2297
2298 template<> inline CSSPrimitiveValue::operator BreakInside() const
2299 {
2300     ASSERT(isValueID());
2301     
2302     switch (m_value.valueID) {
2303     case CSSValueAuto:
2304         return BreakInside::Auto;
2305     case CSSValueAvoid:
2306         return BreakInside::Avoid;
2307     case CSSValueAvoidColumn:
2308         return BreakInside::AvoidColumn;
2309     case CSSValueAvoidPage:
2310         return BreakInside::AvoidPage;
2311     default:
2312         break;
2313     }
2314
2315     ASSERT_NOT_REACHED();
2316     return BreakInside::Auto;
2317 }
2318
2319 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(PositionType e)
2320     : CSSValue(PrimitiveClass)
2321 {
2322     m_primitiveUnitType = CSS_VALUE_ID;
2323     switch (e) {
2324     case PositionType::Static:
2325         m_value.valueID = CSSValueStatic;
2326         break;
2327     case PositionType::Relative:
2328         m_value.valueID = CSSValueRelative;
2329         break;
2330     case PositionType::Absolute:
2331         m_value.valueID = CSSValueAbsolute;
2332         break;
2333     case PositionType::Fixed:
2334         m_value.valueID = CSSValueFixed;
2335         break;
2336     case PositionType::Sticky:
2337         m_value.valueID = CSSValueWebkitSticky;
2338         break;
2339     }
2340 }
2341
2342 template<> inline CSSPrimitiveValue::operator PositionType() const
2343 {
2344     ASSERT(isValueID());
2345
2346     switch (m_value.valueID) {
2347     case CSSValueStatic:
2348         return PositionType::Static;
2349     case CSSValueRelative:
2350         return PositionType::Relative;
2351     case CSSValueAbsolute:
2352         return PositionType::Absolute;
2353     case CSSValueFixed:
2354         return PositionType::Fixed;
2355     case CSSValueWebkitSticky:
2356         return PositionType::Sticky;
2357     default:
2358         break;
2359     }
2360
2361     ASSERT_NOT_REACHED();
2362     return PositionType::Static;
2363 }
2364
2365 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Resize e)
2366     : CSSValue(PrimitiveClass)
2367 {
2368     m_primitiveUnitType = CSS_VALUE_ID;
2369     switch (e) {
2370     case Resize::Both:
2371         m_value.valueID = CSSValueBoth;
2372         break;
2373     case Resize::Horizontal:
2374         m_value.valueID = CSSValueHorizontal;
2375         break;
2376     case Resize::Vertical:
2377         m_value.valueID = CSSValueVertical;
2378         break;
2379     case Resize::None:
2380         m_value.valueID = CSSValueNone;
2381         break;
2382     }
2383 }
2384
2385 template<> inline CSSPrimitiveValue::operator Resize() const
2386 {
2387     ASSERT(isValueID());
2388
2389     switch (m_value.valueID) {
2390     case CSSValueBoth:
2391         return Resize::Both;
2392     case CSSValueHorizontal:
2393         return Resize::Horizontal;
2394     case CSSValueVertical:
2395         return Resize::Vertical;
2396     case CSSValueAuto:
2397         ASSERT_NOT_REACHED(); // Depends on settings, thus should be handled by the caller.
2398         return Resize::None;
2399     case CSSValueNone:
2400         return Resize::None;
2401     default:
2402         break;
2403     }
2404
2405     ASSERT_NOT_REACHED();
2406     return Resize::None;
2407 }
2408
2409 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TableLayoutType e)
2410     : CSSValue(PrimitiveClass)
2411 {
2412     m_primitiveUnitType = CSS_VALUE_ID;
2413     switch (e) {
2414     case TableLayoutType::Auto:
2415         m_value.valueID = CSSValueAuto;
2416         break;
2417     case TableLayoutType::Fixed:
2418         m_value.valueID = CSSValueFixed;
2419         break;
2420     }
2421 }
2422
2423 template<> inline CSSPrimitiveValue::operator TableLayoutType() const
2424 {
2425     ASSERT(isValueID());
2426
2427     switch (m_value.valueID) {
2428     case CSSValueFixed:
2429         return TableLayoutType::Fixed;
2430     case CSSValueAuto:
2431         return TableLayoutType::Auto;
2432     default:
2433         break;
2434     }
2435
2436     ASSERT_NOT_REACHED();
2437     return TableLayoutType::Auto;
2438 }
2439
2440 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextAlignMode e)
2441     : CSSValue(PrimitiveClass)
2442 {
2443     m_primitiveUnitType = CSS_VALUE_ID;
2444     switch (e) {
2445     case TextAlignMode::Start:
2446         m_value.valueID = CSSValueStart;
2447         break;
2448     case TextAlignMode::End:
2449         m_value.valueID = CSSValueEnd;
2450         break;
2451     case TextAlignMode::Left:
2452         m_value.valueID = CSSValueLeft;
2453         break;
2454     case TextAlignMode::Right:
2455         m_value.valueID = CSSValueRight;
2456         break;
2457     case TextAlignMode::Center:
2458         m_value.valueID = CSSValueCenter;
2459         break;
2460     case TextAlignMode::Justify:
2461         m_value.valueID = CSSValueJustify;
2462         break;
2463     case TextAlignMode::WebKitLeft:
2464         m_value.valueID = CSSValueWebkitLeft;
2465         break;
2466     case TextAlignMode::WebKitRight:
2467         m_value.valueID = CSSValueWebkitRight;
2468         break;
2469     case TextAlignMode::WebKitCenter:
2470         m_value.valueID = CSSValueWebkitCenter;
2471         break;
2472     }
2473 }
2474
2475 template<> inline CSSPrimitiveValue::operator TextAlignMode() const
2476 {
2477     ASSERT(isValueID());
2478
2479     switch (m_value.valueID) {
2480     case CSSValueWebkitAuto: // Legacy -webkit-auto. Eqiuvalent to start.
2481     case CSSValueStart:
2482         return TextAlignMode::Start;
2483     case CSSValueEnd:
2484         return TextAlignMode::End;
2485     default:
2486         return static_cast<TextAlignMode>(m_value.valueID - CSSValueLeft);
2487     }
2488 }
2489
2490 #if ENABLE(CSS3_TEXT)
2491 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextAlignLast e)
2492     : CSSValue(PrimitiveClass)
2493 {
2494     m_primitiveUnitType = CSS_VALUE_ID;
2495     switch (e) {
2496     case TextAlignLast::Start:
2497         m_value.valueID = CSSValueStart;
2498         break;
2499     case TextAlignLast::End:
2500         m_value.valueID = CSSValueEnd;
2501         break;
2502     case TextAlignLast::Left:
2503         m_value.valueID = CSSValueLeft;
2504         break;
2505     case TextAlignLast::Right:
2506         m_value.valueID = CSSValueRight;
2507         break;
2508     case TextAlignLast::Center:
2509         m_value.valueID = CSSValueCenter;
2510         break;
2511     case TextAlignLast::Justify:
2512         m_value.valueID = CSSValueJustify;
2513         break;
2514     case TextAlignLast::Auto:
2515         m_value.valueID = CSSValueAuto;
2516         break;
2517     }
2518 }
2519
2520 template<> inline CSSPrimitiveValue::operator TextAlignLast() const
2521 {
2522     ASSERT(isValueID());
2523
2524     switch (m_value.valueID) {
2525     case CSSValueAuto:
2526         return TextAlignLast::Auto;
2527     case CSSValueStart:
2528         return TextAlignLast::Start;
2529     case CSSValueEnd:
2530         return TextAlignLast::End;
2531     case CSSValueLeft:
2532         return TextAlignLast::Left;
2533     case CSSValueRight:
2534         return TextAlignLast::Right;
2535     case CSSValueCenter:
2536         return TextAlignLast::Center;
2537     case CSSValueJustify:
2538         return TextAlignLast::Justify;
2539     default:
2540         break;
2541     }
2542
2543     ASSERT_NOT_REACHED();
2544     return TextAlignLast::Auto;
2545 }
2546
2547 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextJustify e)
2548     : CSSValue(PrimitiveClass)
2549 {
2550     m_primitiveUnitType = CSS_VALUE_ID;
2551     switch (e) {
2552     case TextJustify::Auto:
2553         m_value.valueID = CSSValueAuto;
2554         break;
2555     case TextJustify::None:
2556         m_value.valueID = CSSValueNone;
2557         break;
2558     case TextJustify::InterWord:
2559         m_value.valueID = CSSValueInterWord;
2560         break;
2561     case TextJustify::Distribute:
2562         m_value.valueID = CSSValueDistribute;
2563         break;
2564     }
2565 }
2566
2567 template<> inline CSSPrimitiveValue::operator TextJustify() const
2568 {
2569     ASSERT(isValueID());
2570
2571     switch (m_value.valueID) {
2572     case CSSValueAuto:
2573         return TextJustify::Auto;
2574     case CSSValueNone:
2575         return TextJustify::None;
2576     case CSSValueInterWord:
2577         return TextJustify::InterWord;
2578     case CSSValueDistribute:
2579         return TextJustify::Distribute;
2580     default:
2581         break;
2582     }
2583
2584     ASSERT_NOT_REACHED();
2585     return TextJustify::Auto;
2586 }
2587 #endif // CSS3_TEXT
2588
2589 template<> inline CSSPrimitiveValue::operator OptionSet<TextDecoration>() const
2590 {
2591     ASSERT(isValueID());
2592
2593     switch (m_value.valueID) {
2594     case CSSValueNone:
2595         return OptionSet<TextDecoration> { };
2596     case CSSValueUnderline:
2597         return TextDecoration::Underline;
2598     case CSSValueOverline:
2599         return TextDecoration::Overline;
2600     case CSSValueLineThrough:
2601         return TextDecoration::LineThrough;
2602     case CSSValueBlink:
2603         return TextDecoration::Blink;
2604 #if ENABLE(LETTERPRESS)
2605     case CSSValueWebkitLetterpress:
2606         return TextDecoration::Letterpress;
2607 #endif
2608     default:
2609         break;
2610     }
2611
2612     ASSERT_NOT_REACHED();
2613     return OptionSet<TextDecoration> { };
2614 }
2615
2616 template<> inline CSSPrimitiveValue::operator TextDecorationStyle() const
2617 {
2618     ASSERT(isValueID());
2619
2620     switch (m_value.valueID) {
2621     case CSSValueSolid:
2622         return TextDecorationStyle::Solid;
2623     case CSSValueDouble:
2624         return TextDecorationStyle::Double;
2625     case CSSValueDotted:
2626         return TextDecorationStyle::Dotted;
2627     case CSSValueDashed:
2628         return TextDecorationStyle::Dashed;
2629     case CSSValueWavy:
2630         return TextDecorationStyle::Wavy;
2631     default:
2632         break;
2633     }
2634
2635     ASSERT_NOT_REACHED();
2636     return TextDecorationStyle::Solid;
2637 }
2638
2639 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextUnderlinePosition position)
2640     : CSSValue(PrimitiveClass)
2641 {
2642     m_primitiveUnitType = CSS_VALUE_ID;
2643     switch (position) {
2644     case TextUnderlinePosition::Auto:
2645         m_value.valueID = CSSValueAuto;
2646         break;
2647     case TextUnderlinePosition::Under:
2648         m_value.valueID = CSSValueUnder;
2649         break;
2650     case TextUnderlinePosition::FromFont:
2651         m_value.valueID = CSSValueFromFont;
2652         break;
2653     }
2654
2655     // FIXME: Implement support for 'under left' and 'under right' values.
2656 }
2657
2658 template<> inline CSSPrimitiveValue::operator TextUnderlinePosition() const
2659 {
2660     ASSERT(isValueID());
2661
2662     switch (m_value.valueID) {
2663     case CSSValueAuto:
2664         return TextUnderlinePosition::Auto;
2665     case CSSValueUnder:
2666         return TextUnderlinePosition::Under;
2667     case CSSValueFromFont:
2668         return TextUnderlinePosition::FromFont;
2669     default:
2670         break;
2671     }
2672
2673     // FIXME: Implement support for 'under left' and 'under right' values.
2674     ASSERT_NOT_REACHED();
2675     return TextUnderlinePosition::Auto;
2676 }
2677
2678 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextSecurity e)
2679     : CSSValue(PrimitiveClass)
2680 {
2681     m_primitiveUnitType = CSS_VALUE_ID;
2682     switch (e) {
2683     case TextSecurity::None:
2684         m_value.valueID = CSSValueNone;
2685         break;
2686     case TextSecurity::Disc:
2687         m_value.valueID = CSSValueDisc;
2688         break;
2689     case TextSecurity::Circle:
2690         m_value.valueID = CSSValueCircle;
2691         break;
2692     case TextSecurity::Square:
2693         m_value.valueID = CSSValueSquare;
2694         break;
2695     }
2696 }
2697
2698 template<> inline CSSPrimitiveValue::operator TextSecurity() const
2699 {
2700     ASSERT(isValueID());
2701
2702     switch (m_value.valueID) {
2703     case CSSValueNone:
2704         return TextSecurity::None;
2705     case CSSValueDisc:
2706         return TextSecurity::Disc;
2707     case CSSValueCircle:
2708         return TextSecurity::Circle;
2709     case CSSValueSquare:
2710         return TextSecurity::Square;
2711     default:
2712         break;
2713     }
2714
2715     ASSERT_NOT_REACHED();
2716     return TextSecurity::None;
2717 }
2718
2719 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextTransform e)
2720     : CSSValue(PrimitiveClass)
2721 {
2722     m_primitiveUnitType = CSS_VALUE_ID;
2723     switch (e) {
2724     case TextTransform::Capitalize:
2725         m_value.valueID = CSSValueCapitalize;
2726         break;
2727     case TextTransform::Uppercase:
2728         m_value.valueID = CSSValueUppercase;
2729         break;
2730     case TextTransform::Lowercase:
2731         m_value.valueID = CSSValueLowercase;
2732         break;
2733     case TextTransform::None:
2734         m_value.valueID = CSSValueNone;
2735         break;
2736     }
2737 }
2738
2739 template<> inline CSSPrimitiveValue::operator TextTransform() const
2740 {
2741     ASSERT(isValueID());
2742
2743     switch (m_value.valueID) {
2744     case CSSValueCapitalize:
2745         return TextTransform::Capitalize;
2746     case CSSValueUppercase:
2747         return TextTransform::Uppercase;
2748     case CSSValueLowercase:
2749         return TextTransform::Lowercase;
2750     case CSSValueNone:
2751         return TextTransform::None;
2752     default:
2753         break;
2754     }
2755
2756     ASSERT_NOT_REACHED();
2757     return TextTransform::None;
2758 }
2759
2760 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(EUnicodeBidi e)
2761     : CSSValue(PrimitiveClass)
2762 {
2763     m_primitiveUnitType = CSS_VALUE_ID;
2764     switch (e) {
2765     case UBNormal:
2766         m_value.valueID = CSSValueNormal;
2767         break;
2768     case Embed:
2769         m_value.valueID = CSSValueEmbed;
2770         break;
2771     case Override:
2772         m_value.valueID = CSSValueBidiOverride;
2773         break;
2774     case Isolate:
2775         m_value.valueID = CSSValueIsolate;
2776         break;
2777     case IsolateOverride:
2778         m_value.valueID = CSSValueIsolateOverride;
2779         break;
2780     case Plaintext:
2781         m_value.valueID = CSSValuePlaintext;
2782         break;
2783     }
2784 }
2785
2786 template<> inline CSSPrimitiveValue::operator EUnicodeBidi() const
2787 {
2788     ASSERT(isValueID());
2789
2790     switch (m_value.valueID) {
2791     case CSSValueNormal:
2792         return UBNormal;
2793     case CSSValueEmbed:
2794         return Embed;
2795     case CSSValueBidiOverride:
2796         return Override;
2797     case CSSValueIsolate:
2798     case CSSValueWebkitIsolate:
2799         return Isolate;
2800     case CSSValueIsolateOverride:
2801     case CSSValueWebkitIsolateOverride:
2802         return IsolateOverride;
2803     case CSSValuePlaintext:
2804     case CSSValueWebkitPlaintext:
2805         return Plaintext;
2806     default:
2807         break;
2808     }
2809
2810     ASSERT_NOT_REACHED();
2811     return UBNormal;
2812 }
2813
2814 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserDrag e)
2815     : CSSValue(PrimitiveClass)
2816 {
2817     m_primitiveUnitType = CSS_VALUE_ID;
2818     switch (e) {
2819     case UserDrag::Auto:
2820         m_value.valueID = CSSValueAuto;
2821         break;
2822     case UserDrag::None:
2823         m_value.valueID = CSSValueNone;
2824         break;
2825     case UserDrag::Element:
2826         m_value.valueID = CSSValueElement;
2827         break;
2828     default:
2829         break;
2830     }
2831 }
2832
2833 template<> inline CSSPrimitiveValue::operator UserDrag() const
2834 {
2835     ASSERT(isValueID());
2836
2837     switch (m_value.valueID) {
2838     case CSSValueAuto:
2839         return UserDrag::Auto;
2840     case CSSValueNone:
2841         return UserDrag::None;
2842     case CSSValueElement:
2843         return UserDrag::Element;
2844     default:
2845         break;
2846     }
2847
2848     ASSERT_NOT_REACHED();
2849     return UserDrag::Auto;
2850 }
2851
2852 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserModify e)
2853     : CSSValue(PrimitiveClass)
2854 {
2855     m_primitiveUnitType = CSS_VALUE_ID;
2856     switch (e) {
2857     case UserModify::ReadOnly:
2858         m_value.valueID = CSSValueReadOnly;
2859         break;
2860     case UserModify::ReadWrite:
2861         m_value.valueID = CSSValueReadWrite;
2862         break;
2863     case UserModify::ReadWritePlaintextOnly:
2864         m_value.valueID = CSSValueReadWritePlaintextOnly;
2865         break;
2866     }
2867 }
2868
2869 template<> inline CSSPrimitiveValue::operator UserModify() const
2870 {
2871     ASSERT(isValueID());
2872
2873     switch (m_value.valueID) {
2874     case CSSValueReadOnly:
2875         return UserModify::ReadOnly;
2876     case CSSValueReadWrite:
2877         return UserModify::ReadWrite;
2878     case CSSValueReadWritePlaintextOnly:
2879         return UserModify::ReadWritePlaintextOnly;
2880     default:
2881         break;
2882     }
2883
2884     ASSERT_NOT_REACHED();
2885     return UserModify::ReadOnly;
2886 }
2887
2888 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserSelect e)
2889     : CSSValue(PrimitiveClass)
2890 {
2891     m_primitiveUnitType = CSS_VALUE_ID;
2892     switch (e) {
2893     case UserSelect::None:
2894         m_value.valueID = CSSValueNone;
2895         break;
2896     case UserSelect::Text:
2897         m_value.valueID = CSSValueText;
2898         break;
2899     case UserSelect::All:
2900         m_value.valueID = CSSValueAll;
2901         break;
2902     }
2903 }
2904
2905 template<> inline CSSPrimitiveValue::operator UserSelect() const
2906 {
2907     ASSERT(isValueID());
2908
2909     switch (m_value.valueID) {
2910     case CSSValueAuto:
2911         return UserSelect::Text;
2912     case CSSValueNone:
2913         return UserSelect::None;
2914     case CSSValueText:
2915         return UserSelect::Text;
2916     case CSSValueAll:
2917         return UserSelect::All;
2918     default:
2919         break;
2920     }
2921
2922     ASSERT_NOT_REACHED();
2923     return UserSelect::Text;
2924 }
2925
2926 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(VerticalAlign a)
2927     : CSSValue(PrimitiveClass)
2928 {
2929     m_primitiveUnitType = CSS_VALUE_ID;
2930     switch (a) {
2931     case VerticalAlign::Top:
2932         m_value.valueID = CSSValueTop;
2933         break;
2934     case VerticalAlign::Bottom:
2935         m_value.valueID = CSSValueBottom;
2936         break;
2937     case VerticalAlign::Middle:
2938         m_value.valueID = CSSValueMiddle;
2939         break;
2940     case VerticalAlign::Baseline:
2941         m_value.valueID = CSSValueBaseline;
2942         break;
2943     case VerticalAlign::TextBottom:
2944         m_value.valueID = CSSValueTextBottom;
2945         break;
2946     case VerticalAlign::TextTop:
2947         m_value.valueID = CSSValueTextTop;
2948         break;
2949     case VerticalAlign::Sub:
2950         m_value.valueID = CSSValueSub;
2951         break;
2952     case VerticalAlign::Super:
2953         m_value.valueID = CSSValueSuper;
2954         break;
2955     case VerticalAlign::BaselineMiddle:
2956         m_value.valueID = CSSValueWebkitBaselineMiddle;
2957         break;
2958     case VerticalAlign::Length:
2959         m_value.valueID = CSSValueInvalid;
2960     }
2961 }
2962
2963 template<> inline CSSPrimitiveValue::operator VerticalAlign() const
2964 {
2965     ASSERT(isValueID());
2966
2967     switch (m_value.valueID) {
2968     case CSSValueTop:
2969         return VerticalAlign::Top;
2970     case CSSValueBottom:
2971         return VerticalAlign::Bottom;
2972     case CSSValueMiddle:
2973         return VerticalAlign::Middle;
2974     case CSSValueBaseline:
2975         return VerticalAlign::Baseline;
2976     case CSSValueTextBottom:
2977         return VerticalAlign::TextBottom;
2978     case CSSValueTextTop:
2979         return VerticalAlign::TextTop;
2980     case CSSValueSub:
2981         return VerticalAlign::Sub;
2982     case CSSValueSuper:
2983         return VerticalAlign::Super;
2984     case CSSValueWebkitBaselineMiddle:
2985         return VerticalAlign::BaselineMiddle;
2986     default:
2987         break;
2988     }
2989
2990     ASSERT_NOT_REACHED();
2991     return VerticalAlign::Top;
2992 }
2993
2994 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Visibility e)
2995     : CSSValue(PrimitiveClass)
2996 {
2997     m_primitiveUnitType = CSS_VALUE_ID;
2998     switch (e) {
2999     case Visibility::Visible:
3000         m_value.valueID = CSSValueVisible;
3001         break;
3002     case Visibility::Hidden:
3003         m_value.valueID = CSSValueHidden;
3004         break;
3005     case Visibility::Collapse:
3006         m_value.valueID = CSSValueCollapse;
3007         break;
3008     }
3009 }
3010
3011 template<> inline CSSPrimitiveValue::operator Visibility() const
3012 {
3013     ASSERT(isValueID());
3014
3015     switch (m_value.valueID) {
3016     case CSSValueHidden:
3017         return Visibility::Hidden;
3018     case CSSValueVisible:
3019         return Visibility::Visible;
3020     case CSSValueCollapse:
3021         return Visibility::Collapse;
3022     default:
3023         break;
3024     }
3025
3026     ASSERT_NOT_REACHED();
3027     return Visibility::Visible;
3028 }
3029
3030 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WhiteSpace e)
3031     : CSSValue(PrimitiveClass)
3032 {
3033     m_primitiveUnitType = CSS_VALUE_ID;
3034     switch (e) {
3035     case WhiteSpace::Normal:
3036         m_value.valueID = CSSValueNormal;
3037         break;
3038     case WhiteSpace::Pre:
3039         m_value.valueID = CSSValuePre;
3040         break;
3041     case WhiteSpace::PreWrap:
3042         m_value.valueID = CSSValuePreWrap;
3043         break;
3044     case WhiteSpace::PreLine:
3045         m_value.valueID = CSSValuePreLine;
3046         break;
3047     case WhiteSpace::NoWrap:
3048         m_value.valueID = CSSValueNowrap;
3049         break;
3050     case WhiteSpace::KHTMLNoWrap:
3051         m_value.valueID = CSSValueWebkitNowrap;
3052         break;
3053     }
3054 }
3055
3056 template<> inline CSSPrimitiveValue::operator WhiteSpace() const
3057 {
3058     ASSERT(isValueID());
3059
3060     switch (m_value.valueID) {
3061     case CSSValueWebkitNowrap:
3062         return WhiteSpace::KHTMLNoWrap;
3063     case CSSValueNowrap:
3064         return WhiteSpace::NoWrap;
3065     case CSSValuePre:
3066         return WhiteSpace::Pre;
3067     case CSSValuePreWrap:
3068         return WhiteSpace::PreWrap;
3069     case CSSValuePreLine:
3070         return WhiteSpace::PreLine;
3071     case CSSValueNormal:
3072         return WhiteSpace::Normal;
3073     default:
3074         break;
3075     }
3076
3077     ASSERT_NOT_REACHED();
3078     return WhiteSpace::Normal;
3079 }
3080
3081 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WordBreak e)
3082     : CSSValue(PrimitiveClass)
3083 {
3084     m_primitiveUnitType = CSS_VALUE_ID;
3085     switch (e) {
3086     case WordBreak::Normal:
3087         m_value.valueID = CSSValueNormal;
3088         break;
3089     case WordBreak::BreakAll:
3090         m_value.valueID = CSSValueBreakAll;
3091         break;
3092     case WordBreak::KeepAll:
3093         m_value.valueID = CSSValueKeepAll;
3094         break;
3095     case WordBreak::BreakWord:
3096         m_value.valueID = CSSValueBreakWord;
3097         break;
3098     }
3099 }
3100
3101 template<> inline CSSPrimitiveValue::operator WordBreak() const
3102 {
3103     ASSERT(isValueID());
3104
3105     switch (m_value.valueID) {
3106     case CSSValueBreakAll:
3107         return WordBreak::BreakAll;
3108     case CSSValueKeepAll:
3109         return WordBreak::KeepAll;
3110     case CSSValueBreakWord:
3111         return WordBreak::BreakWord;
3112     case CSSValueNormal:
3113         return WordBreak::Normal;
3114     default:
3115         break;
3116     }
3117
3118     ASSERT_NOT_REACHED();
3119     return WordBreak::Normal;
3120 }
3121
3122 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(OverflowWrap e)
3123     : CSSValue(PrimitiveClass)
3124 {
3125     m_primitiveUnitType = CSS_VALUE_ID;
3126     switch (e) {
3127     case OverflowWrap::Normal:
3128         m_value.valueID = CSSValueNormal;
3129         break;
3130     case OverflowWrap::Break:
3131         m_value.valueID = CSSValueBreakWord;
3132         break;
3133     }
3134 }
3135
3136 template<> inline CSSPrimitiveValue::operator OverflowWrap() const
3137 {
3138     ASSERT(isValueID());
3139
3140     switch (m_value.valueID) {
3141     case CSSValueBreakWord:
3142         return OverflowWrap::Break;
3143     case CSSValueNormal:
3144         return OverflowWrap::Normal;
3145     default:
3146         break;
3147     }
3148
3149     ASSERT_NOT_REACHED();
3150     return OverflowWrap::Normal;
3151 }
3152
3153 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextDirection e)
3154     : CSSValue(PrimitiveClass)
3155 {
3156     m_primitiveUnitType = CSS_VALUE_ID;
3157     switch (e) {
3158     case TextDirection::LTR:
3159         m_value.valueID = CSSValueLtr;
3160         break;
3161     case TextDirection::RTL:
3162         m_value.valueID = CSSValueRtl;
3163         break;
3164     }
3165 }
3166
3167 template<> inline CSSPrimitiveValue::operator TextDirection() const
3168 {
3169     ASSERT(isValueID());
3170
3171     switch (m_value.valueID) {
3172     case CSSValueLtr:
3173         return TextDirection::LTR;
3174     case CSSValueRtl:
3175         return TextDirection::RTL;
3176     default:
3177         break;
3178     }
3179
3180     ASSERT_NOT_REACHED();
3181     return TextDirection::LTR;
3182 }
3183
3184 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WritingMode e)
3185     : CSSValue(PrimitiveClass)
3186 {
3187     m_primitiveUnitType = CSS_VALUE_ID;
3188     switch (e) {
3189     case TopToBottomWritingMode:
3190         m_value.valueID = CSSValueHorizontalTb;
3191         break;
3192     case RightToLeftWritingMode:
3193         m_value.valueID = CSSValueVerticalRl;
3194         break;
3195     case LeftToRightWritingMode:
3196         m_value.valueID = CSSValueVerticalLr;
3197         break;
3198     case BottomToTopWritingMode:
3199         m_value.valueID = CSSValueHorizontalBt;
3200         break;
3201     }
3202 }
3203
3204 template<> inline CSSPrimitiveValue::operator WritingMode() const
3205 {
3206     ASSERT(isValueID());
3207
3208     switch (m_value.valueID) {
3209     case CSSValueHorizontalTb:
3210     case CSSValueLr:
3211     case CSSValueLrTb:
3212     case CSSValueRl:
3213     case CSSValueRlTb:
3214         return TopToBottomWritingMode;
3215     case CSSValueVerticalRl:
3216     case CSSValueTb:
3217     case CSSValueTbRl:
3218         return RightToLeftWritingMode;
3219     case CSSValueVerticalLr:
3220         return LeftToRightWritingMode;
3221     case CSSValueHorizontalBt:
3222         return BottomToTopWritingMode;
3223     default:
3224         break;
3225     }
3226
3227     ASSERT_NOT_REACHED();
3228     return TopToBottomWritingMode;
3229 }
3230
3231 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextCombine e)
3232     : CSSValue(PrimitiveClass)
3233 {
3234     m_primitiveUnitType = CSS_VALUE_ID;
3235     switch (e) {
3236     case TextCombine::None:
3237         m_value.valueID = CSSValueNone;
3238         break;
3239     case TextCombine::Horizontal:
3240         m_value.valueID = CSSValueHorizontal;
3241         break;
3242     }
3243 }
3244
3245 template<> inline CSSPrimitiveValue::operator TextCombine() const
3246 {
3247     ASSERT(isValueID());
3248
3249     switch (m_value.valueID) {
3250     case CSSValueNone:
3251         return TextCombine::None;
3252     case CSSValueHorizontal:
3253         return TextCombine::Horizontal;
3254     default:
3255         break;
3256     }
3257
3258     ASSERT_NOT_REACHED();
3259     return TextCombine::None;
3260 }
3261
3262 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(RubyPosition position)
3263     : CSSValue(PrimitiveClass)
3264 {
3265     m_primitiveUnitType = CSS_VALUE_ID;
3266     switch (position) {
3267     case RubyPosition::Before:
3268         m_value.valueID = CSSValueBefore;
3269         break;
3270     case RubyPosition::After:
3271         m_value.valueID = CSSValueAfter;
3272         break;
3273     case RubyPosition::InterCharacter:
3274         m_value.valueID = CSSValueInterCharacter;
3275         break;
3276     }
3277 }
3278
3279 template<> inline CSSPrimitiveValue::operator RubyPosition() const
3280 {
3281     ASSERT(isValueID());
3282
3283     switch (m_value.valueID) {
3284     case CSSValueBefore:
3285         return RubyPosition::Before;
3286     case CSSValueAfter:
3287         return RubyPosition::After;
3288     case CSSValueInterCharacter:
3289         return RubyPosition::InterCharacter;
3290     default:
3291         break;
3292     }
3293
3294     ASSERT_NOT_REACHED();
3295     return RubyPosition::Before;
3296 }
3297
3298 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextOverflow overflow)
3299     : CSSValue(PrimitiveClass)
3300 {
3301     m_primitiveUnitType = CSS_VALUE_ID;
3302     switch (overflow) {
3303     case TextOverflow::Clip:
3304         m_value.valueID = CSSValueClip;
3305         break;
3306     case TextOverflow::Ellipsis:
3307         m_value.valueID = CSSValueEllipsis;
3308         break;
3309     }
3310 }
3311
3312 template<> inline CSSPrimitiveValue::operator TextOverflow() const
3313 {
3314     ASSERT(isValueID());
3315
3316     switch (m_value.valueID) {
3317     case CSSValueClip:
3318         return TextOverflow::Clip;
3319     case CSSValueEllipsis:
3320         return TextOverflow::Ellipsis;
3321     default:
3322         break;
3323     }
3324
3325     ASSERT_NOT_REACHED();
3326     return TextOverflow::Clip;
3327 }
3328
3329 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextEmphasisFill fill)
3330     : CSSValue(PrimitiveClass)
3331 {
3332     m_primitiveUnitType = CSS_VALUE_ID;
3333     switch (fill) {
3334     case TextEmphasisFill::Filled:
3335         m_value.valueID = CSSValueFilled;
3336         break;
3337     case TextEmphasisFill::Open:
3338         m_value.valueID = CSSValueOpen;
3339         break;
3340     }
3341 }
3342
3343 template<> inline CSSPrimitiveValue::operator TextEmphasisFill() const
3344 {
3345     ASSERT(isValueID());
3346
3347     switch (m_value.valueID) {
3348     case CSSValueFilled:
3349         return TextEmphasisFill::Filled;
3350     case CSSValueOpen:
3351         return TextEmphasisFill::Open;
3352     default:
3353         break;
3354     }
3355
3356     ASSERT_NOT_REACHED();
3357     return TextEmphasisFill::Filled;
3358 }
3359
3360 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextEmphasisMark mark)
3361     : CSSValue(PrimitiveClass)
3362 {
3363     m_primitiveUnitType = CSS_VALUE_ID;
3364     switch (mark) {
3365     case TextEmphasisMark::Dot:
3366         m_value.valueID = CSSValueDot;
3367         break;
3368     case TextEmphasisMark::Circle:
3369         m_value.valueID = CSSValueCircle;
3370         break;
3371     case TextEmphasisMark::DoubleCircle:
3372         m_value.valueID = CSSValueDoubleCircle;
3373         break;
3374     case TextEmphasisMark::Triangle:
3375         m_value.valueID = CSSValueTriangle;
3376         break;
3377     case TextEmphasisMark::Sesame:
3378         m_value.valueID = CSSValueSesame;
3379         break;
3380     case TextEmphasisMark::None:
3381     case TextEmphasisMark::Auto:
3382     case TextEmphasisMark::Custom:
3383         ASSERT_NOT_REACHED();
3384         m_value.valueID = CSSValueNone;
3385         break;
3386     }
3387 }
3388
3389 template<> inline CSSPrimitiveValue::operator TextEmphasisMark() const
3390 {
3391     ASSERT(isValueID());
3392
3393     switch (m_value.valueID) {
3394     case CSSValueNone:
3395         return TextEmphasisMark::None;
3396     case CSSValueDot:
3397         return TextEmphasisMark::Dot;
3398     case CSSValueCircle:
3399         return TextEmphasisMark::Circle;
3400     case CSSValueDoubleCircle:
3401         return TextEmphasisMark::DoubleCircle;
3402     case CSSValueTriangle:
3403         return TextEmphasisMark::Triangle;
3404     case CSSValueSesame:
3405         return TextEmphasisMark::Sesame;
3406     default:
3407         break;
3408     }
3409
3410     ASSERT_NOT_REACHED();
3411     return TextEmphasisMark::None;
3412 }
3413
3414 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextOrientation e)
3415     : CSSValue(PrimitiveClass)
3416 {
3417     m_primitiveUnitType = CSS_VALUE_ID;
3418     switch (e) {
3419     case TextOrientation::Sideways:
3420         m_value.valueID = CSSValueSideways;
3421         break;
3422     case TextOrientation::Mixed:
3423         m_value.valueID = CSSValueMixed;
3424         break;
3425     case TextOrientation::Upright:
3426         m_value.valueID = CSSValueUpright;
3427         break;
3428     }
3429 }
3430
3431 template<> inline CSSPrimitiveValue::operator TextOrientation() const
3432 {
3433     ASSERT(isValueID());
3434
3435     switch (m_value.valueID) {
3436     case CSSValueSideways:
3437         return TextOrientation::Sideways;
3438     case CSSValueSidewaysRight:
3439         return TextOrientation::Sideways;
3440     case CSSValueVerticalRight:
3441         return TextOrientation::Mixed;
3442     case CSSValueMixed:
3443         return TextOrientation::Mixed;
3444     case CSSValueUpright:
3445         return TextOrientation::Upright;
3446     default:
3447         break;
3448     }
3449
3450     ASSERT_NOT_REACHED();
3451     return TextOrientation::Mixed;
3452 }
3453
3454 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(PointerEvents e)
3455     : CSSValue(PrimitiveClass)
3456 {
3457     m_primitiveUnitType = CSS_VALUE_ID;
3458     switch (e) {
3459     case PointerEvents::None:
3460         m_value.valueID = CSSValueNone;
3461         break;
3462     case PointerEvents::Stroke:
3463         m_value.valueID = CSSValueStroke;
3464         break;
3465     case PointerEvents::Fill:
3466         m_value.valueID = CSSValueFill;
3467         break;
3468     case PointerEvents::Painted:
3469         m_value.valueID = CSSValuePainted;
3470         break;
3471     case PointerEvents::Visible:
3472         m_value.valueID = CSSValueVisible;
3473         break;
3474     case PointerEvents::VisibleStroke:
3475         m_value.valueID = CSSValueVisibleStroke;
3476         break;
3477     case PointerEvents::VisibleFill:
3478         m_value.valueID = CSSValueVisibleFill;
3479         break;
3480     case PointerEvents::VisiblePainted:
3481         m_value.valueID = CSSValueVisiblePainted;
3482         break;
3483     case PointerEvents::Auto:
3484         m_value.valueID = CSSValueAuto;
3485         break;
3486     case PointerEvents::All:
3487         m_value.valueID = CSSValueAll;
3488         break;
3489     }
3490 }
3491
3492 template<> inline CSSPrimitiveValue::operator PointerEvents() const
3493 {
3494     ASSERT(isValueID());
3495
3496     switch (m_value.valueID) {
3497     case CSSValueAll:
3498         return PointerEvents::All;
3499     case CSSValueAuto:
3500         return PointerEvents::Auto;
3501     case CSSValueNone:
3502         return PointerEvents::None;
3503     case CSSValueVisiblePainted:
3504         return PointerEvents::VisiblePainted;
3505     case CSSValueVisibleFill:
3506         return PointerEvents::VisibleFill;
3507     case CSSValueVisibleStroke:
3508         return PointerEvents::VisibleStroke;
3509     case CSSValueVisible:
3510         return PointerEvents::Visible;
3511     case CSSValuePainted:
3512         return PointerEvents::Painted;
3513     case CSSValueFill:
3514         return PointerEvents::Fill;
3515     case CSSValueStroke:
3516         return PointerEvents::Stroke;
3517     default:
3518         break;
3519     }
3520
3521     ASSERT_NOT_REACHED();
3522     return PointerEvents::All;
3523 }
3524
3525 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Kerning kerning)
3526     : CSSValue(PrimitiveClass)
3527 {
3528     m_primitiveUnitType = CSS_VALUE_ID;
3529     switch (kerning) {
3530     case Kerning::Auto:
3531         m_value.valueID = CSSValueAuto;
3532         return;
3533     case Kerning::Normal:
3534         m_value.valueID = CSSValueNormal;
3535         return;
3536     case Kerning::NoShift:
3537         m_value.valueID = CSSValueNone;
3538         return;
3539     }
3540
3541     ASSERT_NOT_REACHED();
3542     m_value.valueID = CSSValueAuto;
3543 }
3544
3545 template<> inline CSSPrimitiveValue::operator Kerning() const
3546 {
3547     ASSERT(isValueID());
3548
3549     switch (m_value.valueID) {
3550     case CSSValueAuto:
3551         return Kerning::Auto;
3552     case CSSValueNormal:
3553         return Kerning::Normal;
3554     case CSSValueNone:
3555         return Kerning::NoShift;
3556     default:
3557         break;
3558     }
3559
3560     ASSERT_NOT_REACHED();
3561     return Kerning::Auto;
3562 }
3563
3564 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ObjectFit fit)
3565     : CSSValue(PrimitiveClass)
3566 {
3567     m_primitiveUnitType = CSS_VALUE_ID;
3568     switch (fit) {
3569     case ObjectFit::Fill:
3570         m_value.valueID = CSSValueFill;
3571         break;
3572     case ObjectFit::Contain:
3573         m_value.valueID = CSSValueContain;
3574         break;
3575     case ObjectFit::Cover:
3576         m_value.valueID = CSSValueCover;
3577         break;
3578     case ObjectFit::None:
3579         m_value.valueID = CSSValueNone;
3580         break;
3581     case ObjectFit::ScaleDown:
3582         m_value.valueID = CSSValueScaleDown;
3583         break;
3584     }
3585 }
3586
3587 template<> inline CSSPrimitiveValue::operator ObjectFit() const
3588 {
3589     ASSERT(isValueID());
3590
3591     switch (m_value.valueID) {
3592     case CSSValueFill:
3593         return ObjectFit::Fill;
3594     case CSSValueContain:
3595         return ObjectFit::Contain;
3596     case CSSValueCover:
3597         return ObjectFit::Cover;
3598     case CSSValueNone:
3599         return ObjectFit::None;
3600     case CSSValueScaleDown:
3601         return ObjectFit::ScaleDown;
3602     default:
3603         ASSERT_NOT_REACHED();
3604         return ObjectFit::Fill;
3605     }
3606 }
3607
3608 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FontSmoothingMode smoothing)
3609     : CSSValue(PrimitiveClass)
3610 {
3611     m_primitiveUnitType = CSS_VALUE_ID;
3612     switch (smoothing) {
3613     case FontSmoothingMode::AutoSmoothing:
3614         m_value.valueID = CSSValueAuto;
3615         return;
3616     case FontSmoothingMode::NoSmoothing:
3617         m_value.valueID = CSSValueNone;
3618         return;
3619     case FontSmoothingMode::Antialiased:
3620         m_value.valueID = CSSValueAntialiased;
3621         return;
3622     case FontSmoothingMode::SubpixelAntialiased:
3623         m_value.valueID = CSSValueSubpixelAntialiased;
3624         return;
3625     }
3626
3627     ASSERT_NOT_REACHED();
3628     m_value.valueID = CSSValueAuto;
3629 }
3630
3631 template<> inline CSSPrimitiveValue::operator FontSmoothingMode() const
3632 {
3633     ASSERT(isValueID());
3634
3635     switch (m_value.valueID) {
3636     case CSSValueAuto:
3637         return FontSmoothingMode::AutoSmoothing;
3638     case CSSValueNone:
3639         return FontSmoothingMode::NoSmoothing;
3640     case CSSValueAntialiased:
3641         return FontSmoothingMode::Antialiased;
3642     case CSSValueSubpixelAntialiased:
3643         return FontSmoothingMode::SubpixelAntialiased;
3644     default:
3645         break;
3646     }
3647
3648     ASSERT_NOT_REACHED();
3649     return FontSmoothingMode::AutoSmoothing;
3650 }
3651
3652 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FontSmallCaps smallCaps)
3653     : CSSValue(PrimitiveClass)
3654 {
3655     m_primitiveUnitType = CSS_VALUE_ID;
3656     switch (smallCaps) {
3657     case FontSmallCaps::Off:
3658         m_value.valueID = CSSValueNormal;
3659         return;
3660     case FontSmallCaps::On:
3661         m_value.valueID = CSSValueSmallCaps;
3662         return;
3663     }
3664
3665     ASSERT_NOT_REACHED();
3666     m_value.valueID = CSSValueNormal;
3667 }
3668
3669 template<> inline CSSPrimitiveValue::operator FontSmallCaps() const
3670 {
3671     ASSERT(isValueID());
3672
3673     switch (m_value.valueID) {
3674     case CSSValueSmallCaps:
3675         return FontSmallCaps::On;
3676     case CSSValueNormal:
3677         return FontSmallCaps::Off;
3678     default:
3679         break;
3680     }
3681     ASSERT_NOT_REACHED();
3682     return FontSmallCaps::Off;
3683 }
3684
3685 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextRenderingMode e)
3686     : CSSValue(PrimitiveClass)
3687 {
3688     m_primitiveUnitType = CSS_VALUE_ID;
3689     switch (e) {
3690     case TextRenderingMode::AutoTextRendering:
3691         m_value.valueID = CSSValueAuto;
3692         break;
3693     case TextRenderingMode::OptimizeSpeed:
3694         m_value.valueID = CSSValueOptimizeSpeed;
3695         break;
3696     case TextRenderingMode::OptimizeLegibility:
3697         m_value.valueID = CSSValueOptimizeLegibility;
3698         break;
3699     case TextRenderingMode::GeometricPrecision:
3700         m_value.valueID = CSSValueGeometricPrecision;
3701         break;
3702     }
3703 }
3704
3705 template<> inline CSSPrimitiveValue::operator TextRenderingMode() const
3706 {
3707     ASSERT(isValueID());
3708
3709     switch (m_value.valueID) {
3710     case CSSValueAuto:
3711         return TextRenderingMode::AutoTextRendering;
3712     case CSSValueOptimizeSpeed:
3713         return TextRenderingMode::OptimizeSpeed;
3714     case CSSValueOptimizeLegibility:
3715         return TextRenderingMode::OptimizeLegibility;
3716     case CSSValueGeometricPrecision:
3717         return TextRenderingMode::GeometricPrecision;
3718     default:
3719         break;
3720     }
3721
3722     ASSERT_NOT_REACHED();
3723     return TextRenderingMode::AutoTextRendering;
3724 }
3725
3726 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Hyphens hyphens)
3727     : CSSValue(PrimitiveClass)
3728 {
3729     m_primitiveUnitType = CSS_VALUE_ID;
3730     switch (hyphens) {
3731     case Hyphens::None:
3732         m_value.valueID = CSSValueNone;
3733         break;
3734     case Hyphens::Manual:
3735         m_value.valueID = CSSValueManual;
3736         break;
3737     case Hyphens::Auto:
3738         m_value.valueID = CSSValueAuto;
3739         break;
3740     }
3741 }
3742
3743 template<> inline CSSPrimitiveValue::operator Hyphens() const
3744 {
3745     ASSERT(isValueID());
3746
3747     switch (m_value.valueID) {
3748     case CSSValueNone:
3749         return Hyphens::None;
3750     case CSSValueManual:
3751         return Hyphens::Manual;
3752     case CSSValueAuto:
3753         return Hyphens::Auto;
3754     default:
3755         break;
3756     }
3757
3758     ASSERT_NOT_REACHED();
3759     return Hyphens::Auto;
3760 }
3761
3762 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineSnap gridSnap)
3763     : CSSValue(PrimitiveClass)
3764 {
3765     m_primitiveUnitType = CSS_VALUE_ID;
3766     switch (gridSnap) {
3767     case LineSnap::None:
3768         m_value.valueID = CSSValueNone;
3769         break;
3770     case LineSnap::Baseline:
3771         m_value.valueID = CSSValueBaseline;
3772         break;
3773     case LineSnap::Contain:
3774         m_value.valueID = CSSValueContain;
3775         break;
3776     }
3777 }
3778
3779 template<> inline CSSPrimitiveValue::operator LineSnap() const
3780 {
3781     ASSERT(isValueID());
3782
3783     switch (m_value.valueID) {
3784     case CSSValueNone:
3785         return LineSnap::None;
3786     case CSSValueBaseline:
3787         return LineSnap::Baseline;
3788     case CSSValueContain:
3789         return LineSnap::Contain;
3790     default:
3791         break;
3792     }
3793
3794     ASSERT_NOT_REACHED();
3795     return LineSnap::None;
3796 }
3797
3798 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineAlign lineAlign)
3799     : CSSValue(PrimitiveClass)
3800 {
3801     m_primitiveUnitType = CSS_VALUE_ID;
3802     switch (lineAlign) {
3803     case LineAlign::None:
3804         m_value.valueID = CSSValueNone;
3805         break;
3806     case LineAlign::Edges:
3807         m_value.valueID = CSSValueEdges;
3808         break;
3809     }
3810 }
3811
3812 template<> inline CSSPrimitiveValue::operator LineAlign() const
3813 {
3814     ASSERT(isValueID());
3815
3816     switch (m_value.valueID) {
3817     case CSSValueNone:
3818         return LineAlign::None;
3819     case CSSValueEdges:
3820         return LineAlign::Edges;
3821     default:
3822         break;
3823     }
3824
3825     ASSERT_NOT_REACHED();
3826     return LineAlign::None;
3827 }
3828
3829 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(SpeakAs e)
3830     : CSSValue(PrimitiveClass)
3831 {
3832     m_primitiveUnitType = CSS_VALUE_ID;
3833     switch (e) {
3834     case SpeakAs::Normal:
3835         m_value.valueID = CSSValueNormal;
3836         break;
3837     case SpeakAs::SpellOut:
3838         m_value.valueID = CSSValueSpellOut;
3839         break;
3840     case SpeakAs::Digits:
3841         m_value.valueID = CSSValueDigits;
3842         break;
3843     case SpeakAs::LiteralPunctuation:
3844         m_value.valueID = CSSValueLiteralPunctuation;
3845         break;
3846     case SpeakAs::NoPunctuation:
3847         m_value.valueID = CSSValueNoPunctuation;
3848         break;
3849     }
3850 }
3851
3852 template<> inline CSSPrimitiveValue::operator Order() const
3853 {
3854     ASSERT(isValueID());
3855
3856     switch (m_value.valueID) {
3857     case CSSValueLogical:
3858         return Order::Logical;
3859     case CSSValueVisual:
3860         return Order::Visual;
3861     default:
3862         break;
3863     }
3864
3865     ASSERT_NOT_REACHED();
3866     return Order::Logical;
3867 }
3868
3869 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Order e)
3870     : CSSValue(PrimitiveClass)
3871 {
3872     m_primitiveUnitType = CSS_VALUE_ID;
3873     switch (e) {
3874     case Order::Logical:
3875         m_value.valueID = CSSValueLogical;
3876         break;
3877     case Order::Visual:
3878         m_value.valueID = CSSValueVisual;
3879         break;
3880     }
3881 }
3882
3883 template<> inline CSSPrimitiveValue::operator OptionSet<SpeakAs>() const
3884 {
3885     ASSERT(isValueID());
3886
3887     switch (m_value.valueID) {
3888     case CSSValueNormal:
3889         return OptionSet<SpeakAs> { };
3890     case CSSValueSpellOut:
3891         return SpeakAs::SpellOut;
3892     case CSSValueDigits:
3893         return SpeakAs::Digits;
3894     case CSSValueLiteralPunctuation:
3895         return SpeakAs::LiteralPunctuation;
3896     case CSSValueNoPunctuation:
3897         return SpeakAs::NoPunctuation;
3898     default:
3899         break;
3900     }
3901
3902     ASSERT_NOT_REACHED();
3903     return OptionSet<SpeakAs> { };
3904 }
3905
3906 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BlendMode blendMode)
3907     : CSSValue(PrimitiveClass)
3908 {
3909     m_primitiveUnitType = CSS_VALUE_ID;
3910     switch (blendMode) {
3911     case BlendMode::Normal:
3912         m_value.valueID = CSSValueNormal;
3913         break;
3914     case BlendMode::Multiply:
3915         m_value.valueID = CSSValueMultiply;
3916         break;
3917     case BlendMode::Screen:
3918         m_value.valueID = CSSValueScreen;
3919         break;
3920     case BlendMode::Overlay:
3921         m_value.valueID = CSSValueOverlay;
3922         break;
3923     case BlendMode::Darken:
3924         m_value.valueID = CSSValueDarken;
3925         break;
3926     case BlendMode::Lighten:
3927         m_value.valueID = CSSValueLighten;
3928         break;
3929     case BlendMode::ColorDodge:
3930         m_value.valueID = CSSValueColorDodge;
3931         break;
3932     case BlendMode::ColorBurn:
3933         m_value.valueID = CSSValueColorBurn;
3934         break;
3935     case BlendMode::HardLight:
3936         m_value.valueID = CSSValueHardLight;
3937         break;
3938     case BlendMode::SoftLight:
3939         m_value.valueID = CSSValueSoftLight;
3940         break;
3941     case BlendMode::Difference:
3942         m_value.valueID = CSSValueDifference;
3943         break;
3944     case BlendMode::Exclusion:
3945         m_value.valueID = CSSValueExclusion;
3946         break;
3947     case BlendMode::Hue:
3948         m_value.valueID = CSSValueHue;
3949         break;
3950     case BlendMode::Saturation:
3951         m_value.valueID = CSSValueSaturation;
3952         break;
3953     case BlendMode::Color:
3954         m_value.valueID = CSSValueColor;
3955         break;
3956     case BlendMode::Luminosity:
3957         m_value.valueID = CSSValueLuminosity;
3958         break;
3959     case BlendMode::PlusDarker:
3960         m_value.valueID = CSSValuePlusDarker;
3961         break;
3962     case BlendMode::PlusLighter:
3963         m_value.valueID = CSSValuePlusLighter;
3964         break;
3965     }
3966 }
3967
3968 template<> inline CSSPrimitiveValue::operator BlendMode() const
3969 {
3970     ASSERT(isValueID());
3971
3972     switch (m_value.valueID) {
3973     case CSSValueNormal:
3974         return BlendMode::Normal;
3975     case CSSValueMultiply:
3976         return BlendMode::Multiply;
3977     case CSSValueScreen:
3978         return BlendMode::Screen;
3979     case CSSValueOverlay:
3980         return BlendMode::Overlay;
3981     case CSSValueDarken:
3982         return BlendMode::Darken;
3983     case CSSValueLighten:
3984         return BlendMode::Lighten;
3985     case CSSValueColorDodge:
3986         return BlendMode::ColorDodge;
3987     case CSSValueColorBurn:
3988         return BlendMode::ColorBurn;
3989     case CSSValueHardLight:
3990         return BlendMode::HardLight;
3991     case CSSValueSoftLight:
3992         return BlendMode::SoftLight;
3993     case CSSValueDifference:
3994         return BlendMode::Difference;
3995     case CSSValueExclusion:
3996         return BlendMode::Exclusion;
3997     case CSSValueHue:
3998         return BlendMode::Hue;
3999     case CSSValueSaturation:
4000         return BlendMode::Saturation;
4001     case CSSValueColor:
4002         return BlendMode::Color;
4003     case CSSValueLuminosity:
4004         return BlendMode::Luminosity;
4005     case CSSValuePlusDarker:
4006         return BlendMode::PlusDarker;
4007     case CSSValuePlusLighter:
4008         return BlendMode::PlusLighter;
4009     default:
4010         break;
4011     }
4012
4013     ASSERT_NOT_REACHED();
4014     return BlendMode::Normal;
4015 }
4016
4017 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Isolation isolation)
4018     : CSSValue(PrimitiveClass)
4019 {
4020     m_primitiveUnitType = CSS_VALUE_ID;
4021     switch (isolation) {
4022     case Isolation::Auto:
4023         m_value.valueID = CSSValueAuto;
4024         break;
4025     case Isolation::Isolate:
4026         m_value.valueID = CSSValueIsolate;
4027         break;
4028     default:
4029         ASSERT_NOT_REACHED();
4030     }
4031 }
4032
4033 template<> inline CSSPrimitiveValue::operator Isolation() const
4034 {
4035     ASSERT(isValueID());
4036     switch (m_value.valueID) {
4037     case CSSValueAuto:
4038         return Isolation::Auto;
4039     case CSSValueIsolate:
4040         return Isolation::Isolate;
4041     default:
4042         break;
4043     }
4044
4045     ASSERT_NOT_REACHED();
4046     return Isolation::Auto;
4047 }
4048
4049 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineCap e)
4050     : CSSValue(PrimitiveClass)
4051 {
4052     m_primitiveUnitType = CSS_VALUE_ID;
4053     switch (e) {
4054     case ButtCap:
4055         m_value.valueID = CSSValueButt;
4056         break;
4057     case RoundCap:
4058         m_value.valueID = CSSValueRound;
4059         break;
4060     case SquareCap:
4061         m_value.valueID = CSSValueSquare;
4062         break;
4063     }
4064 }
4065
4066 template<> inline CSSPrimitiveValue::operator LineCap() const
4067 {
4068     ASSERT(isValueID());
4069
4070     switch (m_value.valueID) {
4071     case CSSValueButt:
4072         return ButtCap;
4073     case CSSValueRound:
4074         return RoundCap;
4075     case CSSValueSquare:
4076         return SquareCap;
4077     default:
4078         break;
4079     }
4080
4081     ASSERT_NOT_REACHED();
4082     return ButtCap;
4083 }
4084
4085 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineJoin e)
4086     : CSSValue(PrimitiveClass)
4087 {
4088     m_primitiveUnitType = CSS_VALUE_ID;
4089     switch (e) {
4090     case MiterJoin:
4091         m_value.valueID = CSSValueMiter;
4092         break;
4093     case RoundJoin:
4094         m_value.valueID = CSSValueRound;
4095         break;
4096     case BevelJoin:
4097         m_value.valueID = CSSValueBevel;
4098         break;
4099     }
4100 }
4101
4102 template<> inline CSSPrimitiveValue::operator LineJoin() const
4103 {
4104     ASSERT(isValueID());
4105
4106     switch (m_value.valueID) {
4107     case CSSValueMiter:
4108         return MiterJoin;
4109     case CSSValueRound:
4110         return RoundJoin;
4111     case CSSValueBevel:
4112         return BevelJoin;
4113     default:
4114         break;
4115     }
4116
4117     ASSERT_NOT_REACHED();
4118     return MiterJoin;
4119 }
4120
4121 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WindRule e)
4122     : CSSValue(PrimitiveClass)
4123 {
4124     m_primitiveUnitType = CSS_VALUE_ID;
4125     switch (e) {
4126     case WindRule::NonZero:
4127         m_value.valueID = CSSValueNonzero;
4128         break;
4129     case WindRule::EvenOdd:
4130         m_value.valueID = CSSValueEvenodd;
4131         break;
4132     }
4133 }
4134
4135 template<> inline CSSPrimitiveValue::operator WindRule() const
4136 {
4137     ASSERT(isValueID());
4138
4139     switch (m_value.valueID) {
4140     case CSSValueNonzero:
4141         return WindRule::NonZero;
4142     case CSSValueEvenodd:
4143         return WindRule::EvenOdd;
4144     default:
4145         break;
4146     }
4147
4148     ASSERT_NOT_REACHED();
4149     return WindRule::NonZero;
4150 }
4151
4152
4153 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(AlignmentBaseline e)
4154     : CSSValue(PrimitiveClass)
4155 {
4156     m_primitiveUnitType = CSS_VALUE_ID;
4157     switch (e) {
4158     case AlignmentBaseline::Auto:
4159         m_value.valueID = CSSValueAuto;
4160         break;
4161     case AlignmentBaseline::Baseline:
4162         m_value.valueID = CSSValueBaseline;
4163         break;
4164     case AlignmentBaseline::BeforeEdge:
4165         m_value.valueID = CSSValueBeforeEdge;
4166         break;
4167     case AlignmentBaseline::TextBeforeEdge:
4168         m_value.valueID = CSSValueTextBeforeEdge;
4169         break;
4170     case AlignmentBaseline::Middle:
4171         m_value.valueID = CSSValueMiddle;
4172         break;
4173     case AlignmentBaseline::Central:
4174         m_value.valueID = CSSValueCentral;
4175         break;
4176     case AlignmentBaseline::AfterEdge:
4177         m_value.valueID = CSSValueAfterEdge;
4178         break;
4179     case AlignmentBaseline::TextAfterEdge:
4180         m_value.valueID = CSSValueTextAfterEdge;
4181         break;
4182     case AlignmentBaseline::Ideographic:
4183         m_value.valueID = CSSValueIdeographic;
4184         break;
4185     case AlignmentBaseline::Alphabetic:
4186         m_value.valueID = CSSValueAlphabetic;
4187         break;
4188     case AlignmentBaseline::Hanging:
4189         m_value.valueID = CSSValueHanging;
4190         break;
4191     case AlignmentBaseline::Mathematical:
4192         m_value.valueID = CSSValueMathematical;
4193         break;
4194     }
4195 }
4196
4197 template<> inline CSSPrimitiveValue::operator AlignmentBaseline() const
4198 {
4199     ASSERT(isValueID());
4200
4201     switch (m_value.valueID) {
4202     case CSSValueAuto:
4203         return AlignmentBaseline::Auto;
4204     case CSSValueBaseline: