2014-12-09 17:33:32 +01:00
|
|
|
window.Nikoli = {}
|
2014-12-08 17:33:04 +01:00
|
|
|
|
2014-12-09 12:30:29 +01:00
|
|
|
class Nikoli.Game
|
2014-12-09 12:47:40 +01:00
|
|
|
constructor: (@board, @name, @url) ->
|
2014-12-09 12:30:29 +01:00
|
|
|
@name = 'nikoli' unless @name?
|
2014-12-09 12:47:40 +01:00
|
|
|
@url = "/data/#{@name}" unless @url?
|
|
|
|
|
2014-12-09 12:30:29 +01:00
|
|
|
@board.classList.add @name
|
|
|
|
|
|
|
|
@grid = document.createElement 'div'
|
|
|
|
@grid.classList.add 'game-container'
|
|
|
|
@board.appendChild @grid
|
|
|
|
|
2014-12-09 12:47:40 +01:00
|
|
|
@getFiles()
|
|
|
|
|
2014-12-09 12:30:29 +01:00
|
|
|
buttons_div = document.createElement 'div'
|
|
|
|
buttons = {check: 'Check', reset: 'Reset', newgame: 'New game', help: '?'}
|
|
|
|
|
|
|
|
for k,v of buttons
|
|
|
|
button = document.createElement 'button'
|
|
|
|
button.innerHTML = v
|
|
|
|
button.classList.add k
|
|
|
|
|
|
|
|
buttons_div.appendChild button
|
|
|
|
|
|
|
|
@board.appendChild buttons_div
|
|
|
|
|
|
|
|
@board.querySelector('.check').addEventListener('click', @check.bind(this))
|
|
|
|
@board.querySelector('.reset').addEventListener('click', @reset.bind(this))
|
2014-12-09 12:47:40 +01:00
|
|
|
@board.querySelector('.newgame').addEventListener('click', @newgame.bind(this))
|
2014-12-09 12:30:29 +01:00
|
|
|
|
2014-12-11 22:29:46 +01:00
|
|
|
congratulations = document.createElement 'div'
|
|
|
|
congratulations.innerHTML = 'Congratulations!'
|
|
|
|
congratulations.classList.add 'congratulations'
|
|
|
|
congratulations.classList.add 'hide'
|
|
|
|
@board.appendChild congratulations
|
|
|
|
|
2014-12-09 12:30:29 +01:00
|
|
|
check: ->
|
|
|
|
errors = @errors()
|
|
|
|
|
|
|
|
if errors.length == 0
|
2014-12-11 22:29:46 +01:00
|
|
|
@board.querySelector('.congratulations').classList.add('show')
|
2014-12-09 12:30:29 +01:00
|
|
|
else
|
|
|
|
alert errors.map((el) -> el.message).join()
|
|
|
|
|
2014-12-12 12:18:22 +01:00
|
|
|
generate: (game, solution = false, cell_class = Nikoli.Cell) ->
|
|
|
|
@game = game if game?
|
|
|
|
|
|
|
|
@grid.innerHTML = ''
|
|
|
|
@game.forEach((row, i) =>
|
|
|
|
row_elem = new Nikoli.Row().create()
|
|
|
|
row.forEach((cell, j) ->
|
|
|
|
row_elem.appendChild new cell_class(i, j).create(cell))
|
|
|
|
|
|
|
|
@grid.appendChild row_elem)
|
|
|
|
|
|
|
|
return
|
|
|
|
|
2014-12-09 12:47:40 +01:00
|
|
|
getFiles: ->
|
|
|
|
xmlhttp = new XMLHttpRequest()
|
|
|
|
xmlhttp.open("GET", "#{@url}.json")
|
|
|
|
|
|
|
|
xmlhttp.addEventListener('load', (evt) =>
|
|
|
|
@setFiles JSON.parse(evt.target.responseText))
|
|
|
|
xmlhttp.send()
|
|
|
|
|
|
|
|
setFiles: (files) ->
|
|
|
|
@files = files
|
|
|
|
@file = @files[0]
|
|
|
|
|
|
|
|
@newgame() unless @game?
|
|
|
|
|
|
|
|
newgame: ->
|
2014-12-11 22:29:46 +01:00
|
|
|
@board.querySelector('.congratulations').classList.remove('show')
|
|
|
|
|
2014-12-09 12:47:40 +01:00
|
|
|
xmlhttp = new XMLHttpRequest()
|
|
|
|
xmlhttp.open("GET", "#{@url}/#{@file}.json")
|
|
|
|
|
|
|
|
xmlhttp.addEventListener('load', (evt) =>
|
|
|
|
@generate JSON.parse(evt.target.responseText))
|
|
|
|
xmlhttp.send()
|
|
|
|
|
2014-12-09 12:30:29 +01:00
|
|
|
reset: ->
|
|
|
|
@generate()
|
|
|
|
|
2014-12-12 12:18:22 +01:00
|
|
|
class Nikoli.Row
|
|
|
|
create: ->
|
|
|
|
row = document.createElement 'div'
|
|
|
|
row.classList.add 'grid-row'
|
|
|
|
|
|
|
|
row
|
|
|
|
|
2014-12-09 17:33:32 +01:00
|
|
|
class Nikoli.Cell
|
|
|
|
constructor: (@x, @y, @game) ->
|
2014-12-12 12:18:22 +01:00
|
|
|
@value = @game[@x][@y] if @game? && @valid()
|
|
|
|
|
|
|
|
create: (value) ->
|
|
|
|
cell = document.createElement 'div'
|
|
|
|
cell.dataset.row = @x
|
|
|
|
cell.dataset.column = @y
|
|
|
|
|
|
|
|
cell.classList.add 'grid-cell'
|
|
|
|
|
|
|
|
cell.innerHTML = ' '
|
|
|
|
|
|
|
|
cell
|
2014-12-09 17:33:32 +01:00
|
|
|
|
|
|
|
toString: -> "#{@x};#{@y}"
|
|
|
|
|
2014-12-11 17:26:08 +01:00
|
|
|
getColumn: ->
|
|
|
|
column = []
|
|
|
|
column.push @game[i][@y] for i in [0...@game.length]
|
|
|
|
column
|
|
|
|
|
|
|
|
getRow: -> @game[@x]
|
|
|
|
|
2014-12-09 17:33:32 +01:00
|
|
|
adjacentCells: ->
|
2014-12-12 12:23:25 +01:00
|
|
|
constructor = Object.getPrototypeOf(this).constructor
|
2014-12-09 17:33:32 +01:00
|
|
|
[
|
2014-12-12 12:23:25 +01:00
|
|
|
new constructor(@x + 1, @y, @game),
|
|
|
|
new constructor(@x - 1, @y, @game),
|
|
|
|
new constructor(@x, @y + 1, @game),
|
|
|
|
new constructor(@x, @y - 1, @game)
|
2014-12-09 17:33:32 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
valid: (value) ->
|
|
|
|
0 <= @x < @game.length && 0 <= @y < @game[@x].length &&
|
|
|
|
(!value? || value < 0 && @game[@x][@y] < 0 || value >= 0 && @game[@x][@y] >= 0)
|
|
|
|
|
2014-12-10 12:52:00 +01:00
|
|
|
duplicatesIn: (array) ->
|
|
|
|
array.filter((cell) => cell == @value).length > 1
|
|
|
|
|
|
|
|
columnDuplicates: ->
|
2014-12-11 17:26:08 +01:00
|
|
|
@duplicatesIn @getColumn()
|
2014-12-10 12:52:00 +01:00
|
|
|
|
|
|
|
rowDuplicates: ->
|
2014-12-11 17:26:08 +01:00
|
|
|
@duplicatesIn @getRow()
|
2014-12-10 12:52:00 +01:00
|
|
|
|
2014-12-10 12:58:02 +01:00
|
|
|
squareDuplicates: (from, size) ->
|
|
|
|
square = []
|
|
|
|
for i in [from.x...(from.x + size)]
|
|
|
|
for j in [from.y...(from.y + size)]
|
|
|
|
square.push @game[i][j]
|
|
|
|
|
|
|
|
@duplicatesIn square
|
|
|
|
|
2014-12-08 17:33:04 +01:00
|
|
|
class Nikoli.Stream
|
|
|
|
constructor: (@game) ->
|
|
|
|
@cells = []
|
|
|
|
|
|
|
|
calculate: (cell) ->
|
|
|
|
@cells = []
|
2014-12-09 17:33:32 +01:00
|
|
|
@type = if cell.value < 0 then 'black' else 'white'
|
2014-12-08 17:33:04 +01:00
|
|
|
|
|
|
|
cells_to_process = [cell]
|
|
|
|
|
|
|
|
while cells_to_process.length > 0
|
|
|
|
cell = cells_to_process.pop()
|
|
|
|
cells_to_process = cells_to_process.concat @process(cell) unless @include(cell)
|
|
|
|
|
|
|
|
@cells
|
|
|
|
|
|
|
|
empty: ->
|
|
|
|
@cells.length == 0
|
|
|
|
|
|
|
|
include: (cell) ->
|
2014-12-09 17:33:32 +01:00
|
|
|
@cells.indexOf(cell.toString()) >= 0
|
2014-12-08 17:33:04 +01:00
|
|
|
|
|
|
|
length: ->
|
|
|
|
@cells.length
|
|
|
|
|
|
|
|
process: (cell) ->
|
2014-12-09 17:33:32 +01:00
|
|
|
@cells.push(cell.toString())
|
|
|
|
|
|
|
|
cell.adjacentCells().filter((adj_cell) -> adj_cell.valid(cell.value))
|