Skip to content

Instantly share code, notes, and snippets.

@jooyunghan
Created September 16, 2015 16:03
Show Gist options
  • Save jooyunghan/f818437fca9024964468 to your computer and use it in GitHub Desktop.
Save jooyunghan/f818437fca9024964468 to your computer and use it in GitHub Desktop.
import Graphics.Collage exposing (filled, rect, collage, toForm)
import Graphics.Element exposing (image, beside, below, above, empty, Element, spacer, color)
import Graphics.Input exposing (customButton, button)
import Color exposing (red, yellow)
import Signal exposing ((<~))
import Markdown
off_bulb = collage 30 30 [
toForm (image 28 28 "")
]
on_bulb = collage 30 30 [
toForm (image 28 28 "")
]
type alias Switch = {row:Int, col:Int}
type Click = Reset | Toggle Switch
click : Signal.Mailbox Click
click = Signal.mailbox Reset
switch_up = collage 30 30 [toForm (image 20 20 "http://www.rw-designer.com/icon-image/10454-128x128x32.png")]
switch_hover = collage 30 30 [toForm (image 22 22 "http://www.rw-designer.com/icon-image/10455-128x128x32.png")]
switch_down = collage 30 30 [toForm (image 20 20 "http://www.rw-designer.com/icon-image/10456-128x128x32.png")]
switch : {row:Int, col:Int} -> Element
switch link = customButton (Signal.message click.address (Toggle link)) switch_up switch_hover switch_down
isOdd n = n % 2 == 1
isEven n = n % 2 == 0
-- model
type alias Cell = {row:Int, col:Int, on:Bool, target:Bool}
type alias Model = List (List Cell)
input = {n=5, on=[(0,4), (1,0)]}
model {n,on} =
List.foldl (\(row,col) model -> target (row*2+1) (col*2+1) model) (List.map (\row -> make_row row n) [1 .. 3]) on
make_row row cols = List.map (\col -> {row=row, col=col, on=False, target=False }) [1 .. (cols*2+1)]
model0 = model input
toggle : Int -> Int -> Model -> Model
toggle row col model =
List.map (\line -> List.map (\cell -> if .row cell == row && .col cell == col then {cell | on <- not (.on cell)} else cell) line ) model
target : Int -> Int -> Model -> Model
target row col model =
List.map (\line -> List.map (\cell -> if .row cell == row && .col cell == col then {cell | target <- True} else cell) line ) model
app_state = {model=model, count=0}
-- view
view model =
Markdown.toElement markdown
`above`
List.foldl above empty (List.map view_line model)
`above`
spacer 20 20
`above`
button (Signal.message click.address Reset) "Reset"
view_line line =
List.foldl beside empty (List.map view_cell line)
view_cell {row,col,on,target} =
if | isOdd row && isOdd col ->
let bulb = if on then on_bulb else off_bulb
in if target then color yellow bulb else bulb
| isEven row && isEven col -> spacer 30 30
| otherwise -> switch {row=row, col=col}
markdown = """
# 전구 켜기
각 전구들 사이에 있는 스위치는 이웃한 전구 두 개를 켜거나 끌 수 있다.
노란 테두리로 표시된 전구들만 모두 켜려면 스위치를 최소한 몇번 눌러야 할까?
"""
-- update
step click model =
case click of
Reset -> model0
Toggle {row,col} ->
if | isOdd row && isEven col -> toggle row (col+1) (toggle row (col-1) model)
| isEven row && isOdd col -> toggle (row-1) col (toggle (row+1) col model)
| otherwise -> model
main = view <~ Signal.foldp step model0 click.signal
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment