X-Git-Url: http://git.scottworley.com/slidingtile/blobdiff_plain/f3f55aff4e3346b297215050c2076c821fd60c83..310a7132ffc617b93969703de7f3bf1c5ea0fa30:/sliding_tile_lib.cc diff --git a/sliding_tile_lib.cc b/sliding_tile_lib.cc index b8ddc15..e4f2f12 100644 --- a/sliding_tile_lib.cc +++ b/sliding_tile_lib.cc @@ -1,9 +1,13 @@ #include "sliding_tile_lib.h" +#include #include +#include +#include #include +#include -signed char adjacent[BOARD_SIZE][5] = { +signed char Step::adjacent[BOARD_SIZE][5] = { 1, 4, -1, -1, -1, 0, 2, 5, -1, -1, 1, 3, 6, -1, -1, @@ -22,7 +26,7 @@ signed char adjacent[BOARD_SIZE][5] = { 11, 14, -1, -1, -1, }; -bool Board::is_valid() { +bool Board::is_valid() const { bool seen[BOARD_SIZE]; for (int i = 0; i < BOARD_SIZE; i++) { seen[i] = false; @@ -45,6 +49,30 @@ bool Board::is_valid() { return true; } +bool Board::operator==(const Board& o) const { + for (int i = 0; i < BOARD_SIZE; i++) { + if (board[i] != o.board[i]) { + return false; + } + } + return true; +} + +bool Board::operator!=(const Board& o) const { + return !operator==(o); +} + +bool Board::operator<(const Board& o) const { + for (int i = 0; i < BOARD_SIZE; i++) { + if (board[i] < o.board[i]) { + return true; + } else if (board[i] > o.board[i]) { + return false; + } + } + return false; +} + std::istream& operator>>(std::istream& is, Board& board) { for (int i = 0; i < BOARD_SIZE; i++) { if (!is.good()) { @@ -65,7 +93,17 @@ std::istream& operator>>(std::istream& is, Board& board) { return is; } -signed char Board::hole() { +std::ostream& operator<<(std::ostream& os, const Board& board) { + for (int i = 0; i < BOARD_SIZE; i++) { + if (i > 0) { + os << " "; + } + os << int(board.board[i]); + } + return os; +} + +signed char Board::hole() const { for (int i = 0; i < BOARD_SIZE; i++) { if (board[i] == 0) { return i; @@ -73,3 +111,52 @@ signed char Board::hole() { } throw std::runtime_error("Board with no hole"); } + +InvertedBoard Board::invert() const { + InvertedBoard inv; + for (int i = 0; i < BOARD_SIZE; i++) { + inv.pos[board[i]] = i; + } + return inv; +} + +int Board::distance(const Board& o) const { + return distance(o.invert()); +} + +int Board::distance(const InvertedBoard& invo) const { + int dist = 0; + for (int i = 0; i < BOARD_SIZE; i++) { + dist += std::abs(i % BOARD_DIM - invo.pos[board[i]] % BOARD_DIM) + + std::abs(i / BOARD_DIM - invo.pos[board[i]] / BOARD_DIM); + } + return dist; +} + +std::vector Step::successors(std::shared_ptr shared_this) { + std::vector suc; + signed char hole_pos = board.hole(); + for (int i = 0; adjacent[hole_pos][i] > 0; i++) { + suc.emplace_back(new Step{board, shared_this}); + std::swap(suc.back()->board.board[hole_pos], suc.back()->board.board[adjacent[hole_pos][i]]); + } + return suc; +} + +std::ostream& operator<<(std::ostream& os, const Step& step) { + if (step.prev != nullptr) { + os << *step.prev; + signed char this_hole = step.board.hole(); + signed char prev_hole = step.prev->board.hole(); + os << int(step.board.board[prev_hole]) << " "; + switch (this_hole - prev_hole) { + case -1: os << "right"; break; + case 1: os << "left"; break; + case -BOARD_DIM: os << "down"; break; + case BOARD_DIM: os << "up"; break; + default: os << "somehow!"; break; + } + os << std::endl; + } + return os; +}