WebCore:
[WebKit-https.git] / WebCore / rendering / Length.h
1 /*
2     This file is part of the KDE libraries
3
4     Copyright (C) 1999 Lars Knoll (knoll@kde.org)
5     Copyright (C) 2006 Apple Computer, Inc.
6
7     This library is free software; you can redistribute it and/or
8     modify it under the terms of the GNU Library General Public
9     License as published by the Free Software Foundation; either
10     version 2 of the License, or (at your option) any later version.
11
12     This library is distributed in the hope that it will be useful,
13     but WITHOUT ANY WARRANTY; without even the implied warranty of
14     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15     Library General Public License for more details.
16
17     You should have received a copy of the GNU Library General Public License
18     along with this library; see the file COPYING.LIB.  If not, write to
19     the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
20     Boston, MA 02110-1301, USA.
21 */
22
23 #ifndef Length_h
24 #define Length_h
25
26 #include <wtf/Assertions.h>
27 #include <wtf/MathExtras.h>
28
29 namespace WebCore {
30
31     const int undefinedLength = -1;
32     const int percentScaleFactor = 128;
33
34     enum LengthType { Auto, Relative, Percent, Fixed, Static, Intrinsic, MinIntrinsic };
35
36     struct Length {
37         Length()
38             : m_value(0)
39         {
40         }
41
42         Length(LengthType t)
43             : m_value(t)
44         {
45         }
46
47         Length(int v, LengthType t, bool q = false)
48             : m_value((v * 16) | (q << 3) | t) // FIXME: Doesn't work if the passed-in value is very large!
49         {
50             ASSERT(t != Percent);
51         }
52
53         Length(double v, LengthType t, bool q = false)
54             : m_value(static_cast<int>(v * percentScaleFactor) * 16 | (q << 3) | t)
55         {
56             ASSERT(t == Percent);
57         }
58
59         bool operator==(const Length& o) const { return m_value == o.m_value; }
60         bool operator!=(const Length& o) const { return m_value != o.m_value; }
61
62         int value() const {
63             ASSERT(type() != Percent);
64             return rawValue();
65         }
66
67         int rawValue() const { return (m_value & ~0xF) / 16; }
68
69         double percent() const
70         {
71             ASSERT(type() == Percent);
72             return static_cast<double>(rawValue()) / percentScaleFactor;
73         }
74
75         LengthType type() const { return static_cast<LengthType>(m_value & 7); }
76         bool quirk() const { return (m_value >> 3) & 1; }
77
78         void setValue(LengthType t, int value)
79         {
80             ASSERT(t != Percent);
81             setRawValue(t, value);
82         }
83
84         void setRawValue(LengthType t, int value) { m_value = value * 16 | (m_value & 0x8) | t; }
85
86         void setValue(int value)
87         {
88             ASSERT(!value || type() != Percent);
89             setRawValue(value);
90         }
91
92         void setRawValue(int value) { m_value = value * 16 | (m_value & 0xF); }
93
94         void setValue(LengthType t, double value)
95         {
96             ASSERT(t == Percent);
97             m_value = static_cast<int>(value * percentScaleFactor) * 16 | (m_value & 0x8) | t;
98         }
99
100         void setValue(double value)
101         {
102             ASSERT(type() == Percent);
103             m_value = static_cast<int>(value * percentScaleFactor) * 16 | (m_value & 0xF);
104         }
105
106         // note: works only for certain types, returns undefinedLength otherwise
107         int calcValue(int maxValue, bool roundPercentages = false) const
108         {
109             switch (type()) {
110                 case Fixed:
111                     return value();
112                 case Percent:
113                     if (roundPercentages)
114                         return static_cast<int>(round(maxValue * percent() / 100.0));
115                     return maxValue * rawValue() / (100 * percentScaleFactor);
116                 case Auto:
117                     return maxValue;
118                 default:
119                     return undefinedLength;
120             }
121         }
122
123         int calcMinValue(int maxValue, bool roundPercentages = false) const
124         {
125             switch (type()) {
126                 case Fixed:
127                     return value();
128                 case Percent:
129                     if (roundPercentages)
130                         return static_cast<int>(round(maxValue * percent() / 100.0));
131                     return maxValue * rawValue() / (100 * percentScaleFactor);
132                 case Auto:
133                 default:
134                     return 0;
135             }
136         }
137
138         bool isUndefined() const { return rawValue() == undefinedLength; }
139         bool isZero() const { return !(m_value & ~0xF); }
140         bool isPositive() const { return rawValue() > 0; }
141         bool isNegative() const { return rawValue() < 0; }
142
143         bool isAuto() const { return type() == Auto; }
144         bool isRelative() const { return type() == Relative; }
145         bool isPercent() const { return type() == Percent; }
146         bool isFixed() const { return type() == Fixed; }
147         bool isStatic() const { return type() == Static; }
148         bool isIntrinsicOrAuto() const { return type() == Auto || type() == MinIntrinsic || type() == Intrinsic; }
149
150         Length blend(const Length& from, double progress) const
151         {
152             // Blend two lengths to produce a new length that is in between them.  Used for animation.
153             if (!from.isZero() && !isZero() && from.type() != type())
154                 return *this;
155     
156             if (from.isZero() && isZero())
157                 return *this;
158             
159             LengthType resultType = type();
160             if (isZero())
161                 resultType = from.type();
162             
163             if (resultType == Percent) {
164                 double fromPercent = from.isZero() ? 0. : from.percent();
165                 double toPercent = isZero() ? 0. : percent();
166                 return Length(fromPercent + (toPercent - fromPercent) * progress, Percent);
167             } 
168                 
169             int fromValue = from.isZero() ? 0 : from.value();
170             int toValue = isZero() ? 0 : value();
171             return Length(int(fromValue + (toValue - fromValue) * progress), resultType);
172         }
173         
174     private:
175         int m_value;
176     };
177
178 } // namespace WebCore
179
180 #endif // Length_h