Implement board value placement heuristics
This commit is contained in:
parent
bfd509f660
commit
ac5de85171
74
board.c
74
board.c
@ -209,3 +209,77 @@ board_is_valid (struct board *board)
|
|||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
board_update_marks (
|
||||||
|
struct board *board,
|
||||||
|
board_pos x,
|
||||||
|
board_pos y
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (is_in_bounds (x, y))
|
||||||
|
{
|
||||||
|
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||||
|
|
||||||
|
/* Mark all values as impossible */
|
||||||
|
elem->potential = 0;
|
||||||
|
|
||||||
|
/* Check x-axis */
|
||||||
|
for (board_pos check_x = 0; check_x < 9; ++check_x)
|
||||||
|
if (check_x != x && board_has_value (board, check_x, y))
|
||||||
|
elem->potential |= 1 << BOARD_ELEM (board, check_x, y)->value;
|
||||||
|
|
||||||
|
/* Check y-axis */
|
||||||
|
for (board_pos check_y = 0; check_y < 9; ++check_y)
|
||||||
|
if (check_y != y && board_has_value (board, x, check_y))
|
||||||
|
elem->potential |= 1 << BOARD_ELEM (board, x, check_y)->value;
|
||||||
|
|
||||||
|
/* Invert matches */
|
||||||
|
elem->potential ^= 0x1FF;
|
||||||
|
}
|
||||||
|
else ERROR("Invalid parameters to function board_update_marks()");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
board_update_all_marks (struct board *board)
|
||||||
|
{
|
||||||
|
for (board_pos y = 0; y < 9; ++y)
|
||||||
|
for (board_pos x = 0; x < 9; ++x)
|
||||||
|
if (! board_has_value (board, x, y))
|
||||||
|
board_update_marks (board, x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
board_place (
|
||||||
|
struct board *board,
|
||||||
|
board_pos x,
|
||||||
|
board_pos y,
|
||||||
|
element_value value
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||||
|
{
|
||||||
|
if (board_can_place_value (board, x, y, value))
|
||||||
|
{
|
||||||
|
/* Unmark x-axis */
|
||||||
|
for (board_pos unmark_x = 0; unmark_x < 9; ++unmark_x)
|
||||||
|
if (unmark_x != x && ! board_has_value (board, unmark_x, y))
|
||||||
|
board_unmark (board, unmark_x, y, value);
|
||||||
|
|
||||||
|
/* Unmark y-axis */
|
||||||
|
for (board_pos unmark_y = 0; unmark_y < 9; ++unmark_y)
|
||||||
|
if (unmark_y != y && ! board_has_value (board, x, unmark_y))
|
||||||
|
board_unmark (board, x, unmark_y, value);
|
||||||
|
|
||||||
|
/* Set value */
|
||||||
|
board_set (board, x, y, value);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else return false;
|
||||||
|
}
|
||||||
|
else ERROR("Invalid parameters to function board_place()");
|
||||||
|
}
|
||||||
|
35
board.h
35
board.h
@ -142,3 +142,38 @@ board_can_place_value (
|
|||||||
*/
|
*/
|
||||||
bool
|
bool
|
||||||
board_is_valid (struct board *board);
|
board_is_valid (struct board *board);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check row and column of given position and mark potential values that could
|
||||||
|
* be set at that position and still leave the board in a valid state
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
board_update_marks (
|
||||||
|
struct board *board,
|
||||||
|
board_pos x,
|
||||||
|
board_pos y
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Refreshes marks of all board elements without a value
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
board_update_all_marks (struct board *board);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to set value at the given position
|
||||||
|
* If value can be placed, this updates all elements without a value on the
|
||||||
|
* same row or column that do not have a value so that their markers reflect
|
||||||
|
* the addition of this value and returns true
|
||||||
|
* If value cannot be placed, just return false
|
||||||
|
*/
|
||||||
|
bool
|
||||||
|
board_place (
|
||||||
|
struct board *board,
|
||||||
|
board_pos x,
|
||||||
|
board_pos y,
|
||||||
|
element_value value
|
||||||
|
);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user