| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630 |
- // This file is part of OpenCV project.
- // It is subject to the license terms in the LICENSE file found in the top-level directory
- // of this distribution and at http://opencv.org/license.html.
- #include "test_precomp.hpp"
- namespace opencv_test { namespace {
- std::string encode_qrcode_images_name[] = {
- "version1_mode1.png", "version1_mode2.png", "version1_mode4.png",
- "version2_mode1.png", "version2_mode2.png", "version2_mode4.png",
- "version3_mode2.png", "version3_mode4.png",
- "version4_mode4.png"
- };
- std::string encode_qrcode_eci_images_name[] = {
- "version1_mode7.png",
- "version2_mode7.png",
- "version3_mode7.png",
- "version4_mode7.png",
- "version5_mode7.png"
- };
- const Size fixed_size = Size(200, 200);
- const float border_width = 2.0;
- int establishCapacity(QRCodeEncoder::EncodeMode mode, int version, int capacity)
- {
- int result = 0;
- capacity *= 8;
- capacity -= 4;
- switch (mode)
- {
- case QRCodeEncoder::MODE_NUMERIC:
- {
- if (version >= 10)
- capacity -= 12;
- else
- capacity -= 10;
- int tmp = capacity / 10;
- result = tmp * 3;
- if (tmp * 10 + 7 <= capacity)
- result += 2;
- else if (tmp * 10 + 4 <= capacity)
- result += 1;
- break;
- }
- case QRCodeEncoder::MODE_ALPHANUMERIC:
- {
- if (version < 10)
- capacity -= 9;
- else
- capacity -= 13;
- int tmp = capacity / 11;
- result = tmp * 2;
- if (tmp * 11 + 6 <= capacity)
- result++;
- break;
- }
- case QRCodeEncoder::MODE_BYTE:
- {
- if (version > 9)
- capacity -= 16;
- else
- capacity -= 8;
- result = capacity / 8;
- break;
- }
- default:
- break;
- }
- return result;
- }
- // #define UPDATE_TEST_DATA
- #ifdef UPDATE_TEST_DATA
- TEST(Objdetect_QRCode_Encode, generate_test_data)
- {
- const std::string root = "qrcode/encode";
- const std::string dataset_config = findDataFile(root + "/" + "dataset_config.json");
- FileStorage file_config(dataset_config, FileStorage::WRITE);
- file_config << "test_images" << "[";
- size_t images_count = sizeof(encode_qrcode_images_name) / sizeof(encode_qrcode_images_name[0]);
- for (size_t i = 0; i < images_count; i++)
- {
- file_config << "{:" << "image_name" << encode_qrcode_images_name[i];
- std::string image_path = findDataFile(root + "/" + encode_qrcode_images_name[i]);
- Mat src = imread(image_path, IMREAD_GRAYSCALE);
- Mat straight_barcode;
- EXPECT_TRUE(!src.empty()) << "Can't read image: " << image_path;
- std::vector<Point2f> corners(4);
- corners[0] = Point2f(border_width, border_width);
- corners[1] = Point2f(qrcode.cols * 1.0f - border_width, border_width);
- corners[2] = Point2f(qrcode.cols * 1.0f - border_width, qrcode.rows * 1.0f - border_width);
- corners[3] = Point2f(border_width, qrcode.rows * 1.0f - border_width);
- Mat resized_src;
- resize(qrcode, resized_src, fixed_size, 0, 0, INTER_AREA);
- float width_ratio = resized_src.cols * 1.0f / qrcode.cols;
- float height_ratio = resized_src.rows * 1.0f / qrcode.rows;
- for(size_t j = 0; j < corners.size(); j++)
- {
- corners[j].x = corners[j].x * width_ratio;
- corners[j].y = corners[j].y * height_ratio;
- }
- std::string decoded_info = "";
- EXPECT_TRUE(decodeQRCode(resized_src, corners, decoded_info, straight_barcode)) << "The QR code cannot be decoded: " << image_path;
- file_config << "info" << decoded_info;
- file_config << "}";
- }
- file_config << "]";
- file_config.release();
- }
- #else
- typedef testing::TestWithParam< std::string > Objdetect_QRCode_Encode;
- TEST_P(Objdetect_QRCode_Encode, regression) {
- const int pixels_error = 3;
- const std::string name_current_image = GetParam();
- const std::string root = "qrcode/encode";
- std::string image_path = findDataFile(root + "/" + name_current_image);
- const std::string dataset_config = findDataFile(root + "/" + "dataset_config.json");
- FileStorage file_config(dataset_config, FileStorage::READ);
- ASSERT_TRUE(file_config.isOpened()) << "Can't read validation data: " << dataset_config;
- {
- FileNode images_list = file_config["test_images"];
- size_t images_count = static_cast<size_t>(images_list.size());
- ASSERT_GT(images_count, 0u) << "Can't find validation data entries in 'test_images': " << dataset_config;
- for (size_t index = 0; index < images_count; index++)
- {
- FileNode config = images_list[(int)index];
- std::string name_test_image = config["image_name"];
- if (name_test_image == name_current_image)
- {
- std::string original_info = config["info"];
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create();
- Mat result;
- encoder->encode(original_info, result);
- EXPECT_FALSE(result.empty()) << "Can't generate QR code image";
- Mat src = imread(image_path, IMREAD_GRAYSCALE);
- Mat straight_barcode;
- EXPECT_TRUE(!src.empty()) << "Can't read image: " << image_path;
- double diff_norm = cvtest::norm(result - src, NORM_L1);
- EXPECT_NEAR(diff_norm, 0.0, pixels_error) << "The generated QRcode is not same as test data. The difference: " << diff_norm;
- return; // done
- }
- }
- FAIL() << "Not found results in config file:" << dataset_config
- << "\nRe-run tests with enabled UPDATE_ENCODE_TEST_DATA macro to update test data.";
- }
- }
- typedef testing::TestWithParam< std::string > Objdetect_QRCode_Encode_ECI;
- TEST_P(Objdetect_QRCode_Encode_ECI, regression) {
- const int pixels_error = 3;
- const std::string name_current_image = GetParam();
- const std::string root = "qrcode/encode";
- std::string image_path = findDataFile(root + "/" + name_current_image);
- const std::string dataset_config = findDataFile(root + "/" + "dataset_config.json");
- FileStorage file_config(dataset_config, FileStorage::READ);
- ASSERT_TRUE(file_config.isOpened()) << "Can't read validation data: " << dataset_config;
- {
- FileNode images_list = file_config["test_images"];
- size_t images_count = static_cast<size_t>(images_list.size());
- ASSERT_GT(images_count, 0u) << "Can't find validation data entries in 'test_images': " << dataset_config;
- QRCodeEncoder::Params params;
- params.mode = QRCodeEncoder::MODE_ECI;
- for (size_t index = 0; index < images_count; index++)
- {
- FileNode config = images_list[(int)index];
- std::string name_test_image = config["image_name"];
- if (name_test_image == name_current_image)
- {
- std::string original_info = config["info"];
- Mat result;
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create(params);
- encoder->encode(original_info, result);
- EXPECT_FALSE(result.empty()) << "Can't generate QR code image";
- Mat src = imread(image_path, IMREAD_GRAYSCALE);
- Mat straight_barcode;
- EXPECT_TRUE(!src.empty()) << "Can't read image: " << image_path;
- double diff_norm = cvtest::norm(result - src, NORM_L1);
- EXPECT_NEAR(diff_norm, 0.0, pixels_error) << "The generated QRcode is not same as test data. The difference: " << diff_norm;
- return; // done
- }
- }
- FAIL() << "Not found results in config file:" << dataset_config
- << "\nRe-run tests with enabled UPDATE_ENCODE_TEST_DATA macro to update test data.";
- }
- }
- INSTANTIATE_TEST_CASE_P(/**/, Objdetect_QRCode_Encode, testing::ValuesIn(encode_qrcode_images_name));
- INSTANTIATE_TEST_CASE_P(/**/, Objdetect_QRCode_Encode_ECI, testing::ValuesIn(encode_qrcode_eci_images_name));
- TEST(Objdetect_QRCode_Encode_Decode, regression)
- {
- const std::string root = "qrcode/decode_encode";
- const int min_version = 1;
- const int test_max_version = 5;
- const int max_ec_level = 3;
- const std::string dataset_config = findDataFile(root + "/" + "symbol_sets.json");
- const std::string version_config = findDataFile(root + "/" + "capacity.json");
- FileStorage file_config(dataset_config, FileStorage::READ);
- FileStorage capacity_config(version_config, FileStorage::READ);
- ASSERT_TRUE(file_config.isOpened()) << "Can't read validation data: " << dataset_config;
- ASSERT_TRUE(capacity_config.isOpened()) << "Can't read validation data: " << version_config;
- FileNode mode_list = file_config["symbols_sets"];
- FileNode capacity_list = capacity_config["version_ecc_capacity"];
- size_t mode_count = static_cast<size_t>(mode_list.size());
- ASSERT_GT(mode_count, 0u) << "Can't find validation data entries in 'test_images': " << dataset_config;
- const int testing_modes = 3;
- QRCodeEncoder::EncodeMode modes[testing_modes] = {
- QRCodeEncoder::MODE_NUMERIC,
- QRCodeEncoder::MODE_ALPHANUMERIC,
- QRCodeEncoder::MODE_BYTE
- };
- for (int i = 0; i < testing_modes; i++)
- {
- QRCodeEncoder::EncodeMode mode = modes[i];
- FileNode config = mode_list[i];
- std::string symbol_set = config["symbols_set"];
- for(int version = min_version; version <= test_max_version; version++)
- {
- FileNode capa_config = capacity_list[version - 1];
- for(int level = 0; level <= max_ec_level; level++)
- {
- const int cur_capacity = capa_config["ecc_level"][level];
- int true_capacity = establishCapacity(mode, version, cur_capacity);
- std::string input_info = symbol_set;
- std::mt19937 rand_gen {1};
- std::shuffle(input_info.begin(), input_info.end(), rand_gen);
- int count = 0;
- if((int)input_info.length() > true_capacity)
- {
- input_info = input_info.substr(0, true_capacity);
- }
- else
- {
- while ((int)input_info.length() != true_capacity)
- {
- input_info += input_info.substr(count%(int)input_info.length(), 1);
- count++;
- }
- }
- QRCodeEncoder::Params params;
- params.version = version;
- params.correction_level = static_cast<QRCodeEncoder::CorrectionLevel>(level);
- params.mode = mode;
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create(params);
- Mat qrcode;
- encoder->encode(input_info, qrcode);
- EXPECT_TRUE(!qrcode.empty()) << "Can't generate this QR image (" << "mode: " << (int)mode <<
- " version: "<< version <<" error correction level: "<< (int)level <<")";
- std::vector<Point2f> corners(4);
- corners[0] = Point2f(border_width, border_width);
- corners[1] = Point2f(qrcode.cols * 1.0f - border_width, border_width);
- corners[2] = Point2f(qrcode.cols * 1.0f - border_width, qrcode.rows * 1.0f - border_width);
- corners[3] = Point2f(border_width, qrcode.rows * 1.0f - border_width);
- Mat resized_src;
- resize(qrcode, resized_src, fixed_size, 0, 0, INTER_AREA);
- float width_ratio = resized_src.cols * 1.0f / qrcode.cols;
- float height_ratio = resized_src.rows * 1.0f / qrcode.rows;
- for(size_t k = 0; k < corners.size(); k++)
- {
- corners[k].x = corners[k].x * width_ratio;
- corners[k].y = corners[k].y * height_ratio;
- }
- Mat straight_barcode;
- std::string output_info = QRCodeDetector().decode(resized_src, corners, straight_barcode);
- EXPECT_FALSE(output_info.empty())
- << "The generated QRcode cannot be decoded." << " Mode: " << (int)mode
- << " version: " << version << " error correction level: " << (int)level;
- EXPECT_EQ(input_info, output_info) << "The generated QRcode is not same as test data." << " Mode: " << (int)mode <<
- " version: " << version << " error correction level: " << (int)level;
- }
- }
- }
- }
- TEST(Objdetect_QRCode_Encode_Kanji, regression)
- {
- QRCodeEncoder::Params params;
- params.mode = QRCodeEncoder::MODE_KANJI;
- Mat qrcode;
- const int testing_versions = 3;
- std::string input_infos[testing_versions] = {"\x82\xb1\x82\xf1\x82\xc9\x82\xbf\x82\xcd\x90\xa2\x8a\x45", // "Hello World" in Japanese
- "\x82\xa8\x95\xa0\x82\xaa\x8b\xf3\x82\xa2\x82\xc4\x82\xa2\x82\xdc\x82\xb7", // "I am hungry" in Japanese
- "\x82\xb1\x82\xf1\x82\xc9\x82\xbf\x82\xcd\x81\x41\x8e\x84\x82\xcd\x8f\xad\x82\xb5\x93\xfa\x96\x7b\x8c\xea\x82\xf0\x98\x62\x82\xb5\x82\xdc\x82\xb7" // "Hello, I speak a little Japanese" in Japanese
- };
- for (int i = 0; i < testing_versions; i++)
- {
- std::string input_info = input_infos[i];
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create(params);
- encoder->encode(input_info, qrcode);
- std::vector<Point2f> corners(4);
- corners[0] = Point2f(border_width, border_width);
- corners[1] = Point2f(qrcode.cols * 1.0f - border_width, border_width);
- corners[2] = Point2f(qrcode.cols * 1.0f - border_width, qrcode.rows * 1.0f - border_width);
- corners[3] = Point2f(border_width, qrcode.rows * 1.0f - border_width);
- Mat resized_src;
- resize(qrcode, resized_src, fixed_size, 0, 0, INTER_AREA);
- float width_ratio = resized_src.cols * 1.0f / qrcode.cols;
- float height_ratio = resized_src.rows * 1.0f / qrcode.rows;
- for(size_t j = 0; j < corners.size(); j++)
- {
- corners[j].x = corners[j].x * width_ratio;
- corners[j].y = corners[j].y * height_ratio;
- }
- Mat straight_barcode;
- QRCodeDetector detector;
- std::string decoded_info = detector.decode(resized_src, corners, straight_barcode);
- EXPECT_FALSE(decoded_info.empty()) << "The generated QRcode cannot be decoded.";
- EXPECT_EQ(input_info, decoded_info);
- EXPECT_EQ(detector.getEncoding(), QRCodeEncoder::ECIEncodings::ECI_SHIFT_JIS);
- }
- }
- TEST(Objdetect_QRCode_Encode_Decode_Structured_Append, regression)
- {
- // disabled since QR decoder probably doesn't support structured append mode qr codes
- const std::string root = "qrcode/decode_encode";
- const std::string dataset_config = findDataFile(root + "/" + "symbol_sets.json");
- const std::string version_config = findDataFile(root + "/" + "capacity.json");
- FileStorage file_config(dataset_config, FileStorage::READ);
- ASSERT_TRUE(file_config.isOpened()) << "Can't read validation data: " << dataset_config;
- FileNode mode_list = file_config["symbols_sets"];
- size_t mode_count = static_cast<size_t>(mode_list.size());
- ASSERT_GT(mode_count, 0u) << "Can't find validation data entries in 'test_images': " << dataset_config;
- int modes[] = {1, 2, 4};
- const int min_stuctures_num = 2;
- const int max_stuctures_num = 5;
- for (int i = 0; i < 3; i++)
- {
- int mode = modes[i];
- FileNode config = mode_list[i];
- std::string symbol_set = config["symbols_set"];
- std::string input_info = symbol_set;
- std::mt19937 rand_gen {1};
- std::shuffle(input_info.begin(), input_info.end(), rand_gen);
- for (int j = min_stuctures_num; j < max_stuctures_num; j++)
- {
- QRCodeEncoder::Params params;
- params.structure_number = j;
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create(params);
- vector<Mat> qrcodes;
- encoder->encodeStructuredAppend(input_info, qrcodes);
- EXPECT_TRUE(!qrcodes.empty()) << "Can't generate this QR images";
- CV_CheckEQ(qrcodes.size(), (size_t)j, "Number of QR codes");
- std::vector<Point2f> corners(4 * qrcodes.size());
- for (size_t k = 0; k < qrcodes.size(); k++)
- {
- Mat qrcode = qrcodes[k];
- corners[4 * k] = Point2f(border_width, border_width);
- corners[4 * k + 1] = Point2f(qrcode.cols * 1.0f - border_width, border_width);
- corners[4 * k + 2] = Point2f(qrcode.cols * 1.0f - border_width, qrcode.rows * 1.0f - border_width);
- corners[4 * k + 3] = Point2f(border_width, qrcode.rows * 1.0f - border_width);
- float width_ratio = fixed_size.width * 1.0f / qrcode.cols;
- float height_ratio = fixed_size.height * 1.0f / qrcode.rows;
- resize(qrcode, qrcodes[k], fixed_size, 0, 0, INTER_AREA);
- for (size_t ki = 0; ki < 4; ki++)
- {
- corners[4 * k + ki].x = corners[4 * k + ki].x * width_ratio + fixed_size.width * k;
- corners[4 * k + ki].y = corners[4 * k + ki].y * height_ratio;
- }
- }
- Mat resized_src;
- hconcat(qrcodes, resized_src);
- std::vector<cv::String> decoded_info;
- cv::String output_info;
- EXPECT_TRUE(QRCodeDetector().decodeMulti(resized_src, corners, decoded_info));
- for (size_t k = 0; k < decoded_info.size(); ++k)
- {
- if (!decoded_info[k].empty())
- output_info = decoded_info[k];
- }
- EXPECT_FALSE(output_info.empty())
- << "The generated QRcode cannot be decoded." << " Mode: " << modes[i]
- << " structures number: " << j;
- EXPECT_EQ(input_info, output_info) << "The generated QRcode is not same as test data." << " Mode: " << mode <<
- " structures number: " << j;
- }
- }
- }
- #endif // UPDATE_QRCODE_TEST_DATA
- CV_ENUM(EncodeModes, QRCodeEncoder::EncodeMode::MODE_NUMERIC,
- QRCodeEncoder::EncodeMode::MODE_ALPHANUMERIC,
- QRCodeEncoder::EncodeMode::MODE_BYTE)
- typedef ::testing::TestWithParam<EncodeModes> Objdetect_QRCode_Encode_Decode_Structured_Append_Parameterized;
- TEST_P(Objdetect_QRCode_Encode_Decode_Structured_Append_Parameterized, regression_22205)
- {
- const std::string input_data = "the quick brown fox jumps over the lazy dog";
- std::vector<cv::Mat> result_qrcodes;
- cv::QRCodeEncoder::Params params;
- int encode_mode = GetParam();
- params.mode = static_cast<cv::QRCodeEncoder::EncodeMode>(encode_mode);
- for(size_t struct_num = 2; struct_num < 5; ++struct_num)
- {
- params.structure_number = static_cast<int>(struct_num);
- cv::Ptr<cv::QRCodeEncoder> encoder = cv::QRCodeEncoder::create(params);
- encoder->encodeStructuredAppend(input_data, result_qrcodes);
- EXPECT_EQ(result_qrcodes.size(), struct_num) << "The number of QR Codes requested is not equal"<<
- "to the one returned";
- }
- }
- INSTANTIATE_TEST_CASE_P(/**/, Objdetect_QRCode_Encode_Decode_Structured_Append_Parameterized, EncodeModes::all());
- TEST(Objdetect_QRCode_Encode_Decode, regression_issue22029)
- {
- const cv::String msg = "OpenCV";
- const int min_version = 1;
- const int max_version = 40;
- for ( int v = min_version ; v <= max_version ; v++ )
- {
- SCOPED_TRACE(cv::format("version=%d",v));
- Mat qrimg;
- QRCodeEncoder::Params params;
- params.version = v;
- Ptr<QRCodeEncoder> qrcode_enc = cv::QRCodeEncoder::create(params);
- qrcode_enc->encode(msg, qrimg);
- const int white_margin = 2;
- const int finder_width = 7;
- const int timing_pos = white_margin + 6;
- int i;
- // Horizontal Check
- // (1) White margin(Left)
- for(i = 0; i < white_margin ; i++ )
- {
- ASSERT_EQ((uint8_t)255, qrimg.at<uint8_t>(i, timing_pos)) << "i=" << i;
- }
- // (2) Finder pattern(Left)
- for( ; i < white_margin + finder_width ; i++ )
- {
- ASSERT_EQ((uint8_t)0, qrimg.at<uint8_t>(i, timing_pos)) << "i=" << i;
- }
- // (3) Timing pattern
- for( ; i < qrimg.rows - finder_width - white_margin; i++ )
- {
- ASSERT_EQ((uint8_t)(i % 2 == 0)?0:255, qrimg.at<uint8_t>(i, timing_pos)) << "i=" << i;
- }
- // (4) Finder pattern(Right)
- for( ; i < qrimg.rows - white_margin; i++ )
- {
- ASSERT_EQ((uint8_t)0, qrimg.at<uint8_t>(i, timing_pos)) << "i=" << i;
- }
- // (5) White margin(Right)
- for( ; i < qrimg.rows ; i++ )
- {
- ASSERT_EQ((uint8_t)255, qrimg.at<uint8_t>(i, timing_pos)) << "i=" << i;
- }
- // Vertical Check
- // (1) White margin(Top)
- for(i = 0; i < white_margin ; i++ )
- {
- ASSERT_EQ((uint8_t)255, qrimg.at<uint8_t>(timing_pos, i)) << "i=" << i;
- }
- // (2) Finder pattern(Top)
- for( ; i < white_margin + finder_width ; i++ )
- {
- ASSERT_EQ((uint8_t)0, qrimg.at<uint8_t>(timing_pos, i)) << "i=" << i;
- }
- // (3) Timing pattern
- for( ; i < qrimg.rows - finder_width - white_margin; i++ )
- {
- ASSERT_EQ((uint8_t)(i % 2 == 0)?0:255, qrimg.at<uint8_t>(timing_pos, i)) << "i=" << i;
- }
- // (4) Finder pattern(Bottom)
- for( ; i < qrimg.rows - white_margin; i++ )
- {
- ASSERT_EQ((uint8_t)0, qrimg.at<uint8_t>(timing_pos, i)) << "i=" << i;
- }
- // (5) White margin(Bottom)
- for( ; i < qrimg.rows ; i++ )
- {
- ASSERT_EQ((uint8_t)255, qrimg.at<uint8_t>(timing_pos, i)) << "i=" << i;
- }
- }
- }
- // This test reproduces issue https://github.com/opencv/opencv/issues/24366 only in a loop
- TEST(Objdetect_QRCode_Encode_Decode, auto_version_pick)
- {
- cv::QRCodeEncoder::Params params;
- params.correction_level = cv::QRCodeEncoder::CORRECT_LEVEL_L;
- params.mode = cv::QRCodeEncoder::EncodeMode::MODE_AUTO;
- cv::Ptr<cv::QRCodeEncoder> encoder = cv::QRCodeEncoder::create(params);
- for (int len = 1; len < 19; len++) {
- std::string input;
- input.resize(len);
- cv::randu(Mat(1, len, CV_8U, &input[0]), 'a', 'z' + 1);
- cv::Mat qrcode;
- encoder->encode(input, qrcode);
- }
- }
- // Test two QR codes which error correction procedure requires more number of
- // syndroms that described in the ISO/IEC 18004
- typedef testing::TestWithParam<std::pair<std::string, std::string>> Objdetect_QRCode_decoding;
- TEST_P(Objdetect_QRCode_decoding, error_correction)
- {
- const std::string filename = get<0>(GetParam());
- const std::string expected = get<1>(GetParam());
- QRCodeDetector qrcode;
- cv::String decoded_msg;
- Mat src = cv::imread(findDataFile("qrcode/" + filename), IMREAD_GRAYSCALE);
- std::vector<Point2f> corners(4);
- corners[0] = Point2f(0, 0);
- corners[1] = Point2f(src.cols * 1.0f, 0);
- corners[2] = Point2f(src.cols * 1.0f, src.rows * 1.0f);
- corners[3] = Point2f(0, src.rows * 1.0f);
- Mat resized_src;
- resize(src, resized_src, fixed_size, 0, 0, INTER_AREA);
- float width_ratio = resized_src.cols * 1.0f / src.cols;
- float height_ratio = resized_src.rows * 1.0f / src.rows;
- for(size_t m = 0; m < corners.size(); m++)
- {
- corners[m].x = corners[m].x * width_ratio;
- corners[m].y = corners[m].y * height_ratio;
- }
- Mat straight_barcode;
- EXPECT_NO_THROW(decoded_msg = qrcode.decode(resized_src, corners, straight_barcode));
- ASSERT_FALSE(straight_barcode.empty()) << "Can't decode qrimage " << filename;
- EXPECT_EQ(expected, decoded_msg);
- }
- INSTANTIATE_TEST_CASE_P(/**/, Objdetect_QRCode_decoding, testing::ValuesIn(std::vector<std::pair<std::string, std::string>>{
- {"err_correct_1M.png", "New"},
- {"err_correct_2L.png", "Version 2 QR Code Test Image"},
- }));
- TEST(Objdetect_QRCode_Encode_Decode_Long_Text, regression_issue27183)
- {
- const int len = 135;
- Ptr<QRCodeEncoder> encoder = QRCodeEncoder::create();
- std::string input;
- input.resize(len);
- cv::randu(Mat(1, len, CV_8U, &input[0]), 'a', 'z' + 1);
- Mat qrcode;
- encoder->encode(input, qrcode);
- std::vector<Point2f> corners(4);
- corners[0] = Point2f(border_width, border_width);
- corners[1] = Point2f(qrcode.cols * 1.0f - border_width, border_width);
- corners[2] = Point2f(qrcode.cols * 1.0f - border_width, qrcode.rows * 1.0f - border_width);
- corners[3] = Point2f(border_width, qrcode.rows * 1.0f - border_width);
- Mat resized_src;
- resize(qrcode, resized_src, fixed_size, 0, 0, INTER_AREA);
- float width_ratio = resized_src.cols * 1.0f / qrcode.cols;
- float height_ratio = resized_src.rows * 1.0f / qrcode.rows;
- for(size_t j = 0; j < corners.size(); j++)
- {
- corners[j].x = corners[j].x * width_ratio;
- corners[j].y = corners[j].y * height_ratio;
- }
- QRCodeDetector detector;
- cv::String decoded_msg;
- Mat straight_barcode;
- EXPECT_NO_THROW(decoded_msg = detector.decode(resized_src, corners, straight_barcode));
- ASSERT_FALSE(straight_barcode.empty());
- EXPECT_EQ(input, decoded_msg);
- }
- }} // namespace
|