11
10

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

HaskellAdvent Calendar 2023

Day 5

Monomer という Haskell のハイレベル GUI ライブラリ

Last updated at Posted at 2023-12-30

Windows で動作する GUI アプリケーションを作りたくて、でも Windows では開発したくないのでクロスプラットフォームな GUI ライブラリを模索し、Monomer にたどり着きました。

結果、このライブラリでは要件を満たしていないため使わないことにしましたが、備忘録です。

Monomer とは

内部的に SDL2 を利用していて、その低レベルっぷりを隠蔽し、Elm アーキテクチャに昇華させている良いライブラリだと思いました。

以下のコードで以下のようなアプリが起動できます。

{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TemplateHaskell #-}

module Main where

import Debug.Trace

import Control.Lens
import Data.Maybe
import Data.Text (Text)
import Monomer
import TextShow

import Monomer.Event.Util
import qualified Monomer.Lens as L

newtype AppModel =
  AppModel
    { _command :: Text
    }
  deriving (Eq, Show)

data AppEvent
  = AppInit
  | GotFocus Path
  | LostFocus Path
  deriving (Eq, Show)

makeLenses 'AppModel

buildUI ::
     WidgetEnv AppModel AppEvent -> AppModel -> WidgetNode AppModel AppEvent
buildUI wenv model = widgetTree
  where
    widgetTree =
      vstack
        [ textField_ command [onFocus GotFocus, onBlur LostFocus] `nodeKey`
          "textField"
        ] `styleBasic`
      [padding 10]

handleEvent ::
     WidgetEnv AppModel AppEvent
  -> WidgetNode AppModel AppEvent
  -> AppModel
  -> AppEvent
  -> [AppEventResponse AppModel AppEvent]
handleEvent wenv node model evt =
  case evt of
    AppInit -> []
    GotFocus _ -> trace "GotFocus" [Model $ model & command .~ "Focused!"] -- ここに追加
    LostFocus _ -> trace "LostFocus" [Model $ model & command .~ "Blurred!"] -- ここに追加

main :: IO ()
main = do
  startApp model handleEvent buildUI config
  where
    config =
      [ appTheme darkTheme
      , appFontDef "Regular" "./assets/fonts/Roboto-Regular.ttf"
      , appWindowBorder False
      , appWindowResizable False
      , appInitEvent AppInit
      ]
    model = AppModel ""

スクリーンショット 2023-12-31 1.05.06.png

Elm アーキテクチャを採用しているため、AppModel というモデルと AppEvent というイベントを定義し、buildUI で GUI を定義、handleEvent でイベントを処理するという感じです。

やりたかったこと

このアプリでやりたいことがいくつかあります。

アプリのウィンドウボーダーをなくす

アプリのコンフィグに appWindowBorder False を指定することでなくせます。未指定だと以下のようになります。

スクリーンショット 2023-12-31 1.09.53.png

フォーカスロストイベントのハンドリング

ウィンドウボーダーをなくしたことで、アプリを終了させることができません。なのでアプリがバックグラウンドに行ったら終了させます。このためにはそれなりのイベントを取得しなければなりません。

テキストフィールドのイベントとして GotFocusLostFocus を定義します(GotFocus は検証用)。

data AppEvent
  = AppInit
  | GotFocus Path
  | LostFocus Path
  deriving (Eq, Show)

テキストフィールドにイベントハンドラを定義します。

        [ textField_ command [onFocus GotFocus, onBlur LostFocus] `nodeKey`
          "textField"
        ] `styleBasic`

handleEvent でこれらのイベントを処理するコードを記載します。

handleEvent wenv node model evt =
  case evt of
    AppInit -> []
    GotFocus _ -> trace "GotFocus" [Model $ model & command .~ "Focused!"] -- ここに追加
    LostFocus _ -> trace "LostFocus" [Model $ model & command .~ "Blurred!"] -- ここに追加

これでうまくいくはずだったんですが、アプリをバックグラウンドにしてもイベントが発生しません。このイベントはどうやら同じアプリ内でフォーカスが移動しないと発生しないようで、このやり方ではできませんでした。

どうやらこのイベント処理は Monomer 本体にパッチを当てないと実現できません。

diff --git a/src/Monomer/Main/Core.hs b/src/Monomer/Main/Core.hs
index 7e2b46dd..fcec634f 100644
--- a/src/Monomer/Main/Core.hs
+++ b/src/Monomer/Main/Core.hs
@@ -56,6 +56,7 @@ import Monomer.Helper (catchAny, putStrLnErr, isGhciRunning)
 import Monomer.Widgets.Composite
 
 import qualified Monomer.Lens as L
+import Control.Monad (forM_)
 
 {-|
 Type of response an App event handler can return, with __s__ being the model and
@@ -333,6 +334,15 @@ mainLoop window fontManager config loopArgs = do
   let invertY = fromMaybe False (_apcInvertWheelY config)
   let convertCfg = ConvertEventsCfg _mlOS dpr epr invertX invertY
   let baseSystemEvents = convertEvents convertCfg mousePos eventsPayload
+  let keyboardFocusEventHandlers = _apcKeyboardFocusEvent config
+
+  let windowKeyboardFocusGained = isWindowKeyboardFocusGained eventsPayload
+  when windowKeyboardFocusGained $
+    forM_ keyboardFocusEventHandlers $ \handler -> return $ handler True
+
+  let windowKeyboardFocusLost = isWindowKeyboardFocusLost eventsPayload
+  when windowKeyboardFocusLost $
+    forM_ keyboardFocusEventHandlers $ \handler -> return $ handler False
 
 --  when newSecond $
 --    liftIO . putStrLnErr $ "Frames: " ++ show _mlFrameCount
@@ -449,6 +459,20 @@ mainLoop window fontManager config loopArgs = do
     void $ handleWidgetDispose newWenv newRoot
 
   unless shouldQuit (mainLoop window fontManager config newLoopArgs)
+ where
+  isWindowKeyboardFocusGained :: [SDL.EventPayload] -> Bool
+  isWindowKeyboardFocusGained eventsPayload = any isKeyboardFocusGainedEvent eventsPayload
+
+  isKeyboardFocusGainedEvent :: SDL.EventPayload -> Bool
+  isKeyboardFocusGainedEvent SDL.WindowGainedKeyboardFocusEvent{} = True
+  isKeyboardFocusGainedEvent _ = False
+
+  isWindowKeyboardFocusLost :: [SDL.EventPayload] -> Bool
+  isWindowKeyboardFocusLost eventsPayload = any isKeyboardFocusLostEvent eventsPayload
+
+  isKeyboardFocusLostEvent :: SDL.EventPayload -> Bool
+  isKeyboardFocusLostEvent SDL.WindowLostKeyboardFocusEvent{} = True
+  isKeyboardFocusLostEvent _ = False
 
 {-
 Attempts to initialize a GL context in a separate OS thread to handle rendering
diff --git a/src/Monomer/Main/Types.hs b/src/Monomer/Main/Types.hs
index d8826889..df29b9d6 100644
--- a/src/Monomer/Main/Types.hs
+++ b/src/Monomer/Main/Types.hs
@@ -195,6 +195,8 @@ data AppConfig s e = AppConfig {
   _apcExitEvent :: [e],
   -- | Resize event handler.
   _apcResizeEvent :: [Rect -> e],
+  -- | Keyboard focus event handler.
+  _apcKeyboardFocusEvent :: [Bool -> e],
   -- | Defines which mouse button is considered main.
   _apcMainButton :: Maybe Button,
   -- | Defines which mouse button is considered secondary or context button.
@@ -229,6 +231,7 @@ instance Default (AppConfig s e) where
     _apcDisposeEvent = [],
     _apcExitEvent = [],
     _apcResizeEvent = [],
+    _apcKeyboardFocusEvent = [],
     _apcMainButton = Nothing,
     _apcContextButton = Nothing,
     _apcInvertWheelX = Nothing,
@@ -255,6 +258,7 @@ instance Semigroup (AppConfig s e) where
     _apcDisposeEvent = _apcDisposeEvent a1 ++ _apcDisposeEvent a2,
     _apcExitEvent = _apcExitEvent a1 ++ _apcExitEvent a2,
     _apcResizeEvent = _apcResizeEvent a1 ++ _apcResizeEvent a2,
+    _apcKeyboardFocusEvent = _apcKeyboardFocusEvent a1 ++ _apcKeyboardFocusEvent a2,
     _apcMainButton = _apcMainButton a2 <|> _apcMainButton a1,
     _apcContextButton = _apcContextButton a2 <|> _apcContextButton a1,
     _apcInvertWheelX = _apcInvertWheelX a2 <|> _apcInvertWheelX a1,
@@ -439,6 +443,13 @@ appResizeEvent evt = def {
   _apcResizeEvent = [evt]
 }
 
+-- | Keyboard focus event handler.
+-- True if the application has keyboard focus, False otherwise.
+appKeyboardFocusEvent :: (Bool -> e) -> AppConfig s e
+appKeyboardFocusEvent evt = def {
+  _apcKeyboardFocusEvent = [evt]
+}
+
 -- | Defines which mouse button is considered main.
 appMainButton :: Button -> AppConfig s e
 appMainButton btn = def {

Monomer の SDL2 メインループでキーボードフォーカスイベントを補足、コンフィグでキーボードフォーカスイベントのハンドラを設定する、というような感じにします。何度か試してみましたが、稀にフォーカスを失っていないのに失ったイベントが発生してしまうのが厄介です。原因はわかっていません。

スクロール

これで後はアプリを実装するだけかなと思ったのですが、Monomer の、というか SDL2 のスクロールだと思いますが非常に使い勝手が悪いです。

hAW2LSRz.gif

macOS ならスクロールに慣性がかかって欲しいのですが、引っかかるというかピタッと止まるというか、とにかく好みのスクロール挙動ではありません。Windows なら慣性スクロールとかはないと思うのですが、macOS でも使うかもしれないツールなので、ここが気に入らず Monomer を諦めました。

まとめ

Elm アーキテクチャで提供された GUI ライブラリなのでものすごく期待していて、今でも後ろ髪を引かれる思いですが、別の GUI ライブラリを探す旅に出なければなりません。ただ、GUI ライブラリの評価ポイントがわかっただけでも今回は良しとします。

11
10
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
11
10

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?