-- Implicit CAD. Copyright (C) 2011, Christopher Olah (chris@colah.ca) -- Released under the GNU GPL, see LICENSE {-# LANGUAGE MultiParamTypeClasses, FunctionalDependencies, FlexibleInstances, FlexibleContexts, TypeSynonymInstances, UndecidableInstances #-} -- This file symbolicaly renders contours and contour fillings. -- If it can't, it passes the puck to a marching-squares-like -- algorithm... module Graphics.Implicit.Export.SymbolicObj2 where import Graphics.Implicit.Definitions import Graphics.Implicit.Export.Definitions import Graphics.Implicit.Export.MarchingSquares import Graphics.Implicit.Export.MarchingSquaresFill import Graphics.Implicit.Primitives import Graphics.Implicit.ObjectUtil import Graphics.Implicit.Export.Symbolic.Rebound2 import Graphics.Implicit.Export.Symbolic.Rebound3 import qualified Graphics.Implicit.SaneOperators as S instance DiscreteAproxable SymbolicObj2 [Polyline] where discreteAprox res obj = symbolicGetContour res obj symbolicGetOrientedContour :: ℝ -> SymbolicObj2 -> [Polyline] symbolicGetOrientedContour res symbObj = map orient $ symbolicGetContour res symbObj where obj = getImplicit2 symbObj orient :: Polyline -> Polyline orient points@(x:y:_) = let v = (\(a,b) -> (b, -a)) (y S.- x) dv = v S./ (S.norm v / res / 0.1) in if obj (x S.+ dv) - obj x > 0 then points else reverse points symbolicGetContour :: ℝ -> SymbolicObj2 -> [Polyline] symbolicGetContour _ (RectR 0 (x1,y1) (x2,y2)) = [[ (x1,y1), (x2,y1), (x2,y2), (x1,y2), (x1,y1) ]] symbolicGetContour res (Circle r) = [[ ( r*cos(2*pi*m/n), r*sin(2*pi*m/n) ) | m <- [0.. n] ]] where n = max 5 (fromIntegral $ ceiling $ 2*pi*r/res) symbolicGetContour res (Translate2 v obj) = map (map (S.+ v) ) $ symbolicGetContour res obj symbolicGetContour res (Scale2 s obj) = map (map (S.⋯* s)) $ symbolicGetContour res obj symbolicGetContour res obj = case rebound2 (getImplicit2 obj, getBox2 obj) of (obj, (a,b)) -> getContour a b (res,res) obj symbolicGetContourMesh :: ℝ -> SymbolicObj2 -> [(ℝ2,ℝ2,ℝ2)] symbolicGetContourMesh res (Translate2 v obj) = map (\(a,b,c) -> (a S.+ v, b S.+ v, c S.+ v) ) $ symbolicGetContourMesh res obj symbolicGetContourMesh res (Scale2 s obj) = map (\(a,b,c) -> (a S.⋯* s, b S.⋯* s, c S.⋯* s) ) $ symbolicGetContourMesh res obj symbolicGetContourMesh _ (RectR 0 (x1,y1) (x2,y2)) = [((x1,y1), (x2,y1), (x2,y2)), ((x2,y2), (x1,y2), (x1,y1)) ] symbolicGetContourMesh res (Circle r) = [ ((0,0), (r*cos(2*pi*m/n), r*sin(2*pi*m/n)), (r*cos(2*pi*(m+1)/n), r*sin(2*pi*(m+1)/n)) )| m <- [0.. n-1] ] where n = max 5 (fromIntegral $ ceiling $ 2*pi*r/res) symbolicGetContourMesh res obj = case rebound2 (getImplicit2 obj, getBox2 obj) of (obj, (a,b)) -> getContourMesh a b (res,res) obj