{-# LANGUAGE ScopedTypeVariables #-} -------------------------------------------------------------------------------- -- | -- Module : Algorithms.Geometry.LineSegmentIntersection.BentleyOttmann -- Copyright : (C) Frank Staals -- License : see the LICENSE file -- Maintainer : Frank Staals -- -- The \(O((n+k)\log n)\) time line segment intersection algorithm by Bentley -- and Ottmann. -- -------------------------------------------------------------------------------- module Algorithms.Geometry.LineSegmentIntersection.BentleyOttmannOld where import Algorithms.Geometry.LineSegmentIntersection.TypesNoExt( Intersections , IntersectionPoint(..) , Associated(..) , mkIntersectionPoint ) import Control.Lens hiding (contains) import Data.Ext import qualified Data.Foldable as F import Data.Geometry.Interval import Data.Geometry.LineSegment import Data.Geometry.Point import Data.Geometry.Properties import qualified Data.List as L import Data.List.NonEmpty (NonEmpty(..)) import qualified Data.List.NonEmpty as NonEmpty import qualified Data.Map as M import Data.Maybe import Data.Ord (Down(..), comparing) import qualified Data.OrdSeq as SS -- status struct import qualified Data.Set as EQ -- event queue import Data.Vinyl import Data.Vinyl.CoRec -------------------------------------------------------------------------------- -- todo; use an old copy of the imports as well. -- | Compute all intersections -- -- \(O((n+k)\log n)\), where \(k\) is the number of intersections. intersections :: (Ord r, Fractional r) => [LineSegment 2 p r] -> Intersections p r intersections ss = merge $ sweep pts mempty where pts = EQ.fromAscList . groupStarts . L.sort . concatMap asEventPts $ ss -- | Computes all intersection points p s.t. p lies in the interior of at least -- one of the segments. -- -- \(O((n+k)\log n)\), where \(k\) is the number of intersections. interiorIntersections :: (Ord r, Fractional r) => [LineSegment 2 p r] -> Intersections p r interiorIntersections = M.filter isInteriorIntersection . intersections isInteriorIntersection :: Associated p r -> Bool isInteriorIntersection = not . null . _interiorTo -- | Computes the event points for a given line segment asEventPts :: Ord r => LineSegment 2 p r -> [Event p r] asEventPts s = let [p,q] = L.sortBy ordPoints [s^.start.core,s^.end.core] in [Event p (Start $ s :| []), Event q (End s)] -- | Group the segments with the intersection points merge :: (Ord r, Fractional r) => [IntersectionPoint p r] -> Intersections p r merge = foldr (\(IntersectionPoint p a) -> M.insertWith (<>) p a) M.empty -- | Group the startpoints such that segments with the same start point -- correspond to one event. groupStarts :: Eq r => [Event p r] -> [Event p r] groupStarts [] = [] groupStarts (Event p (Start s) : es) = Event p (Start ss) : groupStarts rest where (ss',rest) = L.span sameStart es -- sort the segs on lower endpoint ss = let (x:|xs) = s in x :| (xs ++ concatMap startSegs ss') sameStart (Event q (Start _)) = p == q sameStart _ = False groupStarts (e : es) = e : groupStarts es -------------------------------------------------------------------------------- -- * Data type for Events -- | Type of segment data EventType s = Start !(NonEmpty s)| Intersection | End !s deriving (Show) instance Eq (EventType s) where a == b = a `compare` b == EQ instance Ord (EventType s) where (Start _) `compare` (Start _) = EQ (Start _) `compare` _ = LT Intersection `compare` (Start _) = GT Intersection `compare` Intersection = EQ Intersection `compare` (End _) = LT (End _) `compare` (End _) = EQ (End _) `compare` _ = GT -- | The actual event consists of a point and its type data Event p r = Event { eventPoint :: !(Point 2 r) , eventType :: !(EventType (LineSegment 2 p r)) } deriving (Show,Eq) instance Ord r => Ord (Event p r) where -- decreasing on the y-coord, then increasing on x-coord, and increasing on event-type (Event p s) `compare` (Event q t) = case ordPoints p q of EQ -> s `compare` t x -> x -- | An ordering that is decreasing on y, increasing on x ordPoints :: Ord r => Point 2 r -> Point 2 r -> Ordering ordPoints a b = let f p = (Down $ p^.yCoord, p^.xCoord) in comparing f a b -- | Get the segments that start at the given event point startSegs :: Event p r -> [LineSegment 2 p r] startSegs e = case eventType e of Start ss -> NonEmpty.toList ss _ -> [] -------------------------------------------------------------------------------- -- * The Main Sweep type EventQueue p r = EQ.Set (Event p r) type StatusStructure p r = SS.OrdSeq (LineSegment 2 p r) -- | Run the sweep handling all events sweep :: (Ord r, Fractional r) => EventQueue p r -> StatusStructure p r -> [IntersectionPoint p r] sweep eq ss = case EQ.minView eq of Nothing -> [] Just (e,eq') -> handle e eq' ss isClosedStart :: Eq r => Point 2 r -> LineSegment 2 p r -> Bool isClosedStart p (LineSegment s e) | p == s^.unEndPoint.core = isClosed s | otherwise = isClosed e -- | Handle an event point handle :: forall r p. (Ord r, Fractional r) => Event p r -> EventQueue p r -> StatusStructure p r -> [IntersectionPoint p r] handle e@(eventPoint -> p) eq ss = toReport <> sweep eq' ss' where starts = startSegs e (before,contains',after) = extractContains p ss (ends,contains) = L.partition (endsAt p) contains' -- starting segments, exluding those that have an open starting point starts' = filter (isClosedStart p) starts toReport = case starts' ++ contains' of (_:_:_) -> [mkIntersectionPoint p (starts' <> ends) contains] _ -> [] -- new status structure ss' = before <> newSegs <> after newSegs = toStatusStruct p $ starts ++ contains -- the new eeventqueue eq' = foldr EQ.insert eq es -- the new events: es | F.null newSegs = maybeToList $ app (findNewEvent p) sl sr | otherwise = let s' = fst <$> SS.minView newSegs s'' = fst <$> SS.maxView newSegs in catMaybes [ app (findNewEvent p) sl s' , app (findNewEvent p) s'' sr ] sl = fst <$> SS.maxView before sr = fst <$> SS.minView after app f x y = do { x' <- x ; y' <- y ; f x' y'} -- | split the status structure, extracting the segments that contain p. -- the result is (before,contains,after) extractContains :: (Fractional r, Ord r) => Point 2 r -> StatusStructure p r -> (StatusStructure p r, [LineSegment 2 p r], StatusStructure p r) extractContains p ss = (before, F.toList $ mid1 <> mid2, after) where (before, mid1, after') = SS.splitOn (xCoordAt $ p^.yCoord) (p^.xCoord) ss -- Make sure to also select the horizontal segments containing p (mid2, after) = SS.splitMonotonic (not . intersects p) after' -- | Given a point and the linesegements that contain it. Create a piece of -- status structure for it. toStatusStruct :: (Fractional r, Ord r) => Point 2 r -> [LineSegment 2 p r] -> StatusStructure p r toStatusStruct p xs = ss <> hors -- ss { SS.nav = ordAtNav $ p^.yCoord } `SS.join` hors where (hors',rest) = L.partition isHorizontal xs ss = SS.fromListBy (ordAtY $ maxY xs) rest hors = SS.fromListBy (comparing rightEndpoint) hors' isHorizontal s = s^.start.core.yCoord == s^.end.core.yCoord -- find the y coord of the first interesting thing below the sweep at y maxY = maximum . filter (< p^.yCoord) . concatMap (\s -> [s^.start.core.yCoord,s^.end.core.yCoord]) -- | Get the right endpoint of a segment rightEndpoint :: Ord r => LineSegment 2 p r -> r rightEndpoint s = (s^.start.core.xCoord) `max` (s^.end.core.xCoord) -- | Test if a segment ends at p endsAt :: Ord r => Point 2 r -> LineSegment 2 p r -> Bool endsAt p (LineSegment' a b) = all (\q -> ordPoints (q^.core) p /= GT) [a,b] -------------------------------------------------------------------------------- -- * Finding New events -- | Find all events findNewEvent :: (Ord r, Fractional r) => Point 2 r -> LineSegment 2 p r -> LineSegment 2 p r -> Maybe (Event p r) findNewEvent p l r = match (l `intersect` r) $ (H $ \NoIntersection -> Nothing) :& (H $ \q -> if ordPoints q p == GT then Just (Event q Intersection) else Nothing) :& (H $ \_ -> Nothing) -- full segment intersectsions are handled -- at insertion time :& RNil