diff --git a/ordle.cc b/ordle.cc index 33b5bda..75587e4 100644 --- a/ordle.cc +++ b/ordle.cc @@ -1,7 +1,7 @@ #include "ordle.h" #include #include -#include +#include #include #include #include @@ -30,22 +30,31 @@ std::vector read_candidates(std::istream &input) { } bool wrong_fn::operator()(const std::string &word) const { - return contains_any_of(word, l); + return contains_any_of(word, letter_set); } bool correct_fn::operator()(const std::string &word) const { - return std::all_of(m.begin(), m.end(), [&word](const auto &pair) { - return contains_at(word, pair.second[0], pair.first); + return std::all_of(map.begin(), map.end(), [&word](const auto &pair) { + return contains_at(word, pair.second, pair.first); }); } +// TODO:Correct but ugly bool misplaced_fn::operator()(const std::string &word) const { - return std::all_of(m.begin(), m.end(), [&word](const auto &pair) { - auto a = contains_but_not_at(word, pair.second[0], pair.first); - if (a) - std::cout << "Misplaced: " << word << std::endl; - return a; - }); + if (std::all_of(map.begin(), map.end(), [&word](const auto &pair) { + return pair.first < word.size() && word[pair.first] == pair.second; + })) { + return false; + } + + if (std::any_of(map.begin(), map.end(), [&word](const auto &pair) { + return std::find(word.begin(), word.end(), pair.second) != + word.end(); + })) { + return true; + } + + return false; } void do_filter(std::vector &candidates, const std::string &wrong, @@ -68,7 +77,7 @@ IndexMap build_list(const std::string &line) { size_type index; while (iss >> letter >> index) { - result[index] = std::string(1, letter); + result[index] = letter; } return result; diff --git a/ordle.h b/ordle.h index ac428df..26930a6 100644 --- a/ordle.h +++ b/ordle.h @@ -27,7 +27,7 @@ using size_type = std::string::size_type; * * @see correct_fn, misplaced_fn, build_list */ -using IndexMap = std::map; +using IndexMap = std::map; /** * Reads a list of words from an input stream where each word is separated by a @@ -92,7 +92,7 @@ struct wrong_fn { * @param letters A string containing the letters that words should not * contain. */ - wrong_fn(const std::string &letters) : l{letters} {} + wrong_fn(const std::string &letters) : letter_set{letters} {} /** * Checks if a word contains any of the specified "wrong" letters. @@ -103,7 +103,7 @@ struct wrong_fn { bool operator()(const std::string &word) const; private: - std::string l; ///< The letters considered "wrong". + std::string letter_set; ///< The letters considered "wrong". }; /** @@ -116,7 +116,7 @@ struct correct_fn { * @param idxs A map of indices to characters representing the exact * matches. */ - correct_fn(const IndexMap &idxs) : m{idxs} {} + correct_fn(const IndexMap &idxs) : map{idxs} {} /** * Checks if a word contains the correct characters at the specified @@ -128,7 +128,7 @@ struct correct_fn { bool operator()(const std::string &word) const; private: - IndexMap m; ///< Map of indices to expected characters. + IndexMap map; ///< Map of indices to expected characters. }; /** @@ -141,20 +141,20 @@ struct misplaced_fn { * @param idxs A map of indices to characters representing misplaced * matches. */ - misplaced_fn(const IndexMap &idxs) : m{idxs} {} + misplaced_fn(const IndexMap &idxs) : map{idxs} {} /** * Checks if a word contains the specified characters but at incorrect * positions. * * @param word The word to be checked. - * @return True if the word contains the characters but at incorrect + * @return True if the word contains ALL of the characters but at incorrect * positions; false otherwise. */ bool operator()(const std::string &word) const; private: - IndexMap m; ///< Map of indices to misplaced characters. + IndexMap map; ///< Map of indices to misplaced characters. }; /** diff --git a/test/test.cc b/test/test.cc index 427c188..454e5a0 100644 --- a/test/test.cc +++ b/test/test.cc @@ -29,6 +29,7 @@ void test_contains_at() { void test_contains_but_not_at() { assert(contains_but_not_at("apple", 'p', 0)); assert(!contains_but_not_at("apple", 'a', 0)); + assert(!contains_but_not_at("abc", 'k', 1000)); std::cout << "test_contains_but_not_at passed.\n"; } @@ -40,7 +41,7 @@ void test_wrong_fn() { } void test_correct_fn() { - IndexMap green = {{0, "a"}, {4, "e"}}; + IndexMap green = {{0, 'a'}, {4, 'e'}}; correct_fn correct(green); assert(correct("apple")); assert(correct("ample")); @@ -49,11 +50,12 @@ void test_correct_fn() { } void test_misplaced_fn() { - IndexMap yellow = {{1, "p"}, {2, "p"}}; + IndexMap yellow = {{1, 'p'}, {2, 'p'}}; misplaced_fn misplaced(yellow); assert(!misplaced("apple")); assert(misplaced("puppy")); + assert(!misplaced("joink")); std::cout << "test_misplaced_fn passed.\n"; } @@ -61,20 +63,58 @@ void test_misplaced_fn() { void test_do_filter() { std::vector candidates = {"apple", "ample", "angle"}; std::string wrong = "iou"; - IndexMap green = {{0, "a"}}; - IndexMap yellow = {{4, "e"}}; + IndexMap green = {{0, 'a'}, {1, 'p'}}; + IndexMap yellow = {{3, 'e'}}; do_filter(candidates, wrong, green, yellow); + std::for_each(candidates.begin(), candidates.end(), + [&](auto &word) { std::cout << word << std::endl; }); + assert(candidates == std::vector{"apple"}); std::cout << "test_do_filter passed.\n"; } void test_build_list() { - auto result = build_list("a:0 e:4"); - auto l = IndexMap{{0, "a"}, {4, "e"}}; - assert(result == l); - std::cout << "test_build_list passed.\n"; + { + std::string input = "a0 b1 c2"; + IndexMap result = build_list(input); + + assert(result.size() == 3); + assert(result[0] == 'a'); + assert(result[1] == 'b'); + assert(result[2] == 'c'); + + std::cout << "Test case 1 passed.\n"; + } + { // Empty input + std::string input = ""; + IndexMap result = build_list(input); + + assert(result.empty()); + + std::cout << "Test case 2 passed.\n"; + } + { // Single input + std::string input = "z9"; + IndexMap result = build_list(input); + + assert(result.size() == 1); + assert(result[9] == 'z'); + + std::cout << "Test case 3 passed.\n"; + } + { // Irregular spacing + std::string input = "x3 y4 z5"; + IndexMap result = build_list(input); + + assert(result.size() == 3); + assert(result[3] == 'x'); + assert(result[4] == 'y'); + assert(result[5] == 'z'); + + std::cout << "Test case 4 passed.\n"; + } } int main() { @@ -84,9 +124,9 @@ int main() { test_contains_but_not_at(); test_wrong_fn(); test_correct_fn(); - // test_misplaced_fn(); // Misbehaves - // test_do_filter(); // Misbehaves - // test_build_list(); // Misbehaves + test_misplaced_fn(); + test_do_filter(); + test_build_list(); std::cout << "All tests passed!\n"; return 0;