{-# LANGUAGE UndecidableInstances #-}
module Web.Hyperbole.HyperView.Handled where
import Data.Kind (Constraint, Type)
import GHC.TypeLits hiding (Mod)
import Web.Hyperbole.HyperView.Types
import Web.Hyperbole.TypeList
import Web.Hyperbole.View (View)
type family ValidDescendents x :: [Type] where
ValidDescendents x = x : NextDescendents '[] '[x]
type family NextDescendents (ex :: [Type]) (xs :: [Type]) where
NextDescendents _ '[] = '[]
NextDescendents ex (x ': xs) =
RemoveAll (x : ex) (Require x)
<++> NextDescendents ((x : ex) <++> Require x) (RemoveAll (x : ex) (Require x))
<++> NextDescendents (x : ex) (RemoveAll (x : ex) xs)
type NotHandled id ctx (views :: [Type]) =
TypeError
( 'Text "HyperView "
:<>: 'ShowType id
:<>: 'Text " not found in (Require "
:<>: 'ShowType ctx
:<>: 'Text ")"
:$$: 'Text " "
:<>: 'ShowType views
:$$: 'Text "Try adding it to the HyperView instance:"
:$$: 'Text " instance HyperView "
:<>: 'ShowType ctx
:<>: 'Text " where"
:$$: 'Text " type Action "
:<>: 'ShowType ctx
:<>: 'Text " = "
:<>: ShowType (Action id)
:<>: 'Text ""
:$$: 'Text " type Require "
:<>: 'ShowType ctx
:<>: 'Text " = ["
:<>: ShowType id
:<>: 'Text ", ...]"
)
type NotDesc id ctx x cs =
TypeError
( 'Text ""
:<>: 'ShowType x
:<>: 'Text ", a child of HyperView "
:<>: 'ShowType id
:<>: 'Text ", not handled by context "
:<>: 'ShowType ctx
:$$: ('Text " Require = " ':<>: 'ShowType cs)
)
type NotInPage x total =
TypeError
( 'Text ""
:<>: 'ShowType x
:<>: 'Text " not included in: "
:$$: 'Text " Page es "
:<>: ShowType total
:$$: 'Text "try expanding the page views to:"
:$$: 'Text " Page es "
:<>: ShowType (x : total)
)
type family HyperViewHandled id ctx :: Constraint where
HyperViewHandled id (View view ()) = TypeError ('Text "View c () is not a valid ViewState, did you forget to pass ViewState into target or runViewContext?")
HyperViewHandled id ctx =
(
ElemOr id (ctx : Require ctx) (NotHandled id ctx (Require ctx))
,
CheckDescendents id ctx
)
type family CheckDescendents id ctx :: Constraint where
CheckDescendents id (Root total) =
( AllInPage (ValidDescendents id) total
)
CheckDescendents id ctx = ()
type family AllInPage ids total :: Constraint where
AllInPage '[] _ = ()
AllInPage (x ': xs) total =
(ElemOr x total (NotInPage x total), AllInPage xs total)