A-Puzzle-A-Day/Pieces.cpp

482 lines
9.5 KiB
C++

#include "Pieces.h"
#include <curses.h>
#include <stdlib.h> // exit()
unsigned PieceHeight(const Piece &piece, Rotation rotation)
{
switch(piece)
{
case PIECE_ONE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 2;
case ROTATION_90:
case ROTATION_270:
return 3;
};
break;
case PIECE_TWO:
case PIECE_THREE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 4;
case ROTATION_90:
case ROTATION_270:
return 2;
};
break;
case PIECE_FOUR:
case PIECE_SEVEN:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 3;
case ROTATION_90:
case ROTATION_270:
return 2;
};
break;
case PIECE_FIVE:
case PIECE_EIGHT:
return 3;
break;
case PIECE_SIX:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 2;
case ROTATION_90:
case ROTATION_270:
return 4;
};
break;
}
std::exit(-1);
}
unsigned PieceWidth(const Piece &piece, Rotation rotation)
{
switch(piece)
{
case PIECE_ONE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 3;
case ROTATION_90:
case ROTATION_270:
return 2;
};
break;
case PIECE_TWO:
case PIECE_THREE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 2;
case ROTATION_90:
case ROTATION_270:
return 4;
};
break;
case PIECE_FOUR:
case PIECE_SEVEN:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 2;
case ROTATION_90:
case ROTATION_270:
return 3;
};
break;
case PIECE_FIVE:
case PIECE_EIGHT:
return 3;
break;
case PIECE_SIX:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
return 4;
case ROTATION_90:
case ROTATION_270:
return 2;
};
break;
}
std::exit(-1);
}
void DrawPiece(const Piece &piece, int y, int x, Rotation rotation)
{
attrset(COLOR_PAIR(static_cast<int>(piece) % 8));
switch(piece)
{
case PIECE_ONE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x, "[][][]");
break;
case ROTATION_90:
case ROTATION_270:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x, "[][]");
mvaddstr(y + 2, x, "[][]");
break;
};
break;
case PIECE_TWO:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x, "[]");
mvaddstr(y + 1, x, "[][]");
mvaddstr(y + 2, x, "[]");
mvaddstr(y + 3, x, "[]");
break;
case ROTATION_90:
mvaddstr(y, x, "[][][][]");
mvaddstr(y + 1, x + 4, "[]");
break;
case ROTATION_180:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x + 2, "[]");
mvaddstr(y + 2, x, "[][]");
mvaddstr(y + 3, x + 2, "[]");
break;
case ROTATION_270:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x, "[][][][]");
break;
};
break;
case PIECE_THREE:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x + 2, "[]");
mvaddstr(y + 2, x + 2, "[]");
mvaddstr(y + 3, x, "[][]");
break;
case ROTATION_90:
mvaddstr(y, x, "[]");
mvaddstr(y + 1, x, "[][][][]");
break;
case ROTATION_180:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x, "[]");
mvaddstr(y + 2, x, "[]");
mvaddstr(y + 3, x, "[]");
break;
case ROTATION_270:
mvaddstr(y, x, "[][][][]");
mvaddstr(y + 1, x + 6, "[]");
break;
};
break;
case PIECE_FOUR:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x, "[]");
mvaddstr(y + 2, x, "[][]");
break;
case ROTATION_90:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x, "[]");
mvaddstr(y + 1, x + 4, "[]"); // same line as above
break;
case ROTATION_180:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x + 2, "[]");
mvaddstr(y + 2, x, "[][]");
break;
case ROTATION_270:
mvaddstr(y, x, "[]");
mvaddstr(y, x + 4, "[]"); // same line as above
mvaddstr(y + 1, x, "[][][]");
break;
};
break;
case PIECE_FIVE:
switch(rotation)
{
default:
case ROTATION_NONE:
case ROTATION_180:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x + 2, "[]");
mvaddstr(y + 2, x + 2, "[][]");
break;
case ROTATION_90:
case ROTATION_270:
mvaddstr(y, x + 4, "[]");
mvaddstr(y + 1, x, "[][][]");
mvaddstr(y + 2, x, "[]");
break;
};
break;
case PIECE_SIX:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x + 2, "[][][]");
break;
case ROTATION_90:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x, "[][]");
mvaddstr(y + 2, x, "[]");
mvaddstr(y + 3, x, "[]");
break;
case ROTATION_180:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x + 4, "[][]");
break;
case ROTATION_270:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x + 2, "[]");
mvaddstr(y + 2, x, "[][]");
mvaddstr(y + 3, x, "[]");
break;
};
break;
case PIECE_SEVEN:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x, "[][]");
mvaddstr(y + 2, x, "[]");
break;
case ROTATION_90:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x + 2, "[][]");
break;
case ROTATION_180:
mvaddstr(y, x + 2, "[]");
mvaddstr(y + 1, x, "[][]");
mvaddstr(y + 2, x, "[][]");
break;
case ROTATION_270:
mvaddstr(y, x, "[][]");
mvaddstr(y + 1, x, "[][][]");
break;
};
break;
case PIECE_EIGHT:
switch(rotation)
{
default:
case ROTATION_NONE:
mvaddstr(y, x + 4, "[]");
mvaddstr(y + 1, x + 4, "[]");
mvaddstr(y + 2, x, "[][][]");
break;
case ROTATION_90:
mvaddstr(y, x, "[]");
mvaddstr(y + 1, x, "[]");
mvaddstr(y + 2, x, "[][][]");
break;
case ROTATION_180:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x, "[]");
mvaddstr(y + 2, x, "[]");
break;
case ROTATION_270:
mvaddstr(y, x, "[][][]");
mvaddstr(y + 1, x + 4, "[]");
mvaddstr(y + 2, x + 4, "[]");
break;
};
break;
default:
std::exit(-1);
break;
}
}
PieceData RotatePieceData(const PieceData &data, Rotation rotation)
{
PieceData rtn;
if(!data.size())
return rtn;
if(!data[0].size())
return rtn;
// Step 1: Size it:
switch(rotation)
{
default:
case Rotation::ROTATION_NONE:
case Rotation::ROTATION_180:
rtn.resize(data.size());
for(auto &row : rtn)
row.resize(data[0].size());
break;
case Rotation::ROTATION_90:
case Rotation::ROTATION_270:
rtn.resize(data[0].size());
for(auto &row : rtn)
row.resize(data.size());
break;
};
// Step 2: Move it:
switch(rotation)
{
default:
case Rotation::ROTATION_NONE:
rtn = data;
break;
case Rotation::ROTATION_90:
for(size_t row = 0; row < data.size(); ++row)
{
size_t opposite_row = data.size() - row - 1;
for(size_t col = 0; col < data[row].size(); ++col)
{
size_t opposite_col = data[opposite_row].size() - col - 1;
// NOTE(dev): x's become y's
// NOTE(dev): y's become max x - x's
rtn[col][opposite_row] = data[row][col];
}
}
break;
case Rotation::ROTATION_180:
for(size_t row = 0; row < data.size(); ++row)
{
size_t opposite_row = data.size() - row - 1;
for(size_t col = 0; col < data[row].size(); ++col)
{
size_t opposite_col = data[opposite_row].size() - col - 1;
rtn[opposite_row][opposite_col] = data[row][col];
}
}
break;
case Rotation::ROTATION_270:
for(size_t row = 0; row < data.size(); ++row)
{
size_t opposite_row = data.size() - row - 1;
for(size_t col = 0; col < data[row].size(); ++col)
{
size_t opposite_col = data[opposite_row].size() - col - 1;
// NOTE(dev): x's become max y - y's
// NOTE(dev): y's become x's
rtn[opposite_col][row] = data[row][col];
}
}
break;
};
return rtn;
}
void FlipPieceData(PieceData &data, Flip flip)
{
if(!data.size())
return;
if(!data[0].size())
return;
switch(flip)
{
default:
case Flip::FLIP_NONE:
return;
case Flip::FLIP_HORIZONTAL:
// Go through EVERY row:
for(size_t row = 0; row < data.size(); ++row)
{
// And swap the front half with the back half
// [1][2][3][2][1]
// ^ ^ ^ ^
// | +-----+ | <-- second
// | |
// +-----------+ <-- first
for(size_t col = 0; col < data[0].size() / 2; ++col)
{
// for each row, swap columns
bool temp = data[row][col];
data[row][col] = data[row][data[0].size() - col - 1];
data[row][data[0].size() - col - 1] = temp;
}
}
break;
case Flip::FLIP_VERTICAL:
// Go through EVERY col:
for(size_t col = 0; col < data[0].size(); ++col)
{
// And swap the top half with the bottom half
for(size_t row = 0; row < data.size() / 2; ++row)
{
// for each row, swap columns
bool temp = data[row][col];
data[row][col] = data[data.size() - row - 1][col];
data[data.size() - row - 1][col] = temp;
}
}
break;
};
}
unsigned PD_PieceHeight(const PieceData &data)
{
return data.size();
}
unsigned PD_PieceWidth(const PieceData &data)
{
size_t rtn = 0;
for(auto row : data)
rtn = std::max(rtn, row.size());
return rtn;
}
void PD_DrawPiece(const PieceData &data, int y, int x, int color)
{
attrset(COLOR_PAIR(static_cast<int>(color) % 8));
for(size_t row = 0; row < data.size(); ++row)
{
for(size_t col = 0; col < data[row].size(); ++col)
{
if(data[row][col])
mvaddstr(y + row, x + (col * 2), "[]");
}
}
}