advent-of-code-2020/day20/day20.cpp

451 lines
14 KiB
C++

#include <fstream>
#include <functional>
#include <iostream>
#include <map>
#include <memory>
#include <numeric>
#include <optional>
#include <regex>
#include <string>
#include <vector>
constexpr auto TILE_ID_PATTERN = R"(Tile (\d+):)";
constexpr int NUM_CAMERA_LINES = 10;
// Plus one for the ID line, plus one for the newline after
constexpr int NUM_INPUT_BLOCK_LINES = NUM_CAMERA_LINES + 2;
class CameraFrame {
public:
CameraFrame(int id, const std::vector<std::string> &frame) : id(id), frame(frame) {
}
int getID() const {
return this->id;
}
const std::vector<std::string> &getFrame() const {
return this->frame;
}
const std::string getTopEdge() const {
return this->frame.front();
}
const std::string getBottomEdge() const {
return this->frame.back();
}
const std::string getLeftEdge() const {
std::string edge;
std::transform(
this->frame.cbegin(),
this->frame.cend(),
std::inserter(edge, edge.end()),
[](const std::string &frameLine) { return frameLine.front(); });
return edge;
}
const std::string getRightEdge() const {
std::string edge;
std::transform(
this->frame.cbegin(),
this->frame.cend(),
std::inserter(edge, edge.end()),
[](const std::string &frameLine) { return frameLine.back(); });
return edge;
}
bool operator==(const CameraFrame &frame) const {
return this->id == frame.id;
}
bool operator!=(const CameraFrame &frame) const {
return !(*this == frame);
}
void flipFrameVertically() {
std::reverse(this->frame.begin(), this->frame.end());
}
void flipFrameHorizontally() {
std::for_each(this->frame.begin(), this->frame.end(), [](std::string &frameLine) {
std::reverse(frameLine.begin(), frameLine.end());
});
}
void rotateFrame90Deg() {
// This algorithm works by rotating each "ring" of the frame 90 degrees
// Consider the following 5x5 frame
// a b c d e
// f g h i j
// k l m n o
// p q r s t
// First we rotate the ring that starts with "a b c d e", then "g h i", and once we hit "m", there's nothing to
// rotate, so we're done.
// Init our rotated frame to be the size of our existing (square) frame.
std::vector<std::string> rotatedFrame(this->frame.size(), std::string(this->frame.at(0).size(), ' '));
int ringStartIndex = 0;
int ringEndIndex = this->frame.size() - 1;
while (ringStartIndex < ringEndIndex) {
// Top Edge -> Right Edge
for (int i = ringStartIndex; i <= ringEndIndex; i++) {
rotatedFrame.at(i).at(ringEndIndex) = frame.at(ringStartIndex).at(i);
}
// Right Edge -> Bottom Edge
for (int i = ringStartIndex; i <= ringEndIndex; i++) {
rotatedFrame.at(ringEndIndex).at(ringEndIndex - i + ringStartIndex) = frame.at(i).at(ringEndIndex);
}
// Bottom Edge -> Left Edge
for (int i = ringStartIndex; i <= ringEndIndex; i++) {
rotatedFrame.at(i).at(ringStartIndex) = frame.at(ringEndIndex).at(i);
}
// Left Edge -> Top Edge
for (int i = ringStartIndex; i <= ringEndIndex; i++) {
rotatedFrame.at(ringStartIndex).at(ringEndIndex - i + ringStartIndex) = frame.at(i).at(ringStartIndex);
}
ringStartIndex++;
ringEndIndex--;
}
// Fill in the middle
auto middle = this->frame.size() / 2;
rotatedFrame.at(middle).at(middle) = frame.at(middle).at(middle);
this->frame = std::move(rotatedFrame);
}
void rotateFrame180Deg() {
this->rotateFrame90Deg();
this->rotateFrame90Deg();
}
void rotateFrame270Deg() {
this->rotateFrame180Deg();
this->rotateFrame90Deg();
}
private:
int id;
std::vector<std::string> frame;
};
std::vector<std::string> readInput(const std::string &filename) {
std::vector<std::string> input;
std::string line;
std::ifstream file(filename);
while (std::getline(file, line)) {
input.push_back(line);
}
return input;
}
int getFrameIDFromIDLine(const std::string &line) {
std::regex pattern(TILE_ID_PATTERN);
std::smatch matches;
if (!std::regex_match(line, matches, pattern)) {
throw std::invalid_argument("Invalid ID line");
}
return std::stoi(matches[1]);
}
std::vector<CameraFrame> parseInput(const std::vector<std::string> &input) {
int i = 0;
std::vector<CameraFrame> cameraFrames;
int currentFrameID;
std::vector<std::string> currentCameraFrame;
for (auto it = input.cbegin(); it != input.cend(); (++it, i++)) {
const std::string &line = *it;
int lineIndex = i % NUM_INPUT_BLOCK_LINES;
if (lineIndex == 0) {
currentFrameID = getFrameIDFromIDLine(line);
} else if (line.size() > 0) {
// If the line isn't blank, this is a frame of an image
currentCameraFrame.push_back(line);
} else {
cameraFrames.emplace_back(currentFrameID, currentCameraFrame);
currentCameraFrame.clear();
}
}
// Handle the last frame
cameraFrames.emplace_back(currentFrameID, currentCameraFrame);
return cameraFrames;
}
std::ostream &operator<<(std::ostream &os, const CameraFrame &frame) {
os << "Tile " << frame.getID() << ":" << std::endl;
std::for_each(frame.getFrame().cbegin(), frame.getFrame().cend(), [&os](const std::string &frameLine) {
os << frameLine << std::endl;
});
return os;
}
void printBoard(const std::map<std::pair<int, int>, CameraFrame> &board, int maxRow, int maxCol) {
CameraFrame emptyFrame(0, std::vector<std::string>(NUM_CAMERA_LINES, std::string(NUM_CAMERA_LINES, ' ')));
for (int i = 0; i < maxRow; i++) {
std::vector<CameraFrame> rowFrames;
rowFrames.reserve(maxCol);
for (int frameRow = 0; frameRow < NUM_CAMERA_LINES; frameRow++) {
for (int j = 0; j < maxCol; j++) {
auto frame = board.find(std::make_pair(i, j));
auto frameRowStr = (frame == board.end() ? emptyFrame : frame->second).getFrame().at(frameRow);
std::cout << frameRowStr << " ";
}
std::cout << std::endl;
}
std::cout << std::endl;
}
}
std::vector<CameraFrame> getPossibleTransforms(const CameraFrame &frame) {
CameraFrame identity(frame);
CameraFrame rotated90Deg(frame);
rotated90Deg.rotateFrame90Deg();
CameraFrame rotated180Deg(frame);
rotated180Deg.rotateFrame180Deg();
CameraFrame rotated270Deg(frame);
rotated270Deg.rotateFrame270Deg();
CameraFrame flippedVertically(frame);
flippedVertically.flipFrameVertically();
CameraFrame flippedHorizontally(frame);
flippedHorizontally.flipFrameHorizontally();
CameraFrame flippedAndRotated90(flippedHorizontally);
flippedAndRotated90.rotateFrame90Deg();
// We don't need flip and rotate 180 because it is equivalent to flipVertically
CameraFrame flippedAndRotated270(flippedHorizontally);
flippedAndRotated270.rotateFrame270Deg();
std::vector<CameraFrame> operations{
identity,
rotated90Deg,
rotated180Deg,
rotated270Deg,
flippedVertically,
flippedHorizontally,
flippedAndRotated90,
flippedAndRotated270};
return operations;
}
bool isBoardFilled(const std::map<std::pair<int, int>, CameraFrame> board, int maxRow, int maxCol) {
for (int i = 0; i < maxRow; i++) {
for (int j = 0; j < maxCol; j++) {
if (board.find(std::make_pair(i, j)) == board.end()) {
return false;
}
}
}
return true;
}
std::optional<std::map<std::pair<int, int>, CameraFrame>> findLinedUpArrangement(
const std::map<std::pair<int, int>, CameraFrame> &board,
const std::vector<CameraFrame> &availableFrames,
std::map<std::pair<int, int>, int> &visited,
int row,
int col,
int maxRow,
int maxCol,
int depth = 0) {
// printBoard(board, maxRow, maxCol);
auto performRecursion = [&board, maxRow, maxCol, depth, &visited](
const CameraFrame &cursor,
const std::vector<CameraFrame> &availableFrames,
const std::string &edge1,
const std::function<std::string(const CameraFrame &)> getEdge2,
int nextRow,
int nextCol) -> std::optional<std::map<std::pair<int, int>, CameraFrame>> {
// std::cout << edge1 << " " << edge2 << std::endl;
// std::string reversedEdge(edge2);
// std::reverse(reversedEdge.begin(), rversedEdge.end());
for (const CameraFrame &candidateFrame : availableFrames) {
auto operations = getPossibleTransforms(candidateFrame);
// if (frame == cursor) {
// continue;
// }
std::vector<CameraFrame> nextAvailableFrames(availableFrames);
nextAvailableFrames.erase(
std::find(nextAvailableFrames.cbegin(), nextAvailableFrames.cend(), candidateFrame));
for (const CameraFrame &operatedFrame : operations) {
auto edge2 = getEdge2(operatedFrame);
if (edge1 == edge2) {
// std::cout << "Match!" << std::endl << std::endl;
std::map<std::pair<int, int>, CameraFrame> nextBoard(board);
nextBoard.emplace(std::make_pair(nextRow, nextCol), operatedFrame);
auto filled = findLinedUpArrangement(
nextBoard, nextAvailableFrames, visited, nextRow, nextCol, maxRow, maxCol, depth + 1);
// for (auto pair : *filled) {
// std::cout << pair.first.first << ", " << pair.first.second << " ";
// }
// std::cout << std::endl;
return filled;
// std::cout << "Throwing it away" << std::endl;
}
}
}
return std::nullopt;
};
// std::cout << availableFrames.size() << std::endl;
// std::cout << board.size() << std::endl;
// std::cout << row << " " << col << "; max of " << maxRow << " " << maxCol << std::endl;
const CameraFrame &cursor = board.at(std::make_pair(row, col));
auto visitedCheck = visited.find(std::make_pair(row, col));
if (isBoardFilled(board, maxRow, maxCol)) {
return board;
} else if (availableFrames.empty() || (visitedCheck != visited.end() && visitedCheck->second == cursor.getID())) {
return std::nullopt;
}
visited.emplace(std::make_pair(row, col), cursor.getID());
// std::cout << "depth=" << depth << " "
// << "size=" << nextAvailableFrames.size() << std::endl;
// Any operation involving the top edge cannot occur if this tile is already at the top
if (row > 0 && board.find(std::make_pair(row - 1, col)) == board.end()) {
// std::cout << "row > 0" << std::endl;
auto result = performRecursion(
cursor,
availableFrames,
cursor.getTopEdge(),
[](const CameraFrame &candidate) { return candidate.getBottomEdge(); },
row - 1,
col);
if (result) {
return *result;
}
}
if (row < maxRow - 1 && board.find(std::make_pair(row + 1, col)) == board.end()) {
// std::cout << "row < maxRow" << std::endl;
auto result = performRecursion(
cursor,
availableFrames,
cursor.getBottomEdge(),
[](const CameraFrame &candidate) { return candidate.getTopEdge(); },
row + 1,
col);
if (result) {
return *result;
}
}
// Any operation involving the left edge cannot occur if this tile is already at the left
if (col > 0 && board.find(std::make_pair(row, col - 1)) == board.end()) {
// std::cout << "col > 0" << std::endl;
auto result = performRecursion(
cursor,
availableFrames,
cursor.getLeftEdge(),
[](const CameraFrame &candidate) { return candidate.getRightEdge(); },
row,
col - 1);
if (result) {
return *result;
}
}
// Any operation involving the right edge cannot occur if the tile is at the right
if (col < maxCol - 1 && board.find(std::make_pair(row, col + 1)) == board.end()) {
// std::cout << "col < maxCol" << std::endl;
auto result = performRecursion(
cursor,
availableFrames,
cursor.getRightEdge(),
[](const CameraFrame &candidate) { return candidate.getLeftEdge(); },
row,
col + 1);
if (result) {
return *result;
}
}
return std::nullopt;
}
long part1(const std::vector<CameraFrame> &frames) {
// TODO: This should not always be 0,0, but I'm doing it so I can test this method.
for (int i = 0; i < frames.size(); i++) {
// std::cout << "WORKING ON FRAME " << frame.getID() << std::endl;
std::vector<CameraFrame> availableFrames(frames);
const CameraFrame &frame = frames.at(i);
availableFrames.erase(std::remove(availableFrames.begin(), availableFrames.end(), frame));
auto operations = getPossibleTransforms(frame);
for (int j = 0; j < operations.size(); j++) {
std::map<std::pair<int, int>, int> visited;
const CameraFrame &operatedFrame = operations.at(j);
std::cout << i << " " << j << std::endl;
std::map<std::pair<int, int>, CameraFrame> board;
board.emplace(std::make_pair(0, 0), operatedFrame);
// TODO: This 3,3 should not be hardcoded
auto res = findLinedUpArrangement(board, availableFrames, visited, 0, 0, 12, 12);
if (res) {
std::cout << res->at(std::make_pair(0, 0)).getID() << " " << res->at(std::make_pair(0, 11)).getID()
<< " " << res->at(std::make_pair(11, 0)).getID() << " "
<< res->at(std::make_pair(11, 11)).getID() << std::endl;
;
return 1L * res->at(std::make_pair(0, 0)).getID() * res->at(std::make_pair(0, 11)).getID() *
res->at(std::make_pair(11, 0)).getID() * res->at(std::make_pair(11, 11)).getID();
}
std::cout << "Finishing iteration of transformation" << std::endl;
}
std::cout << "Finishing iteration" << std::endl;
}
return 0;
}
int main(int argc, char *argv[]) {
if (argc != 2) {
std::cerr << argv[0] << " <input_file>" << std::endl;
return 1;
}
auto input = readInput(argv[1]);
auto parsedInput = parseInput(input);
std::cout << "Raw Frame" << std::endl;
std::cout << parsedInput.at(0) << std::endl;
std::cout << "Flipped Vertically" << std::endl;
CameraFrame flippedVertically(parsedInput.at(0));
flippedVertically.flipFrameVertically();
std::cout << flippedVertically << std::endl;
std::cout << "Flipped Horizontally" << std::endl;
CameraFrame flippedHorizontally(parsedInput.at(0));
flippedHorizontally.flipFrameHorizontally();
std::cout << flippedHorizontally << std::endl;
std::cout << "Rotated 90 deg" << std::endl;
CameraFrame rotated(parsedInput.at(0));
rotated.rotateFrame90Deg();
std::cout << rotated << std::endl;
std::cout << "Rotated 180 deg" << std::endl;
CameraFrame rotated1(parsedInput.at(0));
rotated1.rotateFrame180Deg();
std::cout << rotated1 << std::endl;
std::cout << "Rotated 270 deg" << std::endl;
CameraFrame rotated2(parsedInput.at(0));
rotated2.rotateFrame270Deg();
std::cout << rotated2 << std::endl;
std::cout << part1(parsedInput) << std::endl;
}