Make "overflow: overlay" a synonym for "overflow: auto"
[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(OptionSet<TextUnderlinePosition> e)
2640     : CSSValue(PrimitiveClass)
2641 {
2642     m_primitiveUnitType = CSS_VALUE_ID;
2643     switch (static_cast<TextUnderlinePosition>(e.toRaw())) {
2644     case TextUnderlinePosition::Auto:
2645         m_value.valueID = CSSValueAuto;
2646         break;
2647     case TextUnderlinePosition::Alphabetic:
2648         m_value.valueID = CSSValueAlphabetic;
2649         break;
2650     case TextUnderlinePosition::Under:
2651         m_value.valueID = CSSValueUnder;
2652         break;
2653     }
2654
2655     // FIXME: Implement support for 'under left' and 'under right' values.
2656 }
2657
2658 template<> inline CSSPrimitiveValue::operator OptionSet<TextUnderlinePosition>() const
2659 {
2660     ASSERT(isValueID());
2661
2662     switch (m_value.valueID) {
2663     case CSSValueAuto:
2664         return TextUnderlinePosition::Auto;
2665     case CSSValueAlphabetic:
2666         return TextUnderlinePosition::Alphabetic;
2667     case CSSValueUnder:
2668         return TextUnderlinePosition::Under;
2669     default:
2670         break;
2671     }
2672
2673     // FIXME: Implement support for 'under left' and 'under right' values.
2674
2675     ASSERT_NOT_REACHED();
2676     return TextUnderlinePosition::Auto;
2677 }
2678
2679 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextSecurity e)
2680     : CSSValue(PrimitiveClass)
2681 {
2682     m_primitiveUnitType = CSS_VALUE_ID;
2683     switch (e) {
2684     case TextSecurity::None:
2685         m_value.valueID = CSSValueNone;
2686         break;
2687     case TextSecurity::Disc:
2688         m_value.valueID = CSSValueDisc;
2689         break;
2690     case TextSecurity::Circle:
2691         m_value.valueID = CSSValueCircle;
2692         break;
2693     case TextSecurity::Square:
2694         m_value.valueID = CSSValueSquare;
2695         break;
2696     }
2697 }
2698
2699 template<> inline CSSPrimitiveValue::operator TextSecurity() const
2700 {
2701     ASSERT(isValueID());
2702
2703     switch (m_value.valueID) {
2704     case CSSValueNone:
2705         return TextSecurity::None;
2706     case CSSValueDisc:
2707         return TextSecurity::Disc;
2708     case CSSValueCircle:
2709         return TextSecurity::Circle;
2710     case CSSValueSquare:
2711         return TextSecurity::Square;
2712     default:
2713         break;
2714     }
2715
2716     ASSERT_NOT_REACHED();
2717     return TextSecurity::None;
2718 }
2719
2720 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextTransform e)
2721     : CSSValue(PrimitiveClass)
2722 {
2723     m_primitiveUnitType = CSS_VALUE_ID;
2724     switch (e) {
2725     case TextTransform::Capitalize:
2726         m_value.valueID = CSSValueCapitalize;
2727         break;
2728     case TextTransform::Uppercase:
2729         m_value.valueID = CSSValueUppercase;
2730         break;
2731     case TextTransform::Lowercase:
2732         m_value.valueID = CSSValueLowercase;
2733         break;
2734     case TextTransform::None:
2735         m_value.valueID = CSSValueNone;
2736         break;
2737     }
2738 }
2739
2740 template<> inline CSSPrimitiveValue::operator TextTransform() const
2741 {
2742     ASSERT(isValueID());
2743
2744     switch (m_value.valueID) {
2745     case CSSValueCapitalize:
2746         return TextTransform::Capitalize;
2747     case CSSValueUppercase:
2748         return TextTransform::Uppercase;
2749     case CSSValueLowercase:
2750         return TextTransform::Lowercase;
2751     case CSSValueNone:
2752         return TextTransform::None;
2753     default:
2754         break;
2755     }
2756
2757     ASSERT_NOT_REACHED();
2758     return TextTransform::None;
2759 }
2760
2761 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(EUnicodeBidi e)
2762     : CSSValue(PrimitiveClass)
2763 {
2764     m_primitiveUnitType = CSS_VALUE_ID;
2765     switch (e) {
2766     case UBNormal:
2767         m_value.valueID = CSSValueNormal;
2768         break;
2769     case Embed:
2770         m_value.valueID = CSSValueEmbed;
2771         break;
2772     case Override:
2773         m_value.valueID = CSSValueBidiOverride;
2774         break;
2775     case Isolate:
2776         m_value.valueID = CSSValueIsolate;
2777         break;
2778     case IsolateOverride:
2779         m_value.valueID = CSSValueIsolateOverride;
2780         break;
2781     case Plaintext:
2782         m_value.valueID = CSSValuePlaintext;
2783         break;
2784     }
2785 }
2786
2787 template<> inline CSSPrimitiveValue::operator EUnicodeBidi() const
2788 {
2789     ASSERT(isValueID());
2790
2791     switch (m_value.valueID) {
2792     case CSSValueNormal:
2793         return UBNormal;
2794     case CSSValueEmbed:
2795         return Embed;
2796     case CSSValueBidiOverride:
2797         return Override;
2798     case CSSValueIsolate:
2799     case CSSValueWebkitIsolate:
2800         return Isolate;
2801     case CSSValueIsolateOverride:
2802     case CSSValueWebkitIsolateOverride:
2803         return IsolateOverride;
2804     case CSSValuePlaintext:
2805     case CSSValueWebkitPlaintext:
2806         return Plaintext;
2807     default:
2808         break;
2809     }
2810
2811     ASSERT_NOT_REACHED();
2812     return UBNormal;
2813 }
2814
2815 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserDrag e)
2816     : CSSValue(PrimitiveClass)
2817 {
2818     m_primitiveUnitType = CSS_VALUE_ID;
2819     switch (e) {
2820     case UserDrag::Auto:
2821         m_value.valueID = CSSValueAuto;
2822         break;
2823     case UserDrag::None:
2824         m_value.valueID = CSSValueNone;
2825         break;
2826     case UserDrag::Element:
2827         m_value.valueID = CSSValueElement;
2828         break;
2829     default:
2830         break;
2831     }
2832 }
2833
2834 template<> inline CSSPrimitiveValue::operator UserDrag() const
2835 {
2836     ASSERT(isValueID());
2837
2838     switch (m_value.valueID) {
2839     case CSSValueAuto:
2840         return UserDrag::Auto;
2841     case CSSValueNone:
2842         return UserDrag::None;
2843     case CSSValueElement:
2844         return UserDrag::Element;
2845     default:
2846         break;
2847     }
2848
2849     ASSERT_NOT_REACHED();
2850     return UserDrag::Auto;
2851 }
2852
2853 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserModify e)
2854     : CSSValue(PrimitiveClass)
2855 {
2856     m_primitiveUnitType = CSS_VALUE_ID;
2857     switch (e) {
2858     case UserModify::ReadOnly:
2859         m_value.valueID = CSSValueReadOnly;
2860         break;
2861     case UserModify::ReadWrite:
2862         m_value.valueID = CSSValueReadWrite;
2863         break;
2864     case UserModify::ReadWritePlaintextOnly:
2865         m_value.valueID = CSSValueReadWritePlaintextOnly;
2866         break;
2867     }
2868 }
2869
2870 template<> inline CSSPrimitiveValue::operator UserModify() const
2871 {
2872     ASSERT(isValueID());
2873
2874     switch (m_value.valueID) {
2875     case CSSValueReadOnly:
2876         return UserModify::ReadOnly;
2877     case CSSValueReadWrite:
2878         return UserModify::ReadWrite;
2879     case CSSValueReadWritePlaintextOnly:
2880         return UserModify::ReadWritePlaintextOnly;
2881     default:
2882         break;
2883     }
2884
2885     ASSERT_NOT_REACHED();
2886     return UserModify::ReadOnly;
2887 }
2888
2889 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(UserSelect e)
2890     : CSSValue(PrimitiveClass)
2891 {
2892     m_primitiveUnitType = CSS_VALUE_ID;
2893     switch (e) {
2894     case UserSelect::None:
2895         m_value.valueID = CSSValueNone;
2896         break;
2897     case UserSelect::Text:
2898         m_value.valueID = CSSValueText;
2899         break;
2900     case UserSelect::All:
2901         m_value.valueID = CSSValueAll;
2902         break;
2903     }
2904 }
2905
2906 template<> inline CSSPrimitiveValue::operator UserSelect() const
2907 {
2908     ASSERT(isValueID());
2909
2910     switch (m_value.valueID) {
2911     case CSSValueAuto:
2912         return UserSelect::Text;
2913     case CSSValueNone:
2914         return UserSelect::None;
2915     case CSSValueText:
2916         return UserSelect::Text;
2917     case CSSValueAll:
2918         return UserSelect::All;
2919     default:
2920         break;
2921     }
2922
2923     ASSERT_NOT_REACHED();
2924     return UserSelect::Text;
2925 }
2926
2927 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(VerticalAlign a)
2928     : CSSValue(PrimitiveClass)
2929 {
2930     m_primitiveUnitType = CSS_VALUE_ID;
2931     switch (a) {
2932     case VerticalAlign::Top:
2933         m_value.valueID = CSSValueTop;
2934         break;
2935     case VerticalAlign::Bottom:
2936         m_value.valueID = CSSValueBottom;
2937         break;
2938     case VerticalAlign::Middle:
2939         m_value.valueID = CSSValueMiddle;
2940         break;
2941     case VerticalAlign::Baseline:
2942         m_value.valueID = CSSValueBaseline;
2943         break;
2944     case VerticalAlign::TextBottom:
2945         m_value.valueID = CSSValueTextBottom;
2946         break;
2947     case VerticalAlign::TextTop:
2948         m_value.valueID = CSSValueTextTop;
2949         break;
2950     case VerticalAlign::Sub:
2951         m_value.valueID = CSSValueSub;
2952         break;
2953     case VerticalAlign::Super:
2954         m_value.valueID = CSSValueSuper;
2955         break;
2956     case VerticalAlign::BaselineMiddle:
2957         m_value.valueID = CSSValueWebkitBaselineMiddle;
2958         break;
2959     case VerticalAlign::Length:
2960         m_value.valueID = CSSValueInvalid;
2961     }
2962 }
2963
2964 template<> inline CSSPrimitiveValue::operator VerticalAlign() const
2965 {
2966     ASSERT(isValueID());
2967
2968     switch (m_value.valueID) {
2969     case CSSValueTop:
2970         return VerticalAlign::Top;
2971     case CSSValueBottom:
2972         return VerticalAlign::Bottom;
2973     case CSSValueMiddle:
2974         return VerticalAlign::Middle;
2975     case CSSValueBaseline:
2976         return VerticalAlign::Baseline;
2977     case CSSValueTextBottom:
2978         return VerticalAlign::TextBottom;
2979     case CSSValueTextTop:
2980         return VerticalAlign::TextTop;
2981     case CSSValueSub:
2982         return VerticalAlign::Sub;
2983     case CSSValueSuper:
2984         return VerticalAlign::Super;
2985     case CSSValueWebkitBaselineMiddle:
2986         return VerticalAlign::BaselineMiddle;
2987     default:
2988         break;
2989     }
2990
2991     ASSERT_NOT_REACHED();
2992     return VerticalAlign::Top;
2993 }
2994
2995 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Visibility e)
2996     : CSSValue(PrimitiveClass)
2997 {
2998     m_primitiveUnitType = CSS_VALUE_ID;
2999     switch (e) {
3000     case Visibility::Visible:
3001         m_value.valueID = CSSValueVisible;
3002         break;
3003     case Visibility::Hidden:
3004         m_value.valueID = CSSValueHidden;
3005         break;
3006     case Visibility::Collapse:
3007         m_value.valueID = CSSValueCollapse;
3008         break;
3009     }
3010 }
3011
3012 template<> inline CSSPrimitiveValue::operator Visibility() const
3013 {
3014     ASSERT(isValueID());
3015
3016     switch (m_value.valueID) {
3017     case CSSValueHidden:
3018         return Visibility::Hidden;
3019     case CSSValueVisible:
3020         return Visibility::Visible;
3021     case CSSValueCollapse:
3022         return Visibility::Collapse;
3023     default:
3024         break;
3025     }
3026
3027     ASSERT_NOT_REACHED();
3028     return Visibility::Visible;
3029 }
3030
3031 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WhiteSpace e)
3032     : CSSValue(PrimitiveClass)
3033 {
3034     m_primitiveUnitType = CSS_VALUE_ID;
3035     switch (e) {
3036     case WhiteSpace::Normal:
3037         m_value.valueID = CSSValueNormal;
3038         break;
3039     case WhiteSpace::Pre:
3040         m_value.valueID = CSSValuePre;
3041         break;
3042     case WhiteSpace::PreWrap:
3043         m_value.valueID = CSSValuePreWrap;
3044         break;
3045     case WhiteSpace::PreLine:
3046         m_value.valueID = CSSValuePreLine;
3047         break;
3048     case WhiteSpace::NoWrap:
3049         m_value.valueID = CSSValueNowrap;
3050         break;
3051     case WhiteSpace::KHTMLNoWrap:
3052         m_value.valueID = CSSValueWebkitNowrap;
3053         break;
3054     }
3055 }
3056
3057 template<> inline CSSPrimitiveValue::operator WhiteSpace() const
3058 {
3059     ASSERT(isValueID());
3060
3061     switch (m_value.valueID) {
3062     case CSSValueWebkitNowrap:
3063         return WhiteSpace::KHTMLNoWrap;
3064     case CSSValueNowrap:
3065         return WhiteSpace::NoWrap;
3066     case CSSValuePre:
3067         return WhiteSpace::Pre;
3068     case CSSValuePreWrap:
3069         return WhiteSpace::PreWrap;
3070     case CSSValuePreLine:
3071         return WhiteSpace::PreLine;
3072     case CSSValueNormal:
3073         return WhiteSpace::Normal;
3074     default:
3075         break;
3076     }
3077
3078     ASSERT_NOT_REACHED();
3079     return WhiteSpace::Normal;
3080 }
3081
3082 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WordBreak e)
3083     : CSSValue(PrimitiveClass)
3084 {
3085     m_primitiveUnitType = CSS_VALUE_ID;
3086     switch (e) {
3087     case WordBreak::Normal:
3088         m_value.valueID = CSSValueNormal;
3089         break;
3090     case WordBreak::BreakAll:
3091         m_value.valueID = CSSValueBreakAll;
3092         break;
3093     case WordBreak::KeepAll:
3094         m_value.valueID = CSSValueKeepAll;
3095         break;
3096     case WordBreak::BreakWord:
3097         m_value.valueID = CSSValueBreakWord;
3098         break;
3099     }
3100 }
3101
3102 template<> inline CSSPrimitiveValue::operator WordBreak() const
3103 {
3104     ASSERT(isValueID());
3105
3106     switch (m_value.valueID) {
3107     case CSSValueBreakAll:
3108         return WordBreak::BreakAll;
3109     case CSSValueKeepAll:
3110         return WordBreak::KeepAll;
3111     case CSSValueBreakWord:
3112         return WordBreak::BreakWord;
3113     case CSSValueNormal:
3114         return WordBreak::Normal;
3115     default:
3116         break;
3117     }
3118
3119     ASSERT_NOT_REACHED();
3120     return WordBreak::Normal;
3121 }
3122
3123 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(OverflowWrap e)
3124     : CSSValue(PrimitiveClass)
3125 {
3126     m_primitiveUnitType = CSS_VALUE_ID;
3127     switch (e) {
3128     case OverflowWrap::Normal:
3129         m_value.valueID = CSSValueNormal;
3130         break;
3131     case OverflowWrap::Break:
3132         m_value.valueID = CSSValueBreakWord;
3133         break;
3134     }
3135 }
3136
3137 template<> inline CSSPrimitiveValue::operator OverflowWrap() const
3138 {
3139     ASSERT(isValueID());
3140
3141     switch (m_value.valueID) {
3142     case CSSValueBreakWord:
3143         return OverflowWrap::Break;
3144     case CSSValueNormal:
3145         return OverflowWrap::Normal;
3146     default:
3147         break;
3148     }
3149
3150     ASSERT_NOT_REACHED();
3151     return OverflowWrap::Normal;
3152 }
3153
3154 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextDirection e)
3155     : CSSValue(PrimitiveClass)
3156 {
3157     m_primitiveUnitType = CSS_VALUE_ID;
3158     switch (e) {
3159     case TextDirection::LTR:
3160         m_value.valueID = CSSValueLtr;
3161         break;
3162     case TextDirection::RTL:
3163         m_value.valueID = CSSValueRtl;
3164         break;
3165     }
3166 }
3167
3168 template<> inline CSSPrimitiveValue::operator TextDirection() const
3169 {
3170     ASSERT(isValueID());
3171
3172     switch (m_value.valueID) {
3173     case CSSValueLtr:
3174         return TextDirection::LTR;
3175     case CSSValueRtl:
3176         return TextDirection::RTL;
3177     default:
3178         break;
3179     }
3180
3181     ASSERT_NOT_REACHED();
3182     return TextDirection::LTR;
3183 }
3184
3185 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WritingMode e)
3186     : CSSValue(PrimitiveClass)
3187 {
3188     m_primitiveUnitType = CSS_VALUE_ID;
3189     switch (e) {
3190     case TopToBottomWritingMode:
3191         m_value.valueID = CSSValueHorizontalTb;
3192         break;
3193     case RightToLeftWritingMode:
3194         m_value.valueID = CSSValueVerticalRl;
3195         break;
3196     case LeftToRightWritingMode:
3197         m_value.valueID = CSSValueVerticalLr;
3198         break;
3199     case BottomToTopWritingMode:
3200         m_value.valueID = CSSValueHorizontalBt;
3201         break;
3202     }
3203 }
3204
3205 template<> inline CSSPrimitiveValue::operator WritingMode() const
3206 {
3207     ASSERT(isValueID());
3208
3209     switch (m_value.valueID) {
3210     case CSSValueHorizontalTb:
3211     case CSSValueLr:
3212     case CSSValueLrTb:
3213     case CSSValueRl:
3214     case CSSValueRlTb:
3215         return TopToBottomWritingMode;
3216     case CSSValueVerticalRl:
3217     case CSSValueTb:
3218     case CSSValueTbRl:
3219         return RightToLeftWritingMode;
3220     case CSSValueVerticalLr:
3221         return LeftToRightWritingMode;
3222     case CSSValueHorizontalBt:
3223         return BottomToTopWritingMode;
3224     default:
3225         break;
3226     }
3227
3228     ASSERT_NOT_REACHED();
3229     return TopToBottomWritingMode;
3230 }
3231
3232 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextCombine e)
3233     : CSSValue(PrimitiveClass)
3234 {
3235     m_primitiveUnitType = CSS_VALUE_ID;
3236     switch (e) {
3237     case TextCombine::None:
3238         m_value.valueID = CSSValueNone;
3239         break;
3240     case TextCombine::Horizontal:
3241         m_value.valueID = CSSValueHorizontal;
3242         break;
3243     }
3244 }
3245
3246 template<> inline CSSPrimitiveValue::operator TextCombine() const
3247 {
3248     ASSERT(isValueID());
3249
3250     switch (m_value.valueID) {
3251     case CSSValueNone:
3252         return TextCombine::None;
3253     case CSSValueHorizontal:
3254         return TextCombine::Horizontal;
3255     default:
3256         break;
3257     }
3258
3259     ASSERT_NOT_REACHED();
3260     return TextCombine::None;
3261 }
3262
3263 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(RubyPosition position)
3264     : CSSValue(PrimitiveClass)
3265 {
3266     m_primitiveUnitType = CSS_VALUE_ID;
3267     switch (position) {
3268     case RubyPosition::Before:
3269         m_value.valueID = CSSValueBefore;
3270         break;
3271     case RubyPosition::After:
3272         m_value.valueID = CSSValueAfter;
3273         break;
3274     case RubyPosition::InterCharacter:
3275         m_value.valueID = CSSValueInterCharacter;
3276         break;
3277     }
3278 }
3279
3280 template<> inline CSSPrimitiveValue::operator RubyPosition() const
3281 {
3282     ASSERT(isValueID());
3283
3284     switch (m_value.valueID) {
3285     case CSSValueBefore:
3286         return RubyPosition::Before;
3287     case CSSValueAfter:
3288         return RubyPosition::After;
3289     case CSSValueInterCharacter:
3290         return RubyPosition::InterCharacter;
3291     default:
3292         break;
3293     }
3294
3295     ASSERT_NOT_REACHED();
3296     return RubyPosition::Before;
3297 }
3298
3299 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextOverflow overflow)
3300     : CSSValue(PrimitiveClass)
3301 {
3302     m_primitiveUnitType = CSS_VALUE_ID;
3303     switch (overflow) {
3304     case TextOverflow::Clip:
3305         m_value.valueID = CSSValueClip;
3306         break;
3307     case TextOverflow::Ellipsis:
3308         m_value.valueID = CSSValueEllipsis;
3309         break;
3310     }
3311 }
3312
3313 template<> inline CSSPrimitiveValue::operator TextOverflow() const
3314 {
3315     ASSERT(isValueID());
3316
3317     switch (m_value.valueID) {
3318     case CSSValueClip:
3319         return TextOverflow::Clip;
3320     case CSSValueEllipsis:
3321         return TextOverflow::Ellipsis;
3322     default:
3323         break;
3324     }
3325
3326     ASSERT_NOT_REACHED();
3327     return TextOverflow::Clip;
3328 }
3329
3330 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextEmphasisFill fill)
3331     : CSSValue(PrimitiveClass)
3332 {
3333     m_primitiveUnitType = CSS_VALUE_ID;
3334     switch (fill) {
3335     case TextEmphasisFill::Filled:
3336         m_value.valueID = CSSValueFilled;
3337         break;
3338     case TextEmphasisFill::Open:
3339         m_value.valueID = CSSValueOpen;
3340         break;
3341     }
3342 }
3343
3344 template<> inline CSSPrimitiveValue::operator TextEmphasisFill() const
3345 {
3346     ASSERT(isValueID());
3347
3348     switch (m_value.valueID) {
3349     case CSSValueFilled:
3350         return TextEmphasisFill::Filled;
3351     case CSSValueOpen:
3352         return TextEmphasisFill::Open;
3353     default:
3354         break;
3355     }
3356
3357     ASSERT_NOT_REACHED();
3358     return TextEmphasisFill::Filled;
3359 }
3360
3361 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextEmphasisMark mark)
3362     : CSSValue(PrimitiveClass)
3363 {
3364     m_primitiveUnitType = CSS_VALUE_ID;
3365     switch (mark) {
3366     case TextEmphasisMark::Dot:
3367         m_value.valueID = CSSValueDot;
3368         break;
3369     case TextEmphasisMark::Circle:
3370         m_value.valueID = CSSValueCircle;
3371         break;
3372     case TextEmphasisMark::DoubleCircle:
3373         m_value.valueID = CSSValueDoubleCircle;
3374         break;
3375     case TextEmphasisMark::Triangle:
3376         m_value.valueID = CSSValueTriangle;
3377         break;
3378     case TextEmphasisMark::Sesame:
3379         m_value.valueID = CSSValueSesame;
3380         break;
3381     case TextEmphasisMark::None:
3382     case TextEmphasisMark::Auto:
3383     case TextEmphasisMark::Custom:
3384         ASSERT_NOT_REACHED();
3385         m_value.valueID = CSSValueNone;
3386         break;
3387     }
3388 }
3389
3390 template<> inline CSSPrimitiveValue::operator TextEmphasisMark() const
3391 {
3392     ASSERT(isValueID());
3393
3394     switch (m_value.valueID) {
3395     case CSSValueNone:
3396         return TextEmphasisMark::None;
3397     case CSSValueDot:
3398         return TextEmphasisMark::Dot;
3399     case CSSValueCircle:
3400         return TextEmphasisMark::Circle;
3401     case CSSValueDoubleCircle:
3402         return TextEmphasisMark::DoubleCircle;
3403     case CSSValueTriangle:
3404         return TextEmphasisMark::Triangle;
3405     case CSSValueSesame:
3406         return TextEmphasisMark::Sesame;
3407     default:
3408         break;
3409     }
3410
3411     ASSERT_NOT_REACHED();
3412     return TextEmphasisMark::None;
3413 }
3414
3415 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextOrientation e)
3416     : CSSValue(PrimitiveClass)
3417 {
3418     m_primitiveUnitType = CSS_VALUE_ID;
3419     switch (e) {
3420     case TextOrientation::Sideways:
3421         m_value.valueID = CSSValueSideways;
3422         break;
3423     case TextOrientation::Mixed:
3424         m_value.valueID = CSSValueMixed;
3425         break;
3426     case TextOrientation::Upright:
3427         m_value.valueID = CSSValueUpright;
3428         break;
3429     }
3430 }
3431
3432 template<> inline CSSPrimitiveValue::operator TextOrientation() const
3433 {
3434     ASSERT(isValueID());
3435
3436     switch (m_value.valueID) {
3437     case CSSValueSideways:
3438         return TextOrientation::Sideways;
3439     case CSSValueSidewaysRight:
3440         return TextOrientation::Sideways;
3441     case CSSValueVerticalRight:
3442         return TextOrientation::Mixed;
3443     case CSSValueMixed:
3444         return TextOrientation::Mixed;
3445     case CSSValueUpright:
3446         return TextOrientation::Upright;
3447     default:
3448         break;
3449     }
3450
3451     ASSERT_NOT_REACHED();
3452     return TextOrientation::Mixed;
3453 }
3454
3455 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(PointerEvents e)
3456     : CSSValue(PrimitiveClass)
3457 {
3458     m_primitiveUnitType = CSS_VALUE_ID;
3459     switch (e) {
3460     case PointerEvents::None:
3461         m_value.valueID = CSSValueNone;
3462         break;
3463     case PointerEvents::Stroke:
3464         m_value.valueID = CSSValueStroke;
3465         break;
3466     case PointerEvents::Fill:
3467         m_value.valueID = CSSValueFill;
3468         break;
3469     case PointerEvents::Painted:
3470         m_value.valueID = CSSValuePainted;
3471         break;
3472     case PointerEvents::Visible:
3473         m_value.valueID = CSSValueVisible;
3474         break;
3475     case PointerEvents::VisibleStroke:
3476         m_value.valueID = CSSValueVisibleStroke;
3477         break;
3478     case PointerEvents::VisibleFill:
3479         m_value.valueID = CSSValueVisibleFill;
3480         break;
3481     case PointerEvents::VisiblePainted:
3482         m_value.valueID = CSSValueVisiblePainted;
3483         break;
3484     case PointerEvents::Auto:
3485         m_value.valueID = CSSValueAuto;
3486         break;
3487     case PointerEvents::All:
3488         m_value.valueID = CSSValueAll;
3489         break;
3490     }
3491 }
3492
3493 template<> inline CSSPrimitiveValue::operator PointerEvents() const
3494 {
3495     ASSERT(isValueID());
3496
3497     switch (m_value.valueID) {
3498     case CSSValueAll:
3499         return PointerEvents::All;
3500     case CSSValueAuto:
3501         return PointerEvents::Auto;
3502     case CSSValueNone:
3503         return PointerEvents::None;
3504     case CSSValueVisiblePainted:
3505         return PointerEvents::VisiblePainted;
3506     case CSSValueVisibleFill:
3507         return PointerEvents::VisibleFill;
3508     case CSSValueVisibleStroke:
3509         return PointerEvents::VisibleStroke;
3510     case CSSValueVisible:
3511         return PointerEvents::Visible;
3512     case CSSValuePainted:
3513         return PointerEvents::Painted;
3514     case CSSValueFill:
3515         return PointerEvents::Fill;
3516     case CSSValueStroke:
3517         return PointerEvents::Stroke;
3518     default:
3519         break;
3520     }
3521
3522     ASSERT_NOT_REACHED();
3523     return PointerEvents::All;
3524 }
3525
3526 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Kerning kerning)
3527     : CSSValue(PrimitiveClass)
3528 {
3529     m_primitiveUnitType = CSS_VALUE_ID;
3530     switch (kerning) {
3531     case Kerning::Auto:
3532         m_value.valueID = CSSValueAuto;
3533         return;
3534     case Kerning::Normal:
3535         m_value.valueID = CSSValueNormal;
3536         return;
3537     case Kerning::NoShift:
3538         m_value.valueID = CSSValueNone;
3539         return;
3540     }
3541
3542     ASSERT_NOT_REACHED();
3543     m_value.valueID = CSSValueAuto;
3544 }
3545
3546 template<> inline CSSPrimitiveValue::operator Kerning() const
3547 {
3548     ASSERT(isValueID());
3549
3550     switch (m_value.valueID) {
3551     case CSSValueAuto:
3552         return Kerning::Auto;
3553     case CSSValueNormal:
3554         return Kerning::Normal;
3555     case CSSValueNone:
3556         return Kerning::NoShift;
3557     default:
3558         break;
3559     }
3560
3561     ASSERT_NOT_REACHED();
3562     return Kerning::Auto;
3563 }
3564
3565 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(ObjectFit fit)
3566     : CSSValue(PrimitiveClass)
3567 {
3568     m_primitiveUnitType = CSS_VALUE_ID;
3569     switch (fit) {
3570     case ObjectFit::Fill:
3571         m_value.valueID = CSSValueFill;
3572         break;
3573     case ObjectFit::Contain:
3574         m_value.valueID = CSSValueContain;
3575         break;
3576     case ObjectFit::Cover:
3577         m_value.valueID = CSSValueCover;
3578         break;
3579     case ObjectFit::None:
3580         m_value.valueID = CSSValueNone;
3581         break;
3582     case ObjectFit::ScaleDown:
3583         m_value.valueID = CSSValueScaleDown;
3584         break;
3585     }
3586 }
3587
3588 template<> inline CSSPrimitiveValue::operator ObjectFit() const
3589 {
3590     ASSERT(isValueID());
3591
3592     switch (m_value.valueID) {
3593     case CSSValueFill:
3594         return ObjectFit::Fill;
3595     case CSSValueContain:
3596         return ObjectFit::Contain;
3597     case CSSValueCover:
3598         return ObjectFit::Cover;
3599     case CSSValueNone:
3600         return ObjectFit::None;
3601     case CSSValueScaleDown:
3602         return ObjectFit::ScaleDown;
3603     default:
3604         ASSERT_NOT_REACHED();
3605         return ObjectFit::Fill;
3606     }
3607 }
3608
3609 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FontSmoothingMode smoothing)
3610     : CSSValue(PrimitiveClass)
3611 {
3612     m_primitiveUnitType = CSS_VALUE_ID;
3613     switch (smoothing) {
3614     case FontSmoothingMode::AutoSmoothing:
3615         m_value.valueID = CSSValueAuto;
3616         return;
3617     case FontSmoothingMode::NoSmoothing:
3618         m_value.valueID = CSSValueNone;
3619         return;
3620     case FontSmoothingMode::Antialiased:
3621         m_value.valueID = CSSValueAntialiased;
3622         return;
3623     case FontSmoothingMode::SubpixelAntialiased:
3624         m_value.valueID = CSSValueSubpixelAntialiased;
3625         return;
3626     }
3627
3628     ASSERT_NOT_REACHED();
3629     m_value.valueID = CSSValueAuto;
3630 }
3631
3632 template<> inline CSSPrimitiveValue::operator FontSmoothingMode() const
3633 {
3634     ASSERT(isValueID());
3635
3636     switch (m_value.valueID) {
3637     case CSSValueAuto:
3638         return FontSmoothingMode::AutoSmoothing;
3639     case CSSValueNone:
3640         return FontSmoothingMode::NoSmoothing;
3641     case CSSValueAntialiased:
3642         return FontSmoothingMode::Antialiased;
3643     case CSSValueSubpixelAntialiased:
3644         return FontSmoothingMode::SubpixelAntialiased;
3645     default:
3646         break;
3647     }
3648
3649     ASSERT_NOT_REACHED();
3650     return FontSmoothingMode::AutoSmoothing;
3651 }
3652
3653 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(FontSmallCaps smallCaps)
3654     : CSSValue(PrimitiveClass)
3655 {
3656     m_primitiveUnitType = CSS_VALUE_ID;
3657     switch (smallCaps) {
3658     case FontSmallCaps::Off:
3659         m_value.valueID = CSSValueNormal;
3660         return;
3661     case FontSmallCaps::On:
3662         m_value.valueID = CSSValueSmallCaps;
3663         return;
3664     }
3665
3666     ASSERT_NOT_REACHED();
3667     m_value.valueID = CSSValueNormal;
3668 }
3669
3670 template<> inline CSSPrimitiveValue::operator FontSmallCaps() const
3671 {
3672     ASSERT(isValueID());
3673
3674     switch (m_value.valueID) {
3675     case CSSValueSmallCaps:
3676         return FontSmallCaps::On;
3677     case CSSValueNormal:
3678         return FontSmallCaps::Off;
3679     default:
3680         break;
3681     }
3682     ASSERT_NOT_REACHED();
3683     return FontSmallCaps::Off;
3684 }
3685
3686 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(TextRenderingMode e)
3687     : CSSValue(PrimitiveClass)
3688 {
3689     m_primitiveUnitType = CSS_VALUE_ID;
3690     switch (e) {
3691     case TextRenderingMode::AutoTextRendering:
3692         m_value.valueID = CSSValueAuto;
3693         break;
3694     case TextRenderingMode::OptimizeSpeed:
3695         m_value.valueID = CSSValueOptimizeSpeed;
3696         break;
3697     case TextRenderingMode::OptimizeLegibility:
3698         m_value.valueID = CSSValueOptimizeLegibility;
3699         break;
3700     case TextRenderingMode::GeometricPrecision:
3701         m_value.valueID = CSSValueGeometricPrecision;
3702         break;
3703     }
3704 }
3705
3706 template<> inline CSSPrimitiveValue::operator TextRenderingMode() const
3707 {
3708     ASSERT(isValueID());
3709
3710     switch (m_value.valueID) {
3711     case CSSValueAuto:
3712         return TextRenderingMode::AutoTextRendering;
3713     case CSSValueOptimizeSpeed:
3714         return TextRenderingMode::OptimizeSpeed;
3715     case CSSValueOptimizeLegibility:
3716         return TextRenderingMode::OptimizeLegibility;
3717     case CSSValueGeometricPrecision:
3718         return TextRenderingMode::GeometricPrecision;
3719     default:
3720         break;
3721     }
3722
3723     ASSERT_NOT_REACHED();
3724     return TextRenderingMode::AutoTextRendering;
3725 }
3726
3727 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Hyphens hyphens)
3728     : CSSValue(PrimitiveClass)
3729 {
3730     m_primitiveUnitType = CSS_VALUE_ID;
3731     switch (hyphens) {
3732     case Hyphens::None:
3733         m_value.valueID = CSSValueNone;
3734         break;
3735     case Hyphens::Manual:
3736         m_value.valueID = CSSValueManual;
3737         break;
3738     case Hyphens::Auto:
3739         m_value.valueID = CSSValueAuto;
3740         break;
3741     }
3742 }
3743
3744 template<> inline CSSPrimitiveValue::operator Hyphens() const
3745 {
3746     ASSERT(isValueID());
3747
3748     switch (m_value.valueID) {
3749     case CSSValueNone:
3750         return Hyphens::None;
3751     case CSSValueManual:
3752         return Hyphens::Manual;
3753     case CSSValueAuto:
3754         return Hyphens::Auto;
3755     default:
3756         break;
3757     }
3758
3759     ASSERT_NOT_REACHED();
3760     return Hyphens::Auto;
3761 }
3762
3763 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineSnap gridSnap)
3764     : CSSValue(PrimitiveClass)
3765 {
3766     m_primitiveUnitType = CSS_VALUE_ID;
3767     switch (gridSnap) {
3768     case LineSnap::None:
3769         m_value.valueID = CSSValueNone;
3770         break;
3771     case LineSnap::Baseline:
3772         m_value.valueID = CSSValueBaseline;
3773         break;
3774     case LineSnap::Contain:
3775         m_value.valueID = CSSValueContain;
3776         break;
3777     }
3778 }
3779
3780 template<> inline CSSPrimitiveValue::operator LineSnap() const
3781 {
3782     ASSERT(isValueID());
3783
3784     switch (m_value.valueID) {
3785     case CSSValueNone:
3786         return LineSnap::None;
3787     case CSSValueBaseline:
3788         return LineSnap::Baseline;
3789     case CSSValueContain:
3790         return LineSnap::Contain;
3791     default:
3792         break;
3793     }
3794
3795     ASSERT_NOT_REACHED();
3796     return LineSnap::None;
3797 }
3798
3799 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineAlign lineAlign)
3800     : CSSValue(PrimitiveClass)
3801 {
3802     m_primitiveUnitType = CSS_VALUE_ID;
3803     switch (lineAlign) {
3804     case LineAlign::None:
3805         m_value.valueID = CSSValueNone;
3806         break;
3807     case LineAlign::Edges:
3808         m_value.valueID = CSSValueEdges;
3809         break;
3810     }
3811 }
3812
3813 template<> inline CSSPrimitiveValue::operator LineAlign() const
3814 {
3815     ASSERT(isValueID());
3816
3817     switch (m_value.valueID) {
3818     case CSSValueNone:
3819         return LineAlign::None;
3820     case CSSValueEdges:
3821         return LineAlign::Edges;
3822     default:
3823         break;
3824     }
3825
3826     ASSERT_NOT_REACHED();
3827     return LineAlign::None;
3828 }
3829
3830 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(SpeakAs e)
3831     : CSSValue(PrimitiveClass)
3832 {
3833     m_primitiveUnitType = CSS_VALUE_ID;
3834     switch (e) {
3835     case SpeakAs::Normal:
3836         m_value.valueID = CSSValueNormal;
3837         break;
3838     case SpeakAs::SpellOut:
3839         m_value.valueID = CSSValueSpellOut;
3840         break;
3841     case SpeakAs::Digits:
3842         m_value.valueID = CSSValueDigits;
3843         break;
3844     case SpeakAs::LiteralPunctuation:
3845         m_value.valueID = CSSValueLiteralPunctuation;
3846         break;
3847     case SpeakAs::NoPunctuation:
3848         m_value.valueID = CSSValueNoPunctuation;
3849         break;
3850     }
3851 }
3852
3853 template<> inline CSSPrimitiveValue::operator Order() const
3854 {
3855     ASSERT(isValueID());
3856
3857     switch (m_value.valueID) {
3858     case CSSValueLogical:
3859         return Order::Logical;
3860     case CSSValueVisual:
3861         return Order::Visual;
3862     default:
3863         break;
3864     }
3865
3866     ASSERT_NOT_REACHED();
3867     return Order::Logical;
3868 }
3869
3870 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Order e)
3871     : CSSValue(PrimitiveClass)
3872 {
3873     m_primitiveUnitType = CSS_VALUE_ID;
3874     switch (e) {
3875     case Order::Logical:
3876         m_value.valueID = CSSValueLogical;
3877         break;
3878     case Order::Visual:
3879         m_value.valueID = CSSValueVisual;
3880         break;
3881     }
3882 }
3883
3884 template<> inline CSSPrimitiveValue::operator OptionSet<SpeakAs>() const
3885 {
3886     ASSERT(isValueID());
3887
3888     switch (m_value.valueID) {
3889     case CSSValueNormal:
3890         return OptionSet<SpeakAs> { };
3891     case CSSValueSpellOut:
3892         return SpeakAs::SpellOut;
3893     case CSSValueDigits:
3894         return SpeakAs::Digits;
3895     case CSSValueLiteralPunctuation:
3896         return SpeakAs::LiteralPunctuation;
3897     case CSSValueNoPunctuation:
3898         return SpeakAs::NoPunctuation;
3899     default:
3900         break;
3901     }
3902
3903     ASSERT_NOT_REACHED();
3904     return OptionSet<SpeakAs> { };
3905 }
3906
3907 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(BlendMode blendMode)
3908     : CSSValue(PrimitiveClass)
3909 {
3910     m_primitiveUnitType = CSS_VALUE_ID;
3911     switch (blendMode) {
3912     case BlendMode::Normal:
3913         m_value.valueID = CSSValueNormal;
3914         break;
3915     case BlendMode::Multiply:
3916         m_value.valueID = CSSValueMultiply;
3917         break;
3918     case BlendMode::Screen:
3919         m_value.valueID = CSSValueScreen;
3920         break;
3921     case BlendMode::Overlay:
3922         m_value.valueID = CSSValueOverlay;
3923         break;
3924     case BlendMode::Darken:
3925         m_value.valueID = CSSValueDarken;
3926         break;
3927     case BlendMode::Lighten:
3928         m_value.valueID = CSSValueLighten;
3929         break;
3930     case BlendMode::ColorDodge:
3931         m_value.valueID = CSSValueColorDodge;
3932         break;
3933     case BlendMode::ColorBurn:
3934         m_value.valueID = CSSValueColorBurn;
3935         break;
3936     case BlendMode::HardLight:
3937         m_value.valueID = CSSValueHardLight;
3938         break;
3939     case BlendMode::SoftLight:
3940         m_value.valueID = CSSValueSoftLight;
3941         break;
3942     case BlendMode::Difference:
3943         m_value.valueID = CSSValueDifference;
3944         break;
3945     case BlendMode::Exclusion:
3946         m_value.valueID = CSSValueExclusion;
3947         break;
3948     case BlendMode::Hue:
3949         m_value.valueID = CSSValueHue;
3950         break;
3951     case BlendMode::Saturation:
3952         m_value.valueID = CSSValueSaturation;
3953         break;
3954     case BlendMode::Color:
3955         m_value.valueID = CSSValueColor;
3956         break;
3957     case BlendMode::Luminosity:
3958         m_value.valueID = CSSValueLuminosity;
3959         break;
3960     case BlendMode::PlusDarker:
3961         m_value.valueID = CSSValuePlusDarker;
3962         break;
3963     case BlendMode::PlusLighter:
3964         m_value.valueID = CSSValuePlusLighter;
3965         break;
3966     }
3967 }
3968
3969 template<> inline CSSPrimitiveValue::operator BlendMode() const
3970 {
3971     ASSERT(isValueID());
3972
3973     switch (m_value.valueID) {
3974     case CSSValueNormal:
3975         return BlendMode::Normal;
3976     case CSSValueMultiply:
3977         return BlendMode::Multiply;
3978     case CSSValueScreen:
3979         return BlendMode::Screen;
3980     case CSSValueOverlay:
3981         return BlendMode::Overlay;
3982     case CSSValueDarken:
3983         return BlendMode::Darken;
3984     case CSSValueLighten:
3985         return BlendMode::Lighten;
3986     case CSSValueColorDodge:
3987         return BlendMode::ColorDodge;
3988     case CSSValueColorBurn:
3989         return BlendMode::ColorBurn;
3990     case CSSValueHardLight:
3991         return BlendMode::HardLight;
3992     case CSSValueSoftLight:
3993         return BlendMode::SoftLight;
3994     case CSSValueDifference:
3995         return BlendMode::Difference;
3996     case CSSValueExclusion:
3997         return BlendMode::Exclusion;
3998     case CSSValueHue:
3999         return BlendMode::Hue;
4000     case CSSValueSaturation:
4001         return BlendMode::Saturation;
4002     case CSSValueColor:
4003         return BlendMode::Color;
4004     case CSSValueLuminosity:
4005         return BlendMode::Luminosity;
4006     case CSSValuePlusDarker:
4007         return BlendMode::PlusDarker;
4008     case CSSValuePlusLighter:
4009         return BlendMode::PlusLighter;
4010     default:
4011         break;
4012     }
4013
4014     ASSERT_NOT_REACHED();
4015     return BlendMode::Normal;
4016 }
4017
4018 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(Isolation isolation)
4019     : CSSValue(PrimitiveClass)
4020 {
4021     m_primitiveUnitType = CSS_VALUE_ID;
4022     switch (isolation) {
4023     case Isolation::Auto:
4024         m_value.valueID = CSSValueAuto;
4025         break;
4026     case Isolation::Isolate:
4027         m_value.valueID = CSSValueIsolate;
4028         break;
4029     default:
4030         ASSERT_NOT_REACHED();
4031     }
4032 }
4033
4034 template<> inline CSSPrimitiveValue::operator Isolation() const
4035 {
4036     ASSERT(isValueID());
4037     switch (m_value.valueID) {
4038     case CSSValueAuto:
4039         return Isolation::Auto;
4040     case CSSValueIsolate:
4041         return Isolation::Isolate;
4042     default:
4043         break;
4044     }
4045
4046     ASSERT_NOT_REACHED();
4047     return Isolation::Auto;
4048 }
4049
4050 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineCap e)
4051     : CSSValue(PrimitiveClass)
4052 {
4053     m_primitiveUnitType = CSS_VALUE_ID;
4054     switch (e) {
4055     case ButtCap:
4056         m_value.valueID = CSSValueButt;
4057         break;
4058     case RoundCap:
4059         m_value.valueID = CSSValueRound;
4060         break;
4061     case SquareCap:
4062         m_value.valueID = CSSValueSquare;
4063         break;
4064     }
4065 }
4066
4067 template<> inline CSSPrimitiveValue::operator LineCap() const
4068 {
4069     ASSERT(isValueID());
4070
4071     switch (m_value.valueID) {
4072     case CSSValueButt:
4073         return ButtCap;
4074     case CSSValueRound:
4075         return RoundCap;
4076     case CSSValueSquare:
4077         return SquareCap;
4078     default:
4079         break;
4080     }
4081
4082     ASSERT_NOT_REACHED();
4083     return ButtCap;
4084 }
4085
4086 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(LineJoin e)
4087     : CSSValue(PrimitiveClass)
4088 {
4089     m_primitiveUnitType = CSS_VALUE_ID;
4090     switch (e) {
4091     case MiterJoin:
4092         m_value.valueID = CSSValueMiter;
4093         break;
4094     case RoundJoin:
4095         m_value.valueID = CSSValueRound;
4096         break;
4097     case BevelJoin:
4098         m_value.valueID = CSSValueBevel;
4099         break;
4100     }
4101 }
4102
4103 template<> inline CSSPrimitiveValue::operator LineJoin() const
4104 {
4105     ASSERT(isValueID());
4106
4107     switch (m_value.valueID) {
4108     case CSSValueMiter:
4109         return MiterJoin;
4110     case CSSValueRound:
4111         return RoundJoin;
4112     case CSSValueBevel:
4113         return BevelJoin;
4114     default:
4115         break;
4116     }
4117
4118     ASSERT_NOT_REACHED();
4119     return MiterJoin;
4120 }
4121
4122 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(WindRule e)
4123     : CSSValue(PrimitiveClass)
4124 {
4125     m_primitiveUnitType = CSS_VALUE_ID;
4126     switch (e) {
4127     case WindRule::NonZero:
4128         m_value.valueID = CSSValueNonzero;
4129         break;
4130     case WindRule::EvenOdd:
4131         m_value.valueID = CSSValueEvenodd;
4132         break;
4133     }
4134 }
4135
4136 template<> inline CSSPrimitiveValue::operator WindRule() const
4137 {
4138     ASSERT(isValueID());
4139
4140     switch (m_value.valueID) {
4141     case CSSValueNonzero:
4142         return WindRule::NonZero;
4143     case CSSValueEvenodd:
4144         return WindRule::EvenOdd;
4145     default:
4146         break;
4147     }
4148
4149     ASSERT_NOT_REACHED();
4150     return WindRule::NonZero;
4151 }
4152
4153
4154 template<> inline CSSPrimitiveValue::CSSPrimitiveValue(AlignmentBaseline e)
4155     : CSSValue(PrimitiveClass)
4156 {
4157     m_primitiveUnitType = CSS_VALUE_ID;
4158     switch (e) {
4159     case AlignmentBaseline::Auto:
4160         m_value.valueID = CSSValueAuto;
4161         break;
4162     case AlignmentBaseline::Baseline:
4163         m_value.valueID = CSSValueBaseline;
4164         break;
4165     case AlignmentBaseline::BeforeEdge:
4166         m_value.valueID = CSSValueBeforeEdge;
4167         break;
4168     case AlignmentBaseline::TextBeforeEdge:
4169         m_value.valueID = CSSValueTextBeforeEdge;
4170         break;
4171     case AlignmentBaseline::Middle:
4172         m_value.valueID = CSSValueMiddle;
4173         break;
4174     case AlignmentBaseline::Central:
4175         m_value.valueID = CSSValueCentral;
4176         break;
4177     case AlignmentBaseline::AfterEdge:
4178         m_value.valueID = CSSValueAfterEdge;
4179         break;
4180     case AlignmentBaseline::TextAfterEdge:
4181         m_value.valueID = CSSValueTextAfterEdge;
4182         break;
4183     case AlignmentBaseline::Ideographic:
4184         m_value.valueID = CSSValueIdeographic;
4185         break;
4186     case AlignmentBaseline::Alphabetic:
4187         m_value.valueID = CSSValueAlphabetic;
4188         break;
4189     case AlignmentBaseline::Hanging:
4190         m_value.valueID = CSSValueHanging;
4191         break;
4192     case AlignmentBaseline::Mathematical:
4193         m_value.valueID = CSSValueMathematical;
4194         break;
4195     }
4196 }
4197
4198 template<> inline CSSPrimitiveValue::operator AlignmentBaseline() const
4199 {
4200     ASSERT(isValueID());
4201
4202     switch (m_value.valueID) {
4203     case CSSValueAuto:
4204         return AlignmentBaseline::Auto;