/*
* Copyright 2019 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "SkCurve.h"
#include "SkParticleData.h"
#include "SkRandom.h"
#include "SkReflected.h"
constexpr SkFieldVisitor::EnumStringMapping gCurveSegmentTypeMapping[] = {
{ kConstant_SegmentType, "Constant" },
{ kLinear_SegmentType, "Linear" },
{ kCubic_SegmentType, "Cubic" },
};
static SkColor4f operator+(SkColor4f c1, SkColor4f c2) {
return { c1.fR + c2.fR, c1.fG + c2.fG, c1.fB + c2.fB, c1.fA + c2.fA };
}
static SkColor4f operator-(SkColor4f c1, SkColor4f c2) {
return { c1.fR - c2.fR, c1.fG - c2.fG, c1.fB - c2.fB, c1.fA - c2.fA };
}
template <typename T>
static T eval_cubic(const T* pts, SkScalar x) {
SkScalar ix = (1 - x);
return pts[0]*(ix*ix*ix) + pts[1]*(3*ix*ix*x) + pts[2]*(3*ix*x*x) + pts[3]*(x*x*x);
}
template <typename T>
static T eval_segment(const T* pts, SkScalar x, int type) {
switch (type) {
case kLinear_SegmentType:
return pts[0] + (pts[3] - pts[0]) * x;
case kCubic_SegmentType:
return eval_cubic(pts, x);
case kConstant_SegmentType:
default:
return pts[0];
}
}
SkScalar SkCurveSegment::eval(SkScalar x, SkScalar t, bool negate) const {
SkScalar result = eval_segment(fMin, x, fType);
if (fRanged) {
result += (eval_segment(fMax, x, fType) - result) * t;
}
if (fBidirectional && negate) {
result = -result;
}
return result;
}
void SkCurveSegment::visitFields(SkFieldVisitor* v) {
v->visit("Type", fType, gCurveSegmentTypeMapping, SK_ARRAY_COUNT(gCurveSegmentTypeMapping));
v->visit("Ranged", fRanged);
v->visit("Bidirectional", fBidirectional);
v->visit("A0", fMin[0]);
if (fType == kCubic_SegmentType) {
v->visit("B0", fMin[1]);
v->visit("C0", fMin[2]);
}
if (fType != kConstant_SegmentType) {
v->visit("D0", fMin[3]);
}
if (fRanged) {
v->visit("A1", fMax[0]);
if (fType == kCubic_SegmentType) {
v->visit("B1", fMax[1]);
v->visit("C1", fMax[2]);
}
if (fType != kConstant_SegmentType) {
v->visit("D1", fMax[3]);
}
}
}
SkScalar SkCurve::eval(const SkParticleUpdateParams& params, SkParticleState& ps) const {
SkASSERT(fSegments.count() == fXValues.count() + 1);
float x = fInput.eval(params, ps);
int i = 0;
for (; i < fXValues.count(); ++i) {
if (x <= fXValues[i]) {
break;
}
}
SkScalar rangeMin = (i == 0) ? 0.0f : fXValues[i - 1];
SkScalar rangeMax = (i == fXValues.count()) ? 1.0f : fXValues[i];
SkScalar segmentX = (x - rangeMin) / (rangeMax - rangeMin);
if (!SkScalarIsFinite(segmentX)) {
segmentX = rangeMin;
}
SkASSERT(0.0f <= segmentX && segmentX <= 1.0f);
// Always pull t and negate here, so that the stable generator behaves consistently, even if
// our segments use an inconsistent feature-set.
SkScalar t = ps.fRandom.nextF();
bool negate = ps.fRandom.nextBool();
return fSegments[i].eval(segmentX, t, negate);
}
void SkCurve::visitFields(SkFieldVisitor* v) {
v->visit("Input", fInput);
v->visit("XValues", fXValues);
v->visit("Segments", fSegments);
// Validate and fixup
if (fSegments.empty()) {
fSegments.push_back().setConstant(0.0f);
}
fXValues.resize_back(fSegments.count() - 1);
for (int i = 0; i < fXValues.count(); ++i) {
fXValues[i] = SkTPin(fXValues[i], i > 0 ? fXValues[i - 1] : 0.0f, 1.0f);
}
}
SkColor4f SkColorCurveSegment::eval(SkScalar x, SkScalar t) const {
SkColor4f result = eval_segment(fMin, x, fType);
if (fRanged) {
result = result + (eval_segment(fMax, x, fType) - result) * t;
}
return result;
}
void SkColorCurveSegment::visitFields(SkFieldVisitor* v) {
v->visit("Type", fType, gCurveSegmentTypeMapping, SK_ARRAY_COUNT(gCurveSegmentTypeMapping));
v->visit("Ranged", fRanged);
v->visit("A0", fMin[0]);
if (fType == kCubic_SegmentType) {
v->visit("B0", fMin[1]);
v->visit("C0", fMin[2]);
}
if (fType != kConstant_SegmentType) {
v->visit("D0", fMin[3]);
}
if (fRanged) {
v->visit("A1", fMax[0]);
if (fType == kCubic_SegmentType) {
v->visit("B1", fMax[1]);
v->visit("C1", fMax[2]);
}
if (fType != kConstant_SegmentType) {
v->visit("D1", fMax[3]);
}
}
}
SkColor4f SkColorCurve::eval(const SkParticleUpdateParams& params, SkParticleState& ps) const {
SkASSERT(fSegments.count() == fXValues.count() + 1);
float x = fInput.eval(params, ps);
int i = 0;
for (; i < fXValues.count(); ++i) {
if (x <= fXValues[i]) {
break;
}
}
SkScalar rangeMin = (i == 0) ? 0.0f : fXValues[i - 1];
SkScalar rangeMax = (i == fXValues.count()) ? 1.0f : fXValues[i];
SkScalar segmentX = (x - rangeMin) / (rangeMax - rangeMin);
if (!SkScalarIsFinite(segmentX)) {
segmentX = rangeMin;
}
SkASSERT(0.0f <= segmentX && segmentX <= 1.0f);
return fSegments[i].eval(segmentX, ps.fRandom.nextF());
}
void SkColorCurve::visitFields(SkFieldVisitor* v) {
v->visit("Input", fInput);
v->visit("XValues", fXValues);
v->visit("Segments", fSegments);
// Validate and fixup
if (fSegments.empty()) {
fSegments.push_back().setConstant(SkColor4f{ 1.0f, 1.0f, 1.0f, 1.0f });
}
fXValues.resize_back(fSegments.count() - 1);
for (int i = 0; i < fXValues.count(); ++i) {
fXValues[i] = SkTPin(fXValues[i], i > 0 ? fXValues[i - 1] : 0.0f, 1.0f);
}
}