Skip to content

Instantly share code, notes, and snippets.

@suzuki-shin
Last active August 29, 2015 13:56
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save suzuki-shin/9015999 to your computer and use it in GitHub Desktop.
Save suzuki-shin/9015999 to your computer and use it in GitHub Desktop.
{-# OPTIONS_GHC -Wall #-}
import Data.Array (Array, listArray, assocs, (//), (!))
import Data.List (intersect, transpose)
import Data.Tree (Tree(Node), flatten)
import Data.Maybe (catMaybes)
input :: [String]
input =
[ " GYRR"
, "RYYGYG"
, "GYGYRR"
, "RYGYRG"
, "YGYRYG"
, "GYRYRG"
, "YGYRYR"
, "YGYRYR"
, "YRRGRG"
, "RYGYGG"
, "GRYGYR"
, "GRYGYR"
, "GRYGYR"
]
height :: Int
height = length input
width :: Int
width = length $ head input
type Board = Array Pos Mark
type Mark = Char
type Pos = (Int,Int) -- (y,x)
main :: IO ()
main = mapM_ printBoard $ puyopuyo $ toBoard input
printBoard :: Board -> IO ()
printBoard b = do
mapM_ print $ fromBoard b
putStrLn ""
toBoard :: [String] -> Board
toBoard [] = error "invalid parameter"
toBoard ss = listArray ((0,0), (height-1,width-1)) $ concat ss
fromBoard :: Board -> [String]
fromBoard = groupn width . map snd . assocs
positions :: Board -> [Pos]
positions = (map fst) . assocs
-- | 次の状態を返す
puyo :: Board -> Board
puyo b = fall $ deleteMark b $ concat $ deletable b []
-- | (初期状態から平衡状態までの連続的な)状態のリストを返す
puyopuyo :: Board -> [Board]
puyopuyo b = if b == puyo b
then []
else (b : puyopuyo (puyo b))
-- | 4つ以上同色で連なっているものの座標を返す
deletable :: Board -> [Pos] -> [[Pos]]
deletable b passed = filter ((>=4).length) $ map flatten $ catMaybes $ deletable' b passed $ positions b
where
deletable' :: Board -> [Pos] -> [Pos] -> [Maybe (Tree Pos)]
deletable' _ _ [] = []
deletable' b' passed' (p':ps') = (connectTree b' passed' p') : (deletable' b' (p':passed') ps')
deleteMark :: Board -> [Pos] -> Board
deleteMark board ps = board // [(p,' ')|p<-ps]
-- | 落下(' 'を下に詰める)した状態を返す
fall :: Board -> Board
fall = toBoard . transpose . paddingFront width " " . map (paddingFrontSpace height . deleteSpace) . transpose . fromBoard
where
deleteSpace = filter (/=' ')
-- | リストを定数個ごとに分割する
groupn :: Int -> [a] -> [[a]]
groupn _ [] = []
groupn n xs =
let (xs1, xs2) = splitAt n xs
in xs1 : groupn n xs2
-- | 文字列の先頭に" "を詰めて指定文字数のの文字列を返す
paddingFrontSpace :: Int -> String -> String
paddingFrontSpace n = paddingFront n ' '
-- | リストの先頭に指定した要素を詰めて、指定の数の要素数のリストを返す
paddingFront :: Int -> a ->[a] -> [a]
paddingFront n pad = reverse . take n . (++ (cycle [pad])) . reverse
-- | となり合った座標を返す
neigbors :: Pos -> [Pos]
neigbors (y,x) = [(y',x')|(x',y') <- [(x+1,y),(x-1,y),(x,y+1),(x,y-1)], 0 <= x', x' < width, 0 <= y', y' < height]
-- | 指定した座標のとなりで同色の座標リストを返す
connects :: Board -> Pos -> [Pos]
connects b p = (sameColors b p) `intersect` (neigbors p)
sameColors :: Board -> Pos -> [Pos]
sameColors b p = map fst $ filter (\(_,m) -> (m /= ' ') && (m == (b!p))) $ assocs b
-- | 繋がったマークのPosリストをツリーにして返す(一度通ったところは除外する)
-- >>> connectTree a [] (1,1)
-- Just (Node {rootLabel = (1,1), subForest = [Node {rootLabel = (1,2), subForest = []},Node {rootLabel = (2,1), subForest = [Node {rootLabel = (3,1), subForest = []}]}]})
connectTree :: Board -> [Pos] -> Pos -> Maybe (Tree Pos)
connectTree b passed p = if p `elem` passed
then Nothing
else Just $ Node p $ subTs $ connects b p
where
subTs :: [Pos] -> [Tree Pos]
subTs = catMaybes . map (connectTree b (p:passed))
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment