Created
          September 16, 2015 16:03 
        
      - 
      
- 
        Save jooyunghan/f818437fca9024964468 to your computer and use it in GitHub Desktop. 
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
  | 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