module Graphics.Implicit.Export.Render.HandleSquares (mergedSquareTris) where
import Graphics.Implicit.Definitions
import Graphics.Implicit.Export.Render.Definitions
import GHC.Exts (groupWith)
import Data.List (sortBy)
import Data.VectorSpace
mergedSquareTris sqTris =
let
triTriangles = concat $ map (\(Tris a) -> a) $ filter isTris sqTris
squares = filter (not . isTris) sqTris
planeAligned = groupWith (\(Sq basis z _ _) -> (basis,z)) squares
joined = map
(
concat . (map joinYaligned) . groupWith (\(Sq _ _ _ yS) -> yS)
. concat . (map joinXaligned) . groupWith (\(Sq _ _ xS _) -> xS))
planeAligned
finishedSquares = concat joined
in
triTriangles ++ concat (map squareToTri finishedSquares)
isTris (Tris _) = True
isTris _ = False
joinXaligned quads@((Sq b z xS _):_) =
let
orderedQuads = sortBy
(\(Sq _ _ _ (ya,_)) (Sq _ _ _ (yb,_)) -> compare ya yb)
quads
mergeAdjacent (pres@(Sq _ _ _ (y1a,y2a)) : next@(Sq _ _ _ (y1b,y2b)) : others) =
if y2a == y1b
then mergeAdjacent ((Sq b z xS (y1a,y2b)): others)
else if y1a == y2b
then mergeAdjacent ((Sq b z xS (y1b,y2a)): others)
else pres : mergeAdjacent (next : others)
mergeAdjacent a = a
in
mergeAdjacent orderedQuads
joinXaligned [] = []
joinYaligned quads@((Sq b z _ yS):_) =
let
orderedQuads = sortBy
(\(Sq _ _ (xa,_) _) (Sq _ _ (xb,_) _) -> compare xa xb)
quads
mergeAdjacent (pres@(Sq _ _ (x1a,x2a) _) : next@(Sq _ _ (x1b,x2b) _) : others) =
if x2a == x1b
then mergeAdjacent ((Sq b z (x1a,x2b) yS): others)
else if x1a == x2b
then mergeAdjacent ((Sq b z (x1b,x2a) yS): others)
else pres : mergeAdjacent (next : others)
mergeAdjacent a = a
in
mergeAdjacent orderedQuads
joinYaligned [] = []
squareToTri (Sq (b1,b2,b3) z (x1,x2) (y1,y2)) =
let
zV = b3 ^* z
(x1V, x2V) = (x1 *^ b1, x2 *^ b1)
(y1V, y2V) = (y1 *^ b2, y2 *^ b2)
a = zV ^+^ x1V ^+^ y1V
b = zV ^+^ x2V ^+^ y1V
c = zV ^+^ x1V ^+^ y2V
d = zV ^+^ x2V ^+^ y2V
in
[(a,b,c),(c,b,d)]