1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
|
/*
* Copyright 2006 The Android Open Source Project
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SkGradientShader_DEFINED
#define SkGradientShader_DEFINED
#include "include/core/SkColor.h"
#include "include/core/SkColorSpace.h"
#include "include/core/SkPoint.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkScalar.h"
#include "include/core/SkShader.h" // IWYU pragma: keep
#include "include/core/SkTileMode.h"
#include "include/private/base/SkAPI.h"
#include <cstdint>
#include <utility>
class SkMatrix;
/** \class SkGradientShader
SkGradientShader hosts factories for creating subclasses of SkShader that
render linear and radial gradients. In general, degenerate cases should not
produce surprising results, but there are several types of degeneracies:
* A linear gradient made from the same two points.
* A radial gradient with a radius of zero.
* A sweep gradient where the start and end angle are the same.
* A two point conical gradient where the two centers and the two radii are
the same.
For any degenerate gradient with a decal tile mode, it will draw empty since the interpolating
region is zero area and the outer region is discarded by the decal mode.
For any degenerate gradient with a repeat or mirror tile mode, it will draw a solid color that
is the average gradient color, since infinitely many repetitions of the gradients will fill the
shape.
For a clamped gradient, every type is well-defined at the limit except for linear gradients. The
radial gradient with zero radius becomes the last color. The sweep gradient draws the sector
from 0 to the provided angle with the first color, with a hardstop switching to the last color.
When the provided angle is 0, this is just the solid last color again. Similarly, the two point
conical gradient becomes a circle filled with the first color, sized to the provided radius,
with a hardstop switching to the last color. When the two radii are both zero, this is just the
solid last color.
As a linear gradient approaches the degenerate case, its shader will approach the appearance of
two half planes, each filled by the first and last colors of the gradient. The planes will be
oriented perpendicular to the vector between the two defining points of the gradient. However,
once they become the same point, Skia cannot reconstruct what that expected orientation is. To
provide a stable and predictable color in this case, Skia just uses the last color as a solid
fill to be similar to many of the other degenerate gradients' behaviors in clamp mode.
*/
class SK_API SkGradientShader {
public:
enum Flags {
/** By default gradients will interpolate their colors in unpremul space
* and then premultiply each of the results. By setting this flag, the
* gradients will premultiply their colors first, and then interpolate
* between them.
* example: https://fiddle.skia.org/c/@GradientShader_MakeLinear
*/
kInterpolateColorsInPremul_Flag = 1 << 0,
};
struct Interpolation {
enum class InPremul : bool { kNo = false, kYes = true };
enum class ColorSpace : uint8_t {
// Default Skia behavior: interpolate in the color space of the destination surface
kDestination,
// https://www.w3.org/TR/css-color-4/#interpolation-space
kSRGBLinear,
kLab,
kOKLab,
// This is the same as kOKLab, except it has a simplified version of the CSS gamut
// mapping algorithm (https://www.w3.org/TR/css-color-4/#css-gamut-mapping)
// into Rec2020 space applied to it.
// Warning: This space is experimental and should not be used in production.
kOKLabGamutMap,
kLCH,
kOKLCH,
// This is the same as kOKLCH, except it has the same gamut mapping applied to it
// as kOKLabGamutMap does.
// Warning: This space is experimental and should not be used in production.
kOKLCHGamutMap,
kSRGB,
kHSL,
kHWB,
kDisplayP3,
kRec2020,
kProphotoRGB,
kA98RGB,
kLastColorSpace = kA98RGB,
};
static constexpr int kColorSpaceCount = static_cast<int>(ColorSpace::kLastColorSpace) + 1;
enum class HueMethod : uint8_t {
// https://www.w3.org/TR/css-color-4/#hue-interpolation
kShorter,
kLonger,
kIncreasing,
kDecreasing,
kLastHueMethod = kDecreasing,
};
static constexpr int kHueMethodCount = static_cast<int>(HueMethod::kLastHueMethod) + 1;
InPremul fInPremul = InPremul::kNo;
ColorSpace fColorSpace = ColorSpace::kDestination;
HueMethod fHueMethod = HueMethod::kShorter; // Only relevant for LCH, OKLCH, HSL, or HWB
static Interpolation FromFlags(uint32_t flags) {
return {flags & kInterpolateColorsInPremul_Flag ? InPremul::kYes : InPremul::kNo,
ColorSpace::kDestination,
HueMethod::kShorter};
}
};
/** Returns a shader that generates a linear gradient between the two specified points.
<p />
@param pts The start and end points for the gradient.
@param colors The array[count] of colors, to be distributed between the two points
@param pos May be NULL. array[count] of SkScalars, or NULL, of the relative position of
each corresponding color in the colors array. If this is NULL,
the the colors are distributed evenly between the start and end point.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >=2. The number of colors (and pos if not NULL) entries.
@param mode The tiling mode
example: https://fiddle.skia.org/c/@GradientShader_MakeLinear
*/
static sk_sp<SkShader> MakeLinear(const SkPoint pts[2],
const SkColor colors[], const SkScalar pos[], int count,
SkTileMode mode,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr);
/** Returns a shader that generates a linear gradient between the two specified points.
<p />
@param pts The start and end points for the gradient.
@param colors The array[count] of colors, to be distributed between the two points
@param pos May be NULL. array[count] of SkScalars, or NULL, of the relative position of
each corresponding color in the colors array. If this is NULL,
the the colors are distributed evenly between the start and end point.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >=2. The number of colors (and pos if not NULL) entries.
@param mode The tiling mode
example: https://fiddle.skia.org/c/@GradientShader_MakeLinear
*/
static sk_sp<SkShader> MakeLinear(const SkPoint pts[2],
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count, SkTileMode mode,
const Interpolation& interpolation,
const SkMatrix* localMatrix);
static sk_sp<SkShader> MakeLinear(const SkPoint pts[2],
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count, SkTileMode mode,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr) {
return MakeLinear(pts, colors, std::move(colorSpace), pos, count, mode,
Interpolation::FromFlags(flags), localMatrix);
}
/** Returns a shader that generates a radial gradient given the center and radius.
<p />
@param center The center of the circle for this gradient
@param radius Must be positive. The radius of the circle for this gradient
@param colors The array[count] of colors, to be distributed between the center and edge of the circle
@param pos May be NULL. The array[count] of SkScalars, or NULL, of the relative position of
each corresponding color in the colors array. If this is NULL,
the the colors are distributed evenly between the center and edge of the circle.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >= 2. The number of colors (and pos if not NULL) entries
@param mode The tiling mode
*/
static sk_sp<SkShader> MakeRadial(const SkPoint& center, SkScalar radius,
const SkColor colors[], const SkScalar pos[], int count,
SkTileMode mode,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr);
/** Returns a shader that generates a radial gradient given the center and radius.
<p />
@param center The center of the circle for this gradient
@param radius Must be positive. The radius of the circle for this gradient
@param colors The array[count] of colors, to be distributed between the center and edge of the circle
@param pos May be NULL. The array[count] of SkScalars, or NULL, of the relative position of
each corresponding color in the colors array. If this is NULL,
the the colors are distributed evenly between the center and edge of the circle.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >= 2. The number of colors (and pos if not NULL) entries
@param mode The tiling mode
*/
static sk_sp<SkShader> MakeRadial(const SkPoint& center, SkScalar radius,
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count, SkTileMode mode,
const Interpolation& interpolation,
const SkMatrix* localMatrix);
static sk_sp<SkShader> MakeRadial(const SkPoint& center, SkScalar radius,
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count, SkTileMode mode,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr) {
return MakeRadial(center, radius, colors, std::move(colorSpace), pos, count, mode,
Interpolation::FromFlags(flags), localMatrix);
}
/**
* Returns a shader that generates a conical gradient given two circles, or
* returns NULL if the inputs are invalid. The gradient interprets the
* two circles according to the following HTML spec.
* http://dev.w3.org/html5/2dcontext/#dom-context-2d-createradialgradient
*/
static sk_sp<SkShader> MakeTwoPointConical(const SkPoint& start, SkScalar startRadius,
const SkPoint& end, SkScalar endRadius,
const SkColor colors[], const SkScalar pos[],
int count, SkTileMode mode,
uint32_t flags = 0,
const SkMatrix* localMatrix = nullptr);
/**
* Returns a shader that generates a conical gradient given two circles, or
* returns NULL if the inputs are invalid. The gradient interprets the
* two circles according to the following HTML spec.
* http://dev.w3.org/html5/2dcontext/#dom-context-2d-createradialgradient
*/
static sk_sp<SkShader> MakeTwoPointConical(const SkPoint& start, SkScalar startRadius,
const SkPoint& end, SkScalar endRadius,
const SkColor4f colors[],
sk_sp<SkColorSpace> colorSpace, const SkScalar pos[],
int count, SkTileMode mode,
const Interpolation& interpolation,
const SkMatrix* localMatrix);
static sk_sp<SkShader> MakeTwoPointConical(const SkPoint& start, SkScalar startRadius,
const SkPoint& end, SkScalar endRadius,
const SkColor4f colors[],
sk_sp<SkColorSpace> colorSpace, const SkScalar pos[],
int count, SkTileMode mode,
uint32_t flags = 0,
const SkMatrix* localMatrix = nullptr) {
return MakeTwoPointConical(start, startRadius, end, endRadius, colors,
std::move(colorSpace), pos, count, mode,
Interpolation::FromFlags(flags), localMatrix);
}
/** Returns a shader that generates a sweep gradient given a center.
The shader accepts negative angles and angles larger than 360, draws
between 0 and 360 degrees, similar to the CSS conic-gradient
semantics. 0 degrees means horizontal positive x axis. The start angle
must be less than the end angle, otherwise a null pointer is
returned. If color stops do not contain 0 and 1 but are within this
range, the respective outer color stop is repeated for 0 and 1. Color
stops less than 0 are clamped to 0, and greater than 1 are clamped to 1.
<p />
@param cx The X coordinate of the center of the sweep
@param cx The Y coordinate of the center of the sweep
@param colors The array[count] of colors, to be distributed around the center, within
the gradient angle range.
@param pos May be NULL. The array[count] of SkScalars, or NULL, of the relative
position of each corresponding color in the colors array. If this is
NULL, then the colors are distributed evenly within the angular range.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >= 2. The number of colors (and pos if not NULL) entries
@param mode Tiling mode: controls drawing outside of the gradient angular range.
@param startAngle Start of the angular range, corresponding to pos == 0.
@param endAngle End of the angular range, corresponding to pos == 1.
*/
static sk_sp<SkShader> MakeSweep(SkScalar cx, SkScalar cy,
const SkColor colors[], const SkScalar pos[], int count,
SkTileMode mode,
SkScalar startAngle, SkScalar endAngle,
uint32_t flags, const SkMatrix* localMatrix);
static sk_sp<SkShader> MakeSweep(SkScalar cx, SkScalar cy,
const SkColor colors[], const SkScalar pos[], int count,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr) {
return MakeSweep(cx, cy, colors, pos, count, SkTileMode::kClamp, 0, 360, flags,
localMatrix);
}
/** Returns a shader that generates a sweep gradient given a center.
The shader accepts negative angles and angles larger than 360, draws
between 0 and 360 degrees, similar to the CSS conic-gradient
semantics. 0 degrees means horizontal positive x axis. The start angle
must be less than the end angle, otherwise a null pointer is
returned. If color stops do not contain 0 and 1 but are within this
range, the respective outer color stop is repeated for 0 and 1. Color
stops less than 0 are clamped to 0, and greater than 1 are clamped to 1.
<p />
@param cx The X coordinate of the center of the sweep
@param cx The Y coordinate of the center of the sweep
@param colors The array[count] of colors, to be distributed around the center, within
the gradient angle range.
@param pos May be NULL. The array[count] of SkScalars, or NULL, of the relative
position of each corresponding color in the colors array. If this is
NULL, then the colors are distributed evenly within the angular range.
If this is not null, the values must lie between 0.0 and 1.0, and be
strictly increasing. If the first value is not 0.0, then an additional
color stop is added at position 0.0, with the same color as colors[0].
If the the last value is not 1.0, then an additional color stop is added
at position 1.0, with the same color as colors[count - 1].
@param count Must be >= 2. The number of colors (and pos if not NULL) entries
@param mode Tiling mode: controls drawing outside of the gradient angular range.
@param startAngle Start of the angular range, corresponding to pos == 0.
@param endAngle End of the angular range, corresponding to pos == 1.
*/
static sk_sp<SkShader> MakeSweep(SkScalar cx, SkScalar cy,
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count,
SkTileMode mode,
SkScalar startAngle, SkScalar endAngle,
const Interpolation& interpolation,
const SkMatrix* localMatrix);
static sk_sp<SkShader> MakeSweep(SkScalar cx, SkScalar cy,
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count,
SkTileMode mode,
SkScalar startAngle, SkScalar endAngle,
uint32_t flags, const SkMatrix* localMatrix) {
return MakeSweep(cx, cy, colors, std::move(colorSpace), pos, count, mode, startAngle,
endAngle, Interpolation::FromFlags(flags), localMatrix);
}
static sk_sp<SkShader> MakeSweep(SkScalar cx, SkScalar cy,
const SkColor4f colors[], sk_sp<SkColorSpace> colorSpace,
const SkScalar pos[], int count,
uint32_t flags = 0, const SkMatrix* localMatrix = nullptr) {
return MakeSweep(cx, cy, colors, std::move(colorSpace), pos, count, SkTileMode::kClamp,
0, 360, flags, localMatrix);
}
};
#endif
|