-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpuzzle.rb
138 lines (103 loc) · 3.33 KB
/
puzzle.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
require 'modularity'
# The data structure for the Sudoku Puzzle.
# Separate from the algorithm used to solve the puzzles.
class Puzzle
# The values of rows/columns that are part of the same nonet.
NONET_RANGES = [0..2, 3..5 ,6..8]
# The decimal values for a puzzle.
DECIMAL_REGEX = /^[0-9]$/
DECIMAL_VALUES = (1..9).to_a
#TODO: Yet to be implemented.
# The hex values for a puzzle.
HEX_REGEX = /^[a-zA-z0-9]$/
# The values of blank cells in the puzzle.
BLANKS_REGEX = /^[_\.]$/
# Names of class from: http://en.wikipedia.org/wiki/Sudoku#Terminology
require "./cell"
## Bring in other methods.
##
require './puzzle/creation_trait.rb'
require './puzzle/each_methods_trait.rb'
require "./puzzle/output_trait.rb"
require "./puzzle/houses_trait.rb"
does "puzzle/creation"
does "puzzle/output"
does "puzzle/each_methods"
# Returns the cell from the position [row][column] specified.
# Row and column numbers must be in the range for the base of the puzzle.
def cell(row_num, col_num)
return @puzzle[row_num][col_num]
end
does "puzzle/houses"
# Solves a valid sudoku puzzle.
def solve
# If the puzzle has already been solved, don't solve it again, just return.
return self if solved?
# If the puzzle is not valid, we'll error out.
@algorithm.solve(self) if valid?
self
# Catch puzzle validity errors
rescue => e
puts "Error: #{e.message}"
exit
end
# Add the current row to the puzzle.
def add_row
unless @current_row.nil?
# Add the row to the puzzle.
@puzzle << @current_row
# And clear the row.
@current_row = nil
end
end
# Add a cell to the current row of the puzzle.
def add_cell(cell)
# If there is no row yet, create a blank array for it.
@current_row ||= []
# Add the cell to the end of the row.
@current_row << cell
end
# Check to make sure the puzzle, as read in from the file, is valid and can be solved.
def valid?
# Valid puzzle has 9 rows.
unless @puzzle.size == 9
raise RuntimeError, "Sudoku puzzle must have 9 rows."
end
# Valid puzzle has 9 columns of 9 cells.
each_row do |row|
unless row.size == 9
raise RuntimeError, "Each row in the Sudoku puzzle must have 9 cells."
end
end
# Valid puzzle has valid cells.
each do |cell|
cell.valid?
end
#TODO: Find out if there are conflicts in row, column or nonet.
# If we get this far without error, the puzzle is valid.
return true
end
# Has the puzzle been solved? Meaning, do each of the rows, columns and nonets have values 1-9.
def solved?
# Return whether the total passed in is equal to the total for a valid house.
def is_valid_house_total?(total)
return total == 45
end
# Return the total of adding up all the cells in this house.
def house_total(house)
return house.inject(0) {|sum, cell| sum + cell.to_i }
end
## Check whether all the houses in the puzzle have valid totals.
##
each_row do |row|
return false unless is_valid_house_total?(house_total(row))
end
each_column do |column|
return false unless is_valid_house_total?(house_total(column))
end
each_nonet do |nonet|
return false unless is_valid_house_total?(house_total(nonet))
end
end
private :add_row, :add_cell, :valid?
end