Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(15)

Unified Diff: ui/gfx/geometry/cubic_bezier.h

Issue 1846733003: UI GFX Geometry: Make UnitBezier a wrapper for gfx::CubicBezier (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Rename it. Created 4 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « third_party/WebKit/Source/platform/blink_platform.gypi ('k') | ui/gfx/geometry/cubic_bezier.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: ui/gfx/geometry/cubic_bezier.h
diff --git a/ui/gfx/geometry/cubic_bezier.h b/ui/gfx/geometry/cubic_bezier.h
index 5a885f35db2f71e719184680ae4bbd2dcc8ec0d4..940ca524a0ddc6e398849c2cd6a988c024b2b6b7 100644
--- a/ui/gfx/geometry/cubic_bezier.h
+++ b/ui/gfx/geometry/cubic_bezier.h
@@ -5,6 +5,10 @@
#ifndef UI_GFX_GEOMETRY_CUBIC_BEZIER_H_
#define UI_GFX_GEOMETRY_CUBIC_BEZIER_H_
+#include <algorithm>
+#include <cmath>
+
+#include "base/logging.h"
#include "base/macros.h"
#include "ui/gfx/gfx_export.h"
@@ -12,30 +16,122 @@ namespace gfx {
class GFX_EXPORT CubicBezier {
public:
- CubicBezier(double x1, double y1, double x2, double y2);
- ~CubicBezier();
+ CubicBezier(double p1x, double p1y, double p2x, double p2y);
+
+ static double GetDefaultEpsilon();
+
+ double SampleCurveX(double t) const {
+ // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.
+ return ((ax * t + bx) * t + cx) * t;
+ }
+
+ double SampleCurveY(double t) const { return ((ay * t + by) * t + cy) * t; }
+
+ double SampleCurveDerivativeX(double t) const {
+ return (3.0 * ax * t + 2.0 * bx) * t + cx;
+ }
+
+ double SampleCurveDerivativeY(double t) const {
+ return (3.0 * ay * t + 2.0 * by) * t + cy;
+ }
+
+ // Given an x value, find a parametric value it came from.
+ double SolveCurveX(double x, double epsilon) const {
danakj 2016/04/01 01:35:58 Put this in the .cc file.
loyso (OOO) 2016/04/01 04:40:06 Done.
+ DCHECK(x >= 0.0);
danakj 2016/04/01 01:35:58 DCHECK_GE
loyso (OOO) 2016/04/01 04:40:06 Done.
+ DCHECK(x <= 1.0);
+
+ double t0;
+ double t1;
+ double t2;
+ double x2;
+ double d2;
+ int i;
- // Returns an approximation of y at the given x.
- double Solve(double x) const;
+ // First try a few iterations of Newton's method -- normally very fast.
+ for (t2 = x, i = 0; i < 8; i++) {
+ x2 = SampleCurveX(t2) - x;
+ if (fabs(x2) < epsilon)
+ return t2;
+ d2 = SampleCurveDerivativeX(t2);
+ if (fabs(d2) < 1e-6)
+ break;
+ t2 = t2 - x2 / d2;
+ }
+
+ // Fall back to the bisection method for reliability.
+ t0 = 0.0;
+ t1 = 1.0;
+ t2 = x;
+
+ while (t0 < t1) {
+ x2 = SampleCurveX(t2);
+ if (fabs(x2 - x) < epsilon)
+ return t2;
+ if (x > x2)
+ t0 = t2;
+ else
+ t1 = t2;
+ t2 = (t1 - t0) * .5 + t0;
+ }
+
+ // Failure.
+ return t2;
+ }
+
+ // Evaluates y at the given x.
+ double Solve(double x) const {
+ return SolveWithEpsilon(x, GetDefaultEpsilon());
+ }
+
+ // Evaluates y at the given x. The epsilon parameter provides a hint as to the
+ // required
+ // accuracy and is not guaranteed.
+ double SolveWithEpsilon(double x, double epsilon) const {
+ if (x < 0.0)
+ return 0.0 + start_gradient_ * x;
+ if (x > 1.0)
+ return 1.0 + end_gradient_ * (x - 1.0);
+ return SampleCurveY(SolveCurveX(x, epsilon));
+ }
// Returns an approximation of dy/dx at the given x.
- double Slope(double x) const;
+ double Slope(double x) const {
+ return SlopeWithEpsilon(x, GetDefaultEpsilon());
+ }
+
+ double SlopeWithEpsilon(double x, double epsilon) const {
+ double t = SolveCurveX(x, epsilon);
+ double dx = SampleCurveDerivativeX(t);
+ double dy = SampleCurveDerivativeY(t);
+ return dy / dx;
+ }
// Sets |min| and |max| to the bezier's minimum and maximium y values in the
// interval [0, 1].
- void Range(double* min, double* max) const;
+ void Range(double* min, double* max) const {
danakj 2016/04/01 01:35:58 Just make 2 getters, minimum() and maximum()?
loyso (OOO) 2016/04/01 02:58:03 It would affect the call sites. Can we change the
danakj 2016/04/01 19:26:17 Sure yep. Put a TODO?
danakj 2016/04/01 19:54:33 Or, make the blink wrapper use the min/max accesso
loyso (OOO) 2016/04/04 03:39:58 Done. It's CC-only for now.
+ *min = range_min_;
+ *max = range_max_;
+ }
private:
- void InitGradients();
+ void InitCoefficients(double p1x, double p1y, double p2x, double p2y);
+ void InitGradients(double p1x, double p1y, double p2x, double p2y);
+ void InitRange(double p1y, double p2y);
- double x1_;
- double y1_;
- double x2_;
- double y2_;
+ double ax;
+ double bx;
+ double cx;
+
+ double ay;
+ double by;
+ double cy;
double start_gradient_;
double end_gradient_;
+ double range_min_;
+ double range_max_;
+
DISALLOW_ASSIGN(CubicBezier);
};
« no previous file with comments | « third_party/WebKit/Source/platform/blink_platform.gypi ('k') | ui/gfx/geometry/cubic_bezier.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698