-- Implicit CAD. Copyright (C) 2011, Christopher Olah (chris@colah.ca) -- Copyright (C) 2016, Julia Longtin (julial@turinglace.com) -- Released under the GNU AGPLV3+, see LICENSE -- Allow us to use explicit foralls when writing function type declarations. {-# LANGUAGE ExplicitForAll #-} -- FIXME: required. why? {-# LANGUAGE MultiParamTypeClasses, FunctionalDependencies, FlexibleInstances, FlexibleContexts, TypeSynonymInstances, UndecidableInstances #-} module Graphics.Implicit.ObjectUtil.GetImplicit2 (getImplicit2) where import Prelude(Int, Num, abs, (-), (/), sqrt, (*), (+), (!!), mod, length, map, (<=), (&&), (>=), (||), odd, ($), (>), filter, (<), minimum, (==), maximum, max, cos, sin, head, tail) import Graphics.Implicit.Definitions (SymbolicObj2(RectR, Circle, PolygonR, Complement2, UnionR2, DifferenceR2, IntersectR2, Translate2, Scale2, Rotate2, Shell2, Outset2, EmbedBoxedObj2), Obj2, ℝ, ℝ2, (⋯/)) import Graphics.Implicit.MathUtil (rminimum, rmaximum, distFromLineSeg) import Data.VectorSpace ((^-^)) import Data.List (nub) getImplicit2 :: SymbolicObj2 -> Obj2 -- Primitives getImplicit2 (RectR r (x1,y1) (x2,y2)) = \(x,y) -> rmaximum r [abs (x-dx/2-x1) - dx/2, abs (y-dy/2-y1) - dy/2] where (dx, dy) = (x2-x1, y2-y1) getImplicit2 (Circle r ) = \(x,y) -> sqrt (x * x + y * y) - r getImplicit2 (PolygonR _ points) = \p -> let pair :: Int -> (ℝ2,ℝ2) pair n = (points !! n, points !! (mod (n + 1) (length points) ) ) pairs = [ pair n | n <- [0 .. (length points) - 1] ] relativePairs = map (\(a,b) -> (a ^-^ p, b ^-^ p) ) pairs crossing_points = [x2 ^-^ y2*(x2-x1)/(y2-y1) | ((x1,y1), (x2,y2)) <-relativePairs, ( (y2 <= 0) && (y1 >= 0) ) || ( (y2 >= 0) && (y1 <= 0) ) ] seemsInRight = odd $ length $ filter (>0) $ nub crossing_points seemsInLeft = odd $ length $ filter (<0) $ nub crossing_points isIn = seemsInRight && seemsInLeft dists = map (distFromLineSeg p) pairs :: [ℝ] in minimum dists * if isIn then -1 else 1 -- (Rounded) CSG getImplicit2 (Complement2 symbObj) = let obj = getImplicit2 symbObj in \p -> - obj p getImplicit2 (UnionR2 r symbObjs) = let objs = map getImplicit2 symbObjs in if r == 0 then \p -> minimum $ map ($p) objs else \p -> rminimum r $ map ($p) objs getImplicit2 (DifferenceR2 r symbObjs) = let objs = map getImplicit2 symbObjs obj = head objs complement :: forall a t. Num a => (t -> a) -> t -> a complement obj' = \p -> - obj' p in if r == 0 then \p -> maximum $ map ($p) $ obj:(map complement $ tail objs) else \p -> rmaximum r $ map ($p) $ obj:(map complement $ tail objs) getImplicit2 (IntersectR2 r symbObjs) = let objs = map getImplicit2 symbObjs in if r == 0 then \p -> maximum $ map ($p) objs else \p -> rmaximum r $ map ($p) objs -- Simple transforms getImplicit2 (Translate2 v symbObj) = let obj = getImplicit2 symbObj in \p -> obj (p ^-^ v) getImplicit2 (Scale2 s@(sx,sy) symbObj) = let obj = getImplicit2 symbObj in \p -> (max sx sy) * obj (p ⋯/ s) getImplicit2 (Rotate2 θ symbObj) = let obj = getImplicit2 symbObj in \(x,y) -> obj ( cos(θ)*x + sin(θ)*y, cos(θ)*y - sin(θ)*x) -- Boundary mods getImplicit2 (Shell2 w symbObj) = let obj = getImplicit2 symbObj in \p -> abs (obj p) - w/2 getImplicit2 (Outset2 d symbObj) = let obj = getImplicit2 symbObj in \p -> obj p - d -- Misc getImplicit2 (EmbedBoxedObj2 (obj,_)) = obj