aboutsummaryrefslogtreecommitdiff
path: root/bbcode/src/Text
diff options
context:
space:
mode:
Diffstat (limited to 'bbcode/src/Text')
-rw-r--r--bbcode/src/Text/BBCode.hs48
1 files changed, 43 insertions, 5 deletions
diff --git a/bbcode/src/Text/BBCode.hs b/bbcode/src/Text/BBCode.hs
index 7a328a8..dfa1db7 100644
--- a/bbcode/src/Text/BBCode.hs
+++ b/bbcode/src/Text/BBCode.hs
@@ -1,14 +1,52 @@
1{-# LANGUAGE OverloadedStrings #-} 1{-# LANGUAGE OverloadedStrings #-}
2{-# LANGUAGE DeriveGeneric #-}
2 3
3module Text.BBCode 4module Text.BBCode
4 ( 5 ( TreeError(..)
6 , rose
7 , matches
5 ) where 8 ) where
6 9
7import Data.Attoparsec.Text
8
9import Data.Text (Text) 10import Data.Text (Text)
10import qualified Data.Text as T (singleton, head, tail)
11 11
12import Control.Applicative 12import GHC.Generics (Generic)
13import Control.Exception (Exception)
14import Data.Typeable (Typeable)
15
16import Control.Monad (unless)
13 17
14import Text.BBCode.Lexer (BBToken(..), token) 18import Text.BBCode.Lexer (BBToken(..), token)
19
20import Data.Tree
21import Data.Tree.Zipper (TreePos, Empty, Full)
22import qualified Data.Tree.Zipper as Z
23
24data TreeError = ImbalancedTags Text Text
25 | LeftoverClose Text
26 deriving (Show, Eq, Generic, Typeable)
27
28instance Exception TreeError
29
30matches :: Text -> Text -> Bool
31-- ^ @`matches` "open" "close"@ should be true iff @[/close]@ is a valid closing tag for @[open]@
32--
33-- Until we allow for attributes this is equality according to `(==)`
34matches = (==)
35
36rose :: [BBToken] -> Either TreeError (Forest Text)
37-- ^ Assuming that both tags and content have the same type (we use 'Text') bbcode is a flat representation of a rose tree
38rose = fmap Z.toForest . flip rose' (Z.fromForest [])
39 where
40 rose' :: [BBToken] -> TreePos Empty Text -> Either TreeError (TreePos Empty Text)
41 rose' [] = return
42 rose' (x:xs) = (>>= rose' xs) . rose'' x
43
44 rose'' (BBStr t) = return . Z.nextSpace . Z.insert (Node t [])
45 rose'' (BBOpen t) = return . Z.children . Z.insert (Node t [])
46 rose'' (BBClose t) = close t -- for more pointless
47
48 close :: Text -> TreePos Empty Text -> Either TreeError (TreePos Empty Text)
49 close tag pos = do
50 pos' <- maybe (Left $ LeftoverClose tag) Right $ Z.parent pos
51 unless (Z.label pos' `matches` tag) . Left $ ImbalancedTags (Z.label pos') tag -- The structure shows that this mode of failure is not logically required -- it's just nice to have
52 return $ Z.nextSpace pos'