- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我编写了一个小应用程序来跟踪我在电视剧中的进度。该应用程序是用 Haskell 使用函数响应式(Reactive)编程 (FRP) 编写的 reactive banana .
应用程序可以:
我在编写将新电视剧添加到表中并连接新事件的代码时遇到问题。来自 here 的 CRUD 示例并没有太大帮助我,因为我有更多的要求,然后只是从列表中选择一个元素。
如何编写一个 reactiveTable
函数,例如 CRUD Example 中的 reactiveListDisplay
函数以玻璃钢方式?网络编译完成后,如何为删除按钮以及季节和剧集旋转按钮添加事件?
data Series = Series { name :: String
, season :: Int
, episode :: Int
}
insertIntoTable :: TableClass t => t -> SeriesChangeHandler -> SeriesRemoveHandler -> Series -> IO ()
insertIntoTable table changeHandler removeHandler (Series name s e) = do
(rows, cols) <- tableGetSize table
tableResize table (rows+1) cols
nameLabel <- labelNew $ Just name
adjustmentS <- adjustmentNew (fromIntegral s) 1 1000 1 0 0
adjustmentE <- adjustmentNew (fromIntegral e) 1 1000 1 0 0
seasonButton <- spinButtonNew adjustmentS 1.0 0
episodeButton <- spinButtonNew adjustmentE 1.0 0
removeButton <- buttonNewWithLabel "remove"
let getSeries = do
s <- spinButtonGetValue seasonButton
e <- spinButtonGetValue episodeButton
return $ Series name (round s) (round e)
handleSeries onEvent widget handler = do
onEvent widget $ do
series <- getSeries
handler series
handleSeries onValueSpinned seasonButton changeHandler
handleSeries onValueSpinned episodeButton changeHandler
onPressed removeButton $ do
series <- getSeries
containerRemove table nameLabel
containerRemove table seasonButton
containerRemove table episodeButton
containerRemove table removeButton
removeHandler series
let tadd widget x = tableAdd table widget x (rows - 1)
tadd nameLabel 0
tadd seasonButton 1
tadd episodeButton 2
tadd removeButton 3
widgetShowAll table
main :: IO ()
main = do
initGUI
window <- windowNew
scroll <- scrolledWindowNew Nothing Nothing
table <- tableNew 1 5 True
addButton <- buttonNewWithLabel "add series"
vbox <- vBoxNew False 10
containerAdd window vbox
boxPackStart vbox addButton PackNatural 0
let networkDescription :: forall t. Frameworks t => Moment t ()
networkDescription = do
addEvent <- eventButton addButton
(changeHandler,fireChange) <- liftIO $ newAddHandler
changeEvent <- fromAddHandler changeHandler
(removeHandler,fireRemove) <- liftIO $ newAddHandler
removeEvent <- fromAddHandler removeHandler
let insertIntoTable' = insertIntoTable table fireChange fireRemove
addSeries e = do
s <- addSeriesDialog
liftIO $ insertIntoTable' s
liftIO $ mapM_ insertIntoTable' initSeries
reactimate $ addSeries <$> addEvent
reactimate $ updateSeries conn <$> changeEvent
reactimate $ removeSeries conn <$> removeEvent
network <- compile networkDescription
actuate network
onDestroy window $ do
D.disconnect conn
mainQuit
widgetShowAll window
mainGUI
我想重构 insertIntoTable 方法以使用事件和行为,而不是使用简单的回调。
编辑:
我已经尝试过gtk TreeView与 ListStore后端。在这种情况下,您不需要动态事件切换。我编写了下面的 reactiveList
函数来从插入、更改和删除事件中获取列表行为。有效^^
reactiveList :: (Frameworks t)
=> ListStore a
-> Event t (Int,a) -- insert event
-> Event t (Int,a) -- change event
-> Event t (Int,a) -- remove event
-> Moment t (Behavior t [a])
reactiveList store insertE changeE removeE = do
(listHandler,fireList) <- liftIO $ newAddHandler
let onChange f (i,a) = do
f i a
list <- listStoreToList store
fireList list
reactimate $ onChange (listStoreInsert store) <$> insertE
reactimate $ onChange (listStoreSetValue store) <$> changeE
reactimate $ onChange (const . listStoreRemove store) <$> removeE
initList <- liftIO $ listStoreToList store
fromChanges initList listHandler
main :: IO ()
main = do
initGUI
window <- windowNew
addButton <- buttonNewWithLabel "add series"
vbox <- vBoxNew False 10
seriesList <- listStoreNew (initSeries :: [Series])
listView <- treeViewNewWithModel seriesList
treeViewSetHeadersVisible listView True
let newCol title newRenderer f = do
col <- treeViewColumnNew
treeViewColumnSetTitle col title
renderer <- newRenderer
cellLayoutPackStart col renderer False
cellLayoutSetAttributes col renderer seriesList f
treeViewAppendColumn listView col
return renderer
newCol "Image" cellRendererPixbufNew $ \s -> [cellPixbuf :=> newPixbuf s]
newCol "Name" cellRendererTextNew $ \s -> [cellText := name s]
seasonSpin <- newCol "Season" cellRendererSpinNew $ \s ->
[ cellRendererSpinAdjustment :=> adjustmentNew (fromIntegral (season s)) 1 1000 1 0 0
, cellText := (show $ season s)
, cellTextEditable := True
]
episodeSpin <- newCol "Episode" cellRendererSpinNew $ \s ->
[ cellRendererSpinAdjustment :=> adjustmentNew (fromIntegral (episode s)) 1 1000 1 0 0
, cellText := (show $ episode s)
, cellTextEditable := True
]
containerAdd window vbox
boxPackStart vbox listView PackGrow 0
boxPackStart vbox addButton PackNatural 0
let networkDescription :: forall t. Frameworks t => Moment t ()
networkDescription = do
(addHandler,fireAdd) <- liftIO $ newAddHandler
maybeSeriesE <- fromAddHandler addHandler
(removeHandler,fireRemove) <- liftIO $ newAddHandler
removeE <- fromAddHandler removeHandler
-- when the add button was pressed,
-- open a dialog and return maybe a new series
askSeriesE <- eventButton addButton
reactimate $ (const $ fireAdd =<< askSeries) <$> askSeriesE
-- ommit all nothing series
let insertE = filterJust maybeSeriesE
insert0E = ((,) 0) <$> insertE
seasonSpinE <- eventSpin seasonSpin seriesList
episodeSpinE <- eventSpin episodeSpin seriesList
let changeSeason (i,d,s) = (i,s {season = round d})
changeEpisode (i,d,s) = (i,s {episode = round d})
let changeE = (changeSeason <$> seasonSpinE) `union` (changeEpisode <$> episodeSpinE)
listB <- reactiveList seriesList insert0E changeE removeE
listE <- (changes listB)
reactimate $ (putStrLn . unlines . map show) <$> listE
reactimate $ insertSeries conn <$> insertE
reactimate $ updateSeries conn . snd <$> changeE
reactimate $ removeSeries conn . snd <$> removeE
return ()
network <- compile networkDescription
actuate network
onDestroy window $ do
D.disconnect conn
mainQuit
widgetShowAll window
mainGUI
我愿意接受意见和建议。
最佳答案
听起来你的问题更接近 Bar Tab比 CRUD 的例子。
添加新小部件以及新行为和事件的基本思想是使用所谓的“动态事件切换”。从本质上讲,这是一种将新创建的事件和行为放回到网络中的方法。
创建新小部件的操作分为两部分。第一部分是使用 liftIO 创建小部件。第二个是获取其输入并使用 trimE
或trimB
作为适当的。省略大部分 GTk 特定的细节(我不知道如何使用 GTk :P),它看起来像这样:
let newSeries name = do
label <- liftIO . labelNew $ Just name
liftIO $ tadd labelNew 0
{- ... the rest of your controls here ... -}
seasonNumber <- trimB $ getSpinButtonBehavior seasonButton
{- ... wrap the rest of the inputs using trimB and trimE ... -}
return (label, seasonNumber, ...)
因此,此函数创建一个新的小部件,“修剪”其输入并将值返回给您。现在您必须实际使用这些值:
newSeasons <- execute (FrameworkMoment newSeries <$> nameEvents)
此处 nameEvents
应该是一个 Event String
,其中包含每次要添加的新系列名称的事件。
现在您已经拥有了所有新季节的流,您可以使用诸如 stepper
之类的工具将其全部组合成条目列表的单个行为。
有关更多详细信息(包括从所有小部件中获取聚合信息等内容),请查看实际的示例代码。
关于haskell - 带有反应香蕉和 gtk2hs 的 react 表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16482996/
阅读Yesod的源代码中,我遇到过带有“.hsfiles”扩展名的文件。示例:mongo.hsfiles 。它们是什么,为什么创建它们以及为什么使用它们? 最佳答案 您可以看到here相关代码: ba
我在关注 stack guide我得到了一个新的项目设置(耶!)。 它生成了以下文件布局: . ├── app │ ├── Main.hs ├── .gitignore ├── LICENSE ├
我在我的项目中创建了一个类(javahelp),用于在单击帮助内容(菜单项)时获取java帮助文件。我更新了目录中安装的 javahelp-2.0_05 C:\Program Files\javahe
我有一个巨大的 haskell 文件,可以毫无问题地编译和运行。我想将一些函数和类型定义放在通用 hs 文件中的单独模块中,然后将其导入我的主模块中。虽然主程序编译没有任何错误(它也编译导入的模块),
我正在阅读 Parallel and Concurrent Programming in Haskell 的第 11 章.在那本书中,以下 program用于同时获取 URL: main = do
我一直在尝试在我的 Haskell 项目中使用glpk-hs,但我完全不知道如何去做。我正在使用 stack 来运行我的程序,我的困惑很可能来自于我对其工作原理缺乏了解。 话虽这么说,我已经下载了gl
我遵循此处的指示: https://haskell.fpcomplete.com/tutorial/stack-script 我有一个具有以下文件结构的 cabal/stack 项目: app/Mai
我在基于 arch 的系统之上使用 xmonad。但是升级后我无法通过编译我的 xmonad.hs 文件 $ xmonad --recompile 我似乎在 xmonad 的每个模块上都出错,甚至 i
encoding包使用 HaXml在其构建脚本中(在 Setup.hs 中)。它碰巧使用了在 HaXml-1.19 和 HaXml-1.22 之间更改的接口(interface)位。如果编码包能够使用
我正在使用 Notepad++ 和 WinGHCi 做一些功课,我必须定义一个小“数据库”。格式是任意的,我认为这不是我要去的地方。无论如何,这是我在 *.hs 文件中使用的内容: let stude
我的项目结构如下: ~/.../project_name project_name.cabal Setup.hs src/ Main.hs Data/ ... test
我需要找出在 Setup.hs 中构建的标志(在 cabal 文件中找到的类型)。我想我正在寻找类型为 FlagAssignment 的东西。我怎样才能得到这个? 最佳答案 idris 必须这样做,y
我将 stack.exe v0.1.2.2 放入我的 PATH 目录之一,然后通过 stack setup 安装 ghc-7.8.4。 我尝试安装 ghc-mod 并获得奇怪的错误 Setup.hs:
我正在尝试比较两个不同的目录,其中有很多目录 du -hs User/data/dbs/* home/common/dbs/* 结果是两个目录依次显示出来。 12gb User/data/dbs/fi
我得到了一个 Setup.hs 文件,以及另一个包含源代码的 .hs 文件。每当我尝试通过以下方式配置安装时: runhaskell Setup.hs configure 我得到错误: Setup:
我制作了一个名为time.hs的文件。它包含一个测量另一个函数执行时间的函数。 有没有办法将time.hs文件导入到另一个Haskell脚本中? 我想要这样的东西: module Main where
我试图从 ~/.xmonad/Tree.hs 导入一个名为“Tree.hs”的自定义本地模块,但我得到: Error detected while loading xmonad configurati
组合在 Haskell 中是很正常的,但我只知道我可以定义组合函数的特殊行为,说 Prelude> (floor . sqrt) (10^55) 316227766016837936511293849
我目前正在完成我的第一个 Haskell 项目,在工作的最后一步,在连接不同的 Haskell 文件后,我的 I/O 函数似乎表现得很奇怪。 我有一个主文件(f1.hs),它加载多媒体库的一些信息并将
我正在使用山狮。我打开终端,然后加载 ghci , 我写 :l然后我尝试通过用鼠标将文件从我的桌面拖动到终端来加载我的文件(在我的桌面上),所以我知道位置是正确的,我明白了,提前谢谢你: Prelud
我是一名优秀的程序员,十分优秀!