blob: ecf81a305c0a907cd4c119576a8e69e7d506d668 [file] [log] [blame]
/*
* Copyright (C) 2006-2021 Apple Inc. All rights reserved.
* Copyright (C) 2007 Alp Toker <alp@atoker.com>
* Copyright (C) 2008 Torch Mobile, Inc.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 APPLE INC. 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.
*/
#pragma once
#include "Color.h"
#include "ColorInterpolationMethod.h"
#include "FloatPoint.h"
#include "GradientColorStops.h"
#include "GraphicsTypes.h"
#include <variant>
#include <wtf/Vector.h>
#if USE(CG)
#include "GradientRendererCG.h"
#endif
#if USE(CG)
typedef struct CGContext* CGContextRef;
#endif
#if USE(CAIRO)
typedef struct _cairo_pattern cairo_pattern_t;
#endif
namespace WTF {
class TextStream;
}
namespace WebCore {
class AffineTransform;
class FloatRect;
class GraphicsContext;
class Gradient : public RefCounted<Gradient> {
friend WTF::TextStream& operator<<(WTF::TextStream&, const Gradient&);
public:
struct LinearData {
FloatPoint point0;
FloatPoint point1;
template<typename Encoder> void encode(Encoder&) const;
template<typename Decoder> static std::optional<LinearData> decode(Decoder&);
};
struct RadialData {
FloatPoint point0;
FloatPoint point1;
float startRadius;
float endRadius;
float aspectRatio; // For elliptical gradient, width / height.
template<typename Encoder> void encode(Encoder&) const;
template<typename Decoder> static std::optional<RadialData> decode(Decoder&);
};
struct ConicData {
FloatPoint point0;
float angleRadians;
template<typename Encoder> void encode(Encoder&) const;
template<typename Decoder> static std::optional<ConicData> decode(Decoder&);
};
using Data = std::variant<LinearData, RadialData, ConicData>;
WEBCORE_EXPORT static Ref<Gradient> create(Data&&, ColorInterpolationMethod, GradientSpreadMethod = GradientSpreadMethod::Pad, GradientColorStops&& = { });
bool isZeroSize() const;
const Data& data() const { return m_data; }
WEBCORE_EXPORT void addColorStop(GradientColorStop&&);
const GradientColorStops& stops() const { return m_stops; }
GradientSpreadMethod spreadMethod() const { return m_spreadMethod; }
void fill(GraphicsContext&, const FloatRect&);
void adjustParametersForTiledDrawing(FloatSize&, FloatRect&, const FloatSize& spacing);
unsigned hash() const;
#if USE(CAIRO)
RefPtr<cairo_pattern_t> createPattern(float globalAlpha, const AffineTransform&);
#endif
#if USE(CG)
void paint(GraphicsContext&);
void paint(CGContextRef);
#endif
template<typename Encoder> void encode(Encoder&) const;
template<typename Decoder> static std::optional<Ref<Gradient>> decode(Decoder&);
private:
explicit Gradient(Data&&, ColorInterpolationMethod, GradientSpreadMethod, GradientColorStops&&);
void stopsChanged();
Data m_data;
ColorInterpolationMethod m_colorInterpolationMethod;
GradientSpreadMethod m_spreadMethod;
GradientColorStops m_stops;
mutable unsigned m_cachedHash { 0 };
#if USE(CG)
std::optional<GradientRendererCG> m_platformRenderer;
#endif
};
template<typename Encoder> void Gradient::LinearData::encode(Encoder& encoder) const
{
encoder << point0;
encoder << point1;
}
template<typename Decoder> std::optional<Gradient::LinearData> Gradient::LinearData::decode(Decoder& decoder)
{
std::optional<FloatPoint> point0;
decoder >> point0;
if (!point0)
return std::nullopt;
std::optional<FloatPoint> point1;
decoder >> point1;
if (!point1)
return std::nullopt;
return {{ *point0, *point1 }};
}
template<typename Encoder> void Gradient::RadialData::encode(Encoder& encoder) const
{
encoder << point0;
encoder << point1;
encoder << startRadius;
encoder << endRadius;
encoder << aspectRatio;
}
template<typename Decoder> std::optional<Gradient::RadialData> Gradient::RadialData::decode(Decoder& decoder)
{
std::optional<FloatPoint> point0;
decoder >> point0;
if (!point0)
return std::nullopt;
std::optional<FloatPoint> point1;
decoder >> point1;
if (!point1)
return std::nullopt;
std::optional<float> startRadius;
decoder >> startRadius;
if (!startRadius)
return std::nullopt;
std::optional<float> endRadius;
decoder >> endRadius;
if (!endRadius)
return std::nullopt;
std::optional<float> aspectRatio;
decoder >> aspectRatio;
if (!aspectRatio)
return std::nullopt;
return {{ *point0, *point1, *startRadius, *endRadius, *aspectRatio }};
}
template<typename Encoder> void Gradient::ConicData::encode(Encoder& encoder) const
{
encoder << point0;
encoder << angleRadians;
}
template<typename Decoder> std::optional<Gradient::ConicData> Gradient::ConicData::decode(Decoder& decoder)
{
std::optional<FloatPoint> point0;
decoder >> point0;
if (!point0)
return std::nullopt;
std::optional<float> angleRadians;
decoder >> angleRadians;
if (!angleRadians)
return std::nullopt;
return {{ *point0, *angleRadians }};
}
template<typename Encoder> void Gradient::encode(Encoder& encoder) const
{
encoder << m_data;
encoder << m_colorInterpolationMethod;
encoder << m_spreadMethod;
encoder << m_stops;
}
template<typename Decoder> std::optional<Ref<Gradient>> Gradient::decode(Decoder& decoder)
{
std::optional<Data> data;
decoder >> data;
if (!data)
return std::nullopt;
std::optional<ColorInterpolationMethod> colorInterpolationMethod;
decoder >> colorInterpolationMethod;
if (!colorInterpolationMethod)
return std::nullopt;
std::optional<GradientSpreadMethod> spreadMethod;
decoder >> spreadMethod;
if (!spreadMethod)
return std::nullopt;
std::optional<GradientColorStops> stops;
decoder >> stops;
if (!stops)
return std::nullopt;
return Gradient::create(WTFMove(*data), *colorInterpolationMethod, *spreadMethod, WTFMove(*stops));
}
} // namespace WebCore