我有兴趣为我的monad变换器堆栈获取缩放功能,该功能的定义方式如下:
newtype Awesome a = Awesome (StateT AwesomeState (ExceptT B.ByteString IO) a)
deriving (Functor, Applicative, Monad
, MonadIO, MonadError B.ByteString
, MonadState AwesomeState)
我的 AwesomeState 是深层嵌套的记录,因此使用zoom会极大地帮助我更新一些字段。但问题是缩放对于我的新类型来说并不是开箱即用的。
Couldn't match type ‘Control.Lens.Internal.Zoom.Zoomed Awesome’
with ‘Control.Lens.Internal.Zoom.Zoomed m0’
我找到了一个如何创建Zoom的自定义newtype RWST实例的示例,但是尝试将其调整为我的newtype却没有给我带来任何结果
RWST示例可在此处找到:http://lpaste.net/87737
有没有办法可以开始使用我的monad变换器堆栈进行缩放?为了达到这个目的,我需要做些什么?如果我应该像在RWST示例中那样实现Zoomed / Zoom,那么我需要一个如何做到这一点的指针,因为我尝试过但未能这样做。
答案 0 :(得分:3)
我建议明确Awesome
状态:
{-# LANGUAGE TemplateHaskell, GeneralizedNewtypeDeriving, TypeFamilies,
FlexibleInstances, FunctionalDependencies #-}
import Control.Applicative
import Control.Monad.Error
import Control.Monad.State
import Control.Lens
import Control.Lens.Zoom
import Control.Lens.Internal.Zoom
data AwesomeState = AwesomeState
{ _someRecord :: String
-- ...
}
$(makeLenses ''AwesomeState)
newtype Awesome s a = Awesome (StateT s (ErrorT String IO) a)
deriving ( Functor, Applicative, Monad
, MonadIO, MonadError String
, MonadState s)
然后您可以为它定义Zoom
实例,如下所示:
type instance Zoomed (Awesome s) = Focusing (ErrorT String IO)
instance Zoom (Awesome s) (Awesome t) s t where
zoom l (Awesome m) = Awesome (zoom l m)
然后你会
zoom someRecord :: Awesome String a -> Awesome AwesomeState a