{-# OPTIONS_GHC -fno-warn-unticked-promoted-constructors -fno-warn-orphans #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE OverloadedLabels #-}
{-# LANGUAGE OverloadedLists #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}
module GI.Gtk.Declarative.Container.Patch
( IsContainer(..)
, patchInContainer
)
where
import Data.Foldable (foldMap)
import Data.Vector (Vector, (!?))
import qualified Data.Vector as Vector
import qualified GI.Gtk as Gtk
import GI.Gtk.Declarative.Bin
import GI.Gtk.Declarative.Container.Box
import GI.Gtk.Declarative.Container.Class
import GI.Gtk.Declarative.Patch
import GI.Gtk.Declarative.State
patchInContainer
:: ( Gtk.IsWidget container
, Gtk.IsContainer container
, Patchable child
, IsContainer container child
)
=> StateTree 'ContainerState container child event cs
-> container
-> Vector (child e1)
-> Vector (child e2)
-> IO (StateTree 'ContainerState container child event cs)
patchInContainer (StateTreeContainer top children) container os' ns' = do
let maxLength = maximum ([length children, length os', length ns'] :: [Int])
indices = Vector.enumFromN 0 (fromIntegral maxLength)
newChildren <- foldMap
go
(Vector.zip4 indices
(padMaybes maxLength children)
(padMaybes maxLength os')
(padMaybes maxLength ns')
)
Gtk.widgetQueueResize container
return (StateTreeContainer top newChildren)
where
go = \case
(i, Just oldChildState, Just old, Just new) ->
case patch oldChildState old new of
Modify modify -> pure <$> modify
Replace createWidget -> do
newChildState <- createWidget
oldChildWidget <- someStateWidget oldChildState
newChildWidget <- someStateWidget newChildState
replaceChild container new i oldChildWidget newChildWidget
return (pure newChildState)
Keep -> return (pure oldChildState)
(i, Just oldChildState, Nothing, Just new) -> do
newChildState <- create new
oldChildWidget <- someStateWidget oldChildState
newChildWidget <- someStateWidget newChildState
replaceChild container new i oldChildWidget newChildWidget
return (Vector.singleton newChildState)
(_i, Nothing, _, Just n) -> do
newChildState <- create n
w <- someStateWidget newChildState
appendChild container n w
return (Vector.singleton newChildState)
(_i, Just childState, Just _, Nothing) -> do
Gtk.widgetDestroy =<< someStateWidget childState
return Vector.empty
(_i, Nothing , Just _ , Nothing) -> return Vector.empty
(_i, Just childState, Nothing, Nothing) -> do
Gtk.widgetDestroy =<< someStateWidget childState
return Vector.empty
(_i, Nothing, Nothing, Nothing) -> return Vector.empty
padMaybes :: Int -> Vector a -> Vector (Maybe a)
padMaybes len xs = Vector.generate len (xs !?)
instance IsContainer Gtk.ListBox (Bin Gtk.ListBoxRow) where
appendChild box _ widget' =
Gtk.listBoxInsert box widget' (-1)
replaceChild box _ i old new = do
Gtk.widgetDestroy old
Gtk.listBoxInsert box new i
instance IsContainer Gtk.Box BoxChild where
appendChild box BoxChild {properties = BoxChildProperties {expand, fill, padding}} widget' =
Gtk.boxPackStart box widget' expand fill padding
replaceChild box boxChild' i old new = do
Gtk.widgetDestroy old
appendChild box boxChild' new
Gtk.boxReorderChild box new i