#ifndef CSSGradientValue_h
#define CSSGradientValue_h
#include "core/css/CSSImageGeneratorValue.h"
#include "core/css/CSSPrimitiveValue.h"
#include "wtf/RefPtr.h"
#include "wtf/Vector.h"
namespace WebCore {
class FloatPoint;
class Gradient;
class TextLinkColors;
enum CSSGradientType {
CSSDeprecatedLinearGradient,
CSSDeprecatedRadialGradient,
CSSPrefixedLinearGradient,
CSSPrefixedRadialGradient,
CSSLinearGradient,
CSSRadialGradient
};
enum CSSGradientRepeat { NonRepeating, Repeating };
struct CSSGradientColorStop {
ALLOW_ONLY_INLINE_ALLOCATION();
public:
CSSGradientColorStop() : m_colorIsDerivedFromElement(false) { };
RefPtrWillBeMember<CSSPrimitiveValue> m_position;
RefPtrWillBeMember<CSSPrimitiveValue> m_color;
Color m_resolvedColor;
bool m_colorIsDerivedFromElement;
bool operator==(const CSSGradientColorStop& other) const
{
return compareCSSValuePtr(m_color, other.m_color)
&& compareCSSValuePtr(m_position, other.m_position);
}
void trace(Visitor*);
};
}
namespace WTF {
template <> struct VectorTraits<WebCore::CSSGradientColorStop> : VectorTraitsBase<WebCore::CSSGradientColorStop> {
static const bool canInitializeWithMemset = true;
static const bool canMoveWithMemcpy = true;
};
}
namespace WebCore {
class CSSGradientValue : public CSSImageGeneratorValue {
public:
PassRefPtr<Image> image(RenderObject*, const IntSize&);
void setFirstX(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_firstX = val; }
void setFirstY(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_firstY = val; }
void setSecondX(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_secondX = val; }
void setSecondY(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_secondY = val; }
void addStop(const CSSGradientColorStop& stop) { m_stops.append(stop); }
unsigned stopCount() const { return m_stops.size(); }
void sortStopsIfNeeded();
bool isRepeating() const { return m_repeating; }
CSSGradientType gradientType() const { return m_gradientType; }
bool isFixedSize() const { return false; }
IntSize fixedSize(const RenderObject*) const { return IntSize(); }
bool isPending() const { return false; }
bool knownToBeOpaque(const RenderObject*) const;
void loadSubimages(ResourceFetcher*) { }
PassRefPtrWillBeRawPtr<CSSGradientValue> gradientWithStylesResolved(const TextLinkColors&, Color currentColor);
void traceAfterDispatch(Visitor*);
protected:
CSSGradientValue(ClassType classType, CSSGradientRepeat repeat, CSSGradientType gradientType)
: CSSImageGeneratorValue(classType)
, m_stopsSorted(false)
, m_gradientType(gradientType)
, m_repeating(repeat == Repeating)
{
}
CSSGradientValue(const CSSGradientValue& other, ClassType classType, CSSGradientType gradientType)
: CSSImageGeneratorValue(classType)
, m_firstX(other.m_firstX)
, m_firstY(other.m_firstY)
, m_secondX(other.m_secondX)
, m_secondY(other.m_secondY)
, m_stops(other.m_stops)
, m_stopsSorted(other.m_stopsSorted)
, m_gradientType(gradientType)
, m_repeating(other.isRepeating() ? Repeating : NonRepeating)
{
}
void addStops(Gradient*, const CSSToLengthConversionData&, float maxLengthForRepeat = 0);
FloatPoint computeEndPoint(CSSPrimitiveValue*, CSSPrimitiveValue*, const CSSToLengthConversionData&, const IntSize&);
bool isCacheable() const;
RefPtrWillBeMember<CSSPrimitiveValue> m_firstX;
RefPtrWillBeMember<CSSPrimitiveValue> m_firstY;
RefPtrWillBeMember<CSSPrimitiveValue> m_secondX;
RefPtrWillBeMember<CSSPrimitiveValue> m_secondY;
WillBeHeapVector<CSSGradientColorStop, 2> m_stops;
bool m_stopsSorted;
CSSGradientType m_gradientType;
bool m_repeating;
};
DEFINE_CSS_VALUE_TYPE_CASTS(CSSGradientValue, isGradientValue());
class CSSLinearGradientValue : public CSSGradientValue {
public:
static PassRefPtrWillBeRawPtr<CSSLinearGradientValue> create(CSSGradientRepeat repeat, CSSGradientType gradientType = CSSLinearGradient)
{
return adoptRefWillBeRefCountedGarbageCollected(new CSSLinearGradientValue(repeat, gradientType));
}
void setAngle(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_angle = val; }
String customCSSText() const;
PassRefPtr<Gradient> createGradient(const CSSToLengthConversionData&, const IntSize&);
PassRefPtrWillBeRawPtr<CSSLinearGradientValue> clone() const
{
return adoptRefWillBeRefCountedGarbageCollected(new CSSLinearGradientValue(*this));
}
bool equals(const CSSLinearGradientValue&) const;
void traceAfterDispatch(Visitor*);
private:
CSSLinearGradientValue(CSSGradientRepeat repeat, CSSGradientType gradientType = CSSLinearGradient)
: CSSGradientValue(LinearGradientClass, repeat, gradientType)
{
}
explicit CSSLinearGradientValue(const CSSLinearGradientValue& other)
: CSSGradientValue(other, LinearGradientClass, other.gradientType())
, m_angle(other.m_angle)
{
}
RefPtrWillBeMember<CSSPrimitiveValue> m_angle;
};
DEFINE_CSS_VALUE_TYPE_CASTS(CSSLinearGradientValue, isLinearGradientValue());
class CSSRadialGradientValue : public CSSGradientValue {
public:
static PassRefPtrWillBeRawPtr<CSSRadialGradientValue> create(CSSGradientRepeat repeat, CSSGradientType gradientType = CSSRadialGradient)
{
return adoptRefWillBeRefCountedGarbageCollected(new CSSRadialGradientValue(repeat, gradientType));
}
PassRefPtrWillBeRawPtr<CSSRadialGradientValue> clone() const
{
return adoptRefWillBeRefCountedGarbageCollected(new CSSRadialGradientValue(*this));
}
String customCSSText() const;
void setFirstRadius(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_firstRadius = val; }
void setSecondRadius(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_secondRadius = val; }
void setShape(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_shape = val; }
void setSizingBehavior(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_sizingBehavior = val; }
void setEndHorizontalSize(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_endHorizontalSize = val; }
void setEndVerticalSize(PassRefPtrWillBeRawPtr<CSSPrimitiveValue> val) { m_endVerticalSize = val; }
PassRefPtr<Gradient> createGradient(const CSSToLengthConversionData&, const IntSize&);
bool equals(const CSSRadialGradientValue&) const;
void traceAfterDispatch(Visitor*);
private:
CSSRadialGradientValue(CSSGradientRepeat repeat, CSSGradientType gradientType = CSSRadialGradient)
: CSSGradientValue(RadialGradientClass, repeat, gradientType)
{
}
explicit CSSRadialGradientValue(const CSSRadialGradientValue& other)
: CSSGradientValue(other, RadialGradientClass, other.gradientType())
, m_firstRadius(other.m_firstRadius)
, m_secondRadius(other.m_secondRadius)
, m_shape(other.m_shape)
, m_sizingBehavior(other.m_sizingBehavior)
, m_endHorizontalSize(other.m_endHorizontalSize)
, m_endVerticalSize(other.m_endVerticalSize)
{
}
float resolveRadius(CSSPrimitiveValue*, const CSSToLengthConversionData&, float* widthOrHeight = 0);
RefPtrWillBeMember<CSSPrimitiveValue> m_firstRadius;
RefPtrWillBeMember<CSSPrimitiveValue> m_secondRadius;
RefPtrWillBeMember<CSSPrimitiveValue> m_shape;
RefPtrWillBeMember<CSSPrimitiveValue> m_sizingBehavior;
RefPtrWillBeMember<CSSPrimitiveValue> m_endHorizontalSize;
RefPtrWillBeMember<CSSPrimitiveValue> m_endVerticalSize;
};
DEFINE_CSS_VALUE_TYPE_CASTS(CSSRadialGradientValue, isRadialGradientValue());
}
#endif