/* * Copyright (C) 2011, 2012 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef SKY_ENGINE_CORE_CSS_CSSCALCULATIONVALUE_H_ #define SKY_ENGINE_CORE_CSS_CSSCALCULATIONVALUE_H_ #include "sky/engine/core/css/CSSPrimitiveValue.h" #include "sky/engine/core/css/CSSValue.h" #include "sky/engine/core/css/parser/CSSParserValues.h" #include "sky/engine/platform/CalculationValue.h" #include "sky/engine/wtf/PassOwnPtr.h" #include "sky/engine/wtf/RefCounted.h" #include "sky/engine/wtf/RefPtr.h" namespace blink { class CSSParserValueList; class CSSValueList; class CalculationValue; class CalcExpressionNode; class Length; enum CalcOperator { CalcAdd = '+', CalcSubtract = '-', CalcMultiply = '*', CalcDivide = '/' }; // The order of this enum should not change since its elements are used as indices // in the addSubtractResult matrix. enum CalculationCategory { CalcNumber = 0, CalcLength, CalcPercent, CalcPercentNumber, CalcPercentLength, CalcAngle, CalcTime, CalcFrequency, CalcOther }; class CSSCalcExpressionNode : public RefCounted { DECLARE_EMPTY_VIRTUAL_DESTRUCTOR_WILL_BE_REMOVED(CSSCalcExpressionNode); public: enum Type { CssCalcPrimitiveValue = 1, CssCalcBinaryOperation }; virtual bool isZero() const = 0; virtual double doubleValue() const = 0; virtual double computeLengthPx(const CSSToLengthConversionData&) const = 0; virtual void accumulateLengthArray(CSSLengthArray&, double multiplier) const = 0; virtual void accumulatePixelsAndPercent(const CSSToLengthConversionData&, PixelsAndPercent&, float multiplier = 1) const = 0; virtual String customCSSText() const = 0; virtual bool equals(const CSSCalcExpressionNode& other) const { return m_category == other.m_category && m_isInteger == other.m_isInteger; } virtual Type type() const = 0; CalculationCategory category() const { return m_category; } virtual CSSPrimitiveValue::UnitType primitiveType() const = 0; bool isInteger() const { return m_isInteger; } protected: CSSCalcExpressionNode(CalculationCategory category, bool isInteger) : m_category(category) , m_isInteger(isInteger) { ASSERT(category != CalcOther); } CalculationCategory m_category; bool m_isInteger; }; class CSSCalcValue : public CSSValue { public: static PassRefPtr create(CSSParserString name, CSSParserValueList*, ValueRange); static PassRefPtr create(PassRefPtr, ValueRange = ValueRangeAll); static PassRefPtr createExpressionNode(PassRefPtr, bool isInteger = false); static PassRefPtr createExpressionNode(PassRefPtr, PassRefPtr, CalcOperator); static PassRefPtr createExpressionNode(double pixels, double percent); PassRefPtr toCalcValue(const CSSToLengthConversionData& conversionData) const { PixelsAndPercent value(0, 0); m_expression->accumulatePixelsAndPercent(conversionData, value); return CalculationValue::create(value, m_nonNegative ? ValueRangeNonNegative : ValueRangeAll); } CalculationCategory category() const { return m_expression->category(); } bool isInt() const { return m_expression->isInteger(); } double doubleValue() const; bool isNegative() const { return m_expression->doubleValue() < 0; } ValueRange permittedValueRange() { return m_nonNegative ? ValueRangeNonNegative : ValueRangeAll; } double computeLengthPx(const CSSToLengthConversionData&) const; void accumulateLengthArray(CSSLengthArray& lengthArray, double multiplier) const { m_expression->accumulateLengthArray(lengthArray, multiplier); } CSSCalcExpressionNode* expressionNode() const { return m_expression.get(); } String customCSSText() const; bool equals(const CSSCalcValue&) const; private: CSSCalcValue(PassRefPtr expression, ValueRange range) : CSSValue(CalculationClass) , m_expression(expression) , m_nonNegative(range == ValueRangeNonNegative) { } double clampToPermittedRange(double) const; const RefPtr m_expression; const bool m_nonNegative; }; DEFINE_CSS_VALUE_TYPE_CASTS(CSSCalcValue, isCalcValue()); } // namespace blink #endif // SKY_ENGINE_CORE_CSS_CSSCALCULATIONVALUE_H_