module Geom2D.CubicBezier.Curvature
(curvature, radiusOfCurvature, curvatureExtrema, findRadius)
where
import Geom2D
import Geom2D.CubicBezier.Basic
import Geom2D.CubicBezier.Intersection
import Math.BernsteinPoly
curvature :: CubicBezier Double -> Double -> Double
curvature b t
| t == 0 = curvature' b
| t == 1 = negate $ curvature' $ reorient b
| t < 0.5 = curvature' $ snd $ splitBezier b t
| otherwise = negate $ curvature' $ reorient $ fst $ splitBezier b t
curvature' :: CubicBezier Double -> Double
curvature' (CubicBezier c0 c1 c2 _c3) = 2/3 * b/a^(3::Int)
where
a = vectorDistance c1 c0
b = (c1^-^c0) `vectorCross` (c2^-^c1)
radiusOfCurvature :: CubicBezier Double -> Double -> Double
radiusOfCurvature b t = 1 / curvature b t
extrema :: CubicBezier Double -> BernsteinPoly Double
extrema bez =
let (x, y) = bezierToBernstein bez
x' = bernsteinDeriv x
y' = bernsteinDeriv y
x'' = bernsteinDeriv x'
y'' = bernsteinDeriv y'
x''' = bernsteinDeriv x''
y''' = bernsteinDeriv y''
in (y'~*y' ~+ x'~*x') ~* (x'~*y''' ~- y'~*x''') ~-
3 *~ (x'~*y'' ~- y'~*x'') ~* (y'~*y'' ~+ x'~*x'')
curvatureExtrema :: CubicBezier Double -> Double -> [Double]
curvatureExtrema b eps
| colinear b eps = []
| otherwise = bezierFindRoot (extrema b) 0 1 $
bezierParamTolerance b eps
radiusSquareEq :: CubicBezier Double -> Double -> BernsteinPoly Double
radiusSquareEq bez d =
let (x, y) = bezierToBernstein bez
x' = bernsteinDeriv x
y' = bernsteinDeriv y
x'' = bernsteinDeriv x'
y'' = bernsteinDeriv y'
a = x'~*x' ~+ y'~*y'
b = x'~*y'' ~- x''~*y'
in (a~*a~*a) ~- (d*d) *~ b~*b
findRadius :: CubicBezier Double
-> Double
-> Double
-> [Double]
findRadius b d eps
| colinear b eps = []
| otherwise = bezierFindRoot (radiusSquareEq b d) 0 1 $
bezierParamTolerance b eps