/hsc3-tree/src/lib/Sound/SC3/Tree/Type.hs
http://github.com/8c6794b6/haskell-sc-scratch · Haskell · 381 lines · 239 code · 51 blank · 91 comment · 37 complexity · d08ba27543f57f84ee780f82a39691ab MD5 · raw file
- {-# LANGUAGE DeriveDataTypeable #-}
- {-|
- Module : $Header$
- CopyRight : (c) 8c6794b6
- License : BSD3
- Maintainer : 8c6794b6@gmail.com
- Stability : unstable
- Portability : unknown
- Representation of scsynth node tree.
- -}
- module Sound.SC3.Tree.Type
- ( -- * Types
- SCNode(..)
- , SynthParam(..)
- , nodeId
- , synthName
- , synthParams
- , isSynth
- , isGroup
- , mapSCNode
- , filterSCNode
- -- * Parser
- , parseNode
- -- * Converter
- , treeToNew
- , treeToNewWith
- , treeToSet
- , paramToTuple
- , drawSCNode
- , renderSCNode
- , prettyDump
- -- * Util
- , paramName
- , paramValue
- , updateParams
- , nodeIds
- , hasUniqueIds
- ) where
- {-
- XXX:
- This package may use /Query_Node/ data type from "Sound.SC3.Status" when the
- data type become available.
- -}
- import Control.DeepSeq (NFData(..), deepseq)
- import Control.Monad
- import Data.ByteString.Char8 (unpack)
- import Data.Function (on)
- import Data.Data (Data,Typeable)
- import Data.List (unionBy)
- import Text.PrettyPrint hiding (int, double)
- import Sound.SC3
- import Sound.OSC hiding (int32, string)
- import Sound.SC3.Parser.Datum
- import qualified Text.PrettyPrint as P
- import qualified Data.IntSet as IS
- ------------------------------------------------------------------------------
- --
- -- Types
- --
- ------------------------------------------------------------------------------
- -- | Data type for representing Group and Synth node in scsynth.
- data SCNode = Group {-# UNPACK #-} !Int [SCNode]
- -- ^ Group node
- | Synth {-# UNPACK #-} !Int String [SynthParam]
- -- ^ Synth node
- deriving (Eq,Read,Show,Data,Typeable)
- instance NFData SCNode where
- rnf n = case n of
- Group i ns -> i `seq` ns `deepseq` ()
- Synth i name ps -> i `seq` name `deepseq` ps `deepseq` ()
- -- | Data type for synth param.
- data SynthParam = String := Double
- -- ^ Double value
- | String :<- Int
- -- ^ Mapped control bus id
- | String :<= Int
- -- ^ Mapped audio bus id
- deriving (Eq,Read,Data,Typeable)
- instance Show SynthParam where
- show (f:=x) = show f ++ ":=" ++ show x
- show (f:<-x) = show f ++ ":<-" ++ show x
- show (f:<=x) = show f ++ ":<=" ++ show x
- instance NFData SynthParam where
- rnf p = case p of
- n := v -> n `deepseq` v `seq` ()
- n :<- v -> n `deepseq` v `seq` ()
- n :<= v -> n `deepseq` v `seq` ()
- infixr 5 :=
- infixr 5 :<-
- infixr 5 :<=
- -- | 'True' if given node is synth.
- isSynth :: SCNode -> Bool
- isSynth n = case n of Synth _ _ _ -> True; _ -> False
- -- | 'True' if given node is group.
- isGroup :: SCNode -> Bool
- isGroup = not . isSynth
- -- | Returns node id of synth and group node.
- nodeId :: SCNode -> Int
- nodeId (Group i _) = i
- nodeId (Synth i _ _) = i
- -- | Get name from 'Synth' constructor. Group nodes will return empty
- -- string.
- synthName :: SCNode -> String
- synthName n = case n of
- Synth _ n' _ -> n'
- _ -> ""
- -- | Get 'SynthParam' of given synth node. Returns empty list for group node.
- synthParams :: SCNode -> [SynthParam]
- synthParams n = case n of
- Synth _ _ ps -> ps
- _ -> []
- -- | Map given function to 'SCNode'.
- mapSCNode :: (SCNode -> SCNode) -> SCNode -> SCNode
- mapSCNode f n0 =
- let g n = case n of
- Synth {} -> f n
- Group i ns -> f $ Group i $ foldr (\m ms -> g m : ms) [] ns
- in g n0
- {-# INLINEABLE mapSCNode #-}
- -- | Filter 'SCNode' with given condition.
- filterSCNode :: (SCNode -> Bool) -> SCNode -> SCNode
- filterSCNode p n0 =
- let f n acc =
- case n of
- Group i ns | p n -> Group i (foldr f [] ns) : acc
- | otherwise -> acc
- Synth {} | p n -> n : acc
- | otherwise -> acc
- in head $ foldr f [] [n0]
- {-# INLINEABLE filterSCNode #-}
- -- | Parse osc message returned from \"/g_queryTree\" and returns haskell
- -- representation of scsynth node tree.
- parseNode :: Message -> SCNode
- parseNode o = case o of
- Message "/g_queryTree.reply" ds -> case parseDatum parseGroup ds of
- Right tree -> tree
- Left err -> error $ show err
- _ -> error "not a /g_queryTree.reply response"
- {-# INLINEABLE parseNode #-}
- --
- -- With using simple parser without parsec
- --
- -- parseOSC :: OSC -> SCNode
- -- parseOSC o = case o of
- -- Message "/g_queryTree.reply" ds -> fst $ head $ parse parseGroup (tail ds)
- -- _ -> error "not a g_queryTree.reply message"
- parseGroup :: DatumParser SCNode
- parseGroup = do
- flag <- int32
- case flag of
- 1 -> parseGroupWith parseSynth
- _ -> parseGroupWith parseSynthWithoutParams
- {-# INLINEABLE parseGroup #-}
- parseGroupWith :: (Int -> DatumParser SCNode) -> DatumParser SCNode
- parseGroupWith synthParser = do
- nid <- int32
- numChild <- int32
- if numChild < 0
- then synthParser (fromIntegral nid)
- else Group (fromIntegral nid) `fmap`
- replicateM (fromIntegral numChild) (parseGroupWith synthParser)
- {-# INLINEABLE parseGroupWith #-}
- parseSynth :: Int -> DatumParser SCNode
- parseSynth nId = do
- name <- unpack `fmap` string
- numParams <- int32
- params <- replicateM (fromIntegral numParams) parseParam
- return $ Synth nId name params
- {-# INLINEABLE parseSynth #-}
- parseSynthWithoutParams :: Int -> DatumParser SCNode
- parseSynthWithoutParams nid = do
- name <- unpack `fmap` string
- return $ Synth nid name []
- {-# INLINEABLE parseSynthWithoutParams #-}
- -- | Parse parameter values for each synth.
- parseParam :: DatumParser SynthParam
- parseParam = do
- name <- unpack `fmap` string
- val <- datum
- case val of
- Float x -> return $ name := realToFrac x
- Double x -> return $ name := x
- ASCII_String xs ->
- let xs' = unpack xs
- in case xs' of
- 'c':rest -> return $ name :<- read rest
- 'a':rest -> return $ name :<= read rest
- _ -> error $ "Unknown param: " ++ xs'
- Int32 x -> return $ name := fromIntegral x
- e -> error $ "Cannot make param from: " ++ show e
- {-# INLINEABLE parseParam #-}
- ------------------------------------------------------------------------------
- --
- -- Converting functions
- --
- ------------------------------------------------------------------------------
- -- | SCNode to [OSC] for creating new nodes.
- --
- -- OSC list contains \"g_new\", \"s_new\", and \"n_map\" messages to build
- -- the given SCNode. New node will be added to tail of target id.
- --
- treeToNew :: Int -- ^ Target node id
- -> SCNode -- ^ New nodes
- -> [Message]
- treeToNew = treeToNewWith AddToTail
- treeToNewWith :: AddAction -- ^ Add action for this node
- -> Int -- ^ Target node id
- -> SCNode -- ^ New node to add
- -> [Message]
- treeToNewWith aa tId tree = f tId tree
- where
- f i t = case t of
- Group j ns -> g_new [(j,aa,i)]:concatMap (f j) ns
- Synth j n ps -> s_new n j aa i (concatMap paramToTuple ps):g j ps
- g i xs | not (null cs) && not (null as) = [n_map i cs,n_mapa i as]
- | not (null cs) && null as = [n_map i cs]
- | null cs && not (null as) = [n_mapa i as]
- | otherwise = []
- where (cs,as) = foldr h ([],[]) xs
- h x (cs,as) = case x of
- (name:<-bus) -> ((name,bus):cs,as)
- (name:<=bus) -> (cs,(name,bus):as)
- _ -> (cs,as)
- -- | SCNode to [OSC] for updating nodes.
- --
- -- OSC list contains \"n_set\" and \"n_map\" messages to set parameters.
- --
- treeToSet :: SCNode -- ^ Node with new parameters for already exisitng nodes.
- -> [Message]
- treeToSet tree = f tree
- where
- f t = case t of
- Group _ ns -> concatMap f ns
- Synth _ _ [] -> []
- Synth j _ ps -> n_set j (concatMap paramToTuple ps):g j ps
- g k ps | not (null cs) && not (null as) = [n_map k cs,n_mapa k as]
- | not (null cs) && null as = [n_map k cs]
- | null cs && not (null as) = [n_mapa k as]
- | otherwise = []
- where (cs,as) = foldr h ([],[]) ps
- h x (cs,as) = case x of
- (name:<-bus) -> ((name,bus):cs,as)
- (name:<=bus) -> (cs,(name,bus):as)
- _ -> (cs,as)
- ------------------------------------------------------------------------------
- --
- -- Utils
- --
- ------------------------------------------------------------------------------
- paramToTuple :: SynthParam -> [(String,Double)]
- paramToTuple (name := val) = [(name,val)]
- paramToTuple _ = []
- paramName :: SynthParam -> String
- paramName x = case x of
- (n := _) -> n
- (n :<- _) -> n
- (n :<= _) -> n
- -- | Converts value of 'SynthParam' to 'Double'. Information of mapped buses
- -- will be lost.
- paramValue :: SynthParam -> Double
- paramValue p = case p of
- _ := v -> v
- _ :<- v -> fromIntegral v
- _ :<= v -> fromIntegral v
- updateParams :: [SynthParam] -> SCNode -> SCNode
- updateParams ps node = case node of
- Synth i n ps' -> Synth i n (unionBy ((==) `on` paramName) ps ps')
- g -> g
- hasUniqueIds :: SCNode -> Bool
- hasUniqueIds n = listSize == setSize where
- setSize = IS.size . IS.fromList $ l
- listSize = length l
- l = nodeIds n
- nodeIds :: SCNode -> [Int]
- nodeIds n =
- let f x acc = case x of
- Synth nid _ _ -> nid : acc
- Group nid ns -> nid : foldr f acc ns
- in foldr f [] [n]
- ------------------------------------------------------------------------------
- --
- -- Pretty printers
- --
- ------------------------------------------------------------------------------
- -- | Draw SCNode data.
- drawSCNode :: SCNode -> String
- drawSCNode = renderSCNode True
- -- | Pretty prints SCNode in same format as '/g_dumpTree' OSC message.
- renderSCNode :: Bool -> SCNode -> String
- renderSCNode detail = render . n2doc where
- n2doc n = case n of
- Group i ns ->
- text "NODE TREE Group" <+> P.int i $$ vcat (map (nest 3 . n2doc') ns)
- _ -> n2doc' n
- n2doc' n = case n of
- Group i ns -> P.int i <+> text "group" $$ vcat (map (nest 3 . n2doc') ns)
- Synth i name ps ->
- P.int i <+> text name $$
- (if detail then hsep (map (nest 2 . p2doc) ps) else empty)
- p2doc p = case p of
- n:=v -> text n <> char ':' <+> P.double v
- n:<-v -> text n <> char ':' <+> char 'c' <> P.int v
- n:<=v -> text n <> char ':' <+> char 'a' <> P.int v
- -- | Dump SCNode. Dumped string could be parsed with 'read' function.
- prettyDump :: SCNode -> String
- prettyDump = render . n2doc where
- n2doc :: SCNode -> Doc
- n2doc n = case n of
- Group i ns ->
- text "Group" <+> signedInt i $$
- nest 2 (brackets $ vcat (punctuate comma (map n2doc ns)))
- Synth i name ps ->
- text "Synth" <+> signedInt i <+> doubleQuotes (text name) $$
- nest 2 (brackets $ vcat (punctuate comma (map p2doc ps)))
- p2doc p = case p of
- (k:=v) -> doubleQuotes (text k) <> text ":=" <> signedDouble v
- (k:<-v) -> doubleQuotes (text k) <> text ":<-" <> signedInt v
- (k:<=v) -> doubleQuotes (text k) <> text ":<=" <> signedInt v
- -- | Show signed int.
- signedInt :: Int -> Doc
- signedInt n | n < 0 = parens (P.int n)
- | otherwise = P.int n
- -- | Show signed double.
- signedDouble :: Double -> Doc
- signedDouble n | n < 0 = parens (P.double n)
- | otherwise = P.double n