2021-11-16 01:15:39 +08:00
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2021, Intel Corporation, all rights reserved.
|
|
|
|
|
|
|
|
#include "precomp.hpp"
|
|
|
|
#include "qrcode_encoder_table.inl.hpp"
|
|
|
|
namespace cv
|
|
|
|
{
|
|
|
|
using std::vector;
|
|
|
|
|
|
|
|
const int MAX_PAYLOAD_LEN = 8896;
|
|
|
|
const int MAX_FORMAT_LENGTH = 15;
|
|
|
|
const int MAX_VERSION_LENGTH = 18;
|
|
|
|
const int MODE_BITS_NUM = 4;
|
|
|
|
const uint8_t INVALID_REGION_VALUE = 110;
|
|
|
|
|
|
|
|
static void decToBin(const int dec_number, const int total_bits, std::vector<uint8_t> &bin_number);
|
|
|
|
static uint8_t gfPow(uint8_t x, int power);
|
|
|
|
static uint8_t gfMul(const uint8_t x, const uint8_t y);
|
|
|
|
static void gfPolyMul(const vector<uint8_t> &p, const vector<uint8_t> &q, vector<uint8_t> &product);
|
|
|
|
static void gfPolyDiv(const vector<uint8_t> ÷nd, const vector<uint8_t> &divisor, const int ecc_num, vector<uint8_t> "ient);
|
|
|
|
static void polyGenerator(const int n, vector<uint8_t> &result);
|
|
|
|
static int getBits(const int bits, const vector<uint8_t> &payload, int &pay_index);
|
|
|
|
|
|
|
|
static void decToBin(const int dec_number, const int total_bits, std::vector<uint8_t> &bin_number)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < total_bits; i++)
|
|
|
|
{
|
|
|
|
bin_number[total_bits - i - 1] = (dec_number >> i) % 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void writeDecNumber(const int dec_number, const int total_bits, vector<uint8_t> &output_bits)
|
|
|
|
{
|
|
|
|
std::vector<uint8_t> bin_number(total_bits);
|
|
|
|
decToBin(dec_number, total_bits, bin_number);
|
|
|
|
output_bits.insert(output_bits.end(), bin_number.begin(), bin_number.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t gfPow(uint8_t x, int power)
|
|
|
|
{
|
|
|
|
return gf_exp[(gf_log[x] * power) % 255];
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t gfMul(const uint8_t x, const uint8_t y)
|
|
|
|
{
|
|
|
|
if (x == 0 || y == 0)
|
|
|
|
return 0;
|
|
|
|
return gf_exp[(gf_log[x] + gf_log[y]) % 255];
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gfPolyMul(const vector<uint8_t> &p, const vector<uint8_t> &q, vector<uint8_t> &product)
|
|
|
|
{
|
|
|
|
int len_p = (int)p.size();
|
|
|
|
int len_q = (int)q.size();
|
|
|
|
vector<uint8_t> temp_result(len_p + len_q - 1, 0);
|
|
|
|
|
|
|
|
for (int j = 0; j < len_q; j++)
|
|
|
|
{
|
|
|
|
uint8_t q_val = q[j];
|
|
|
|
if (!q_val)
|
|
|
|
continue;
|
|
|
|
for (int i = 0; i < len_p; i++)
|
|
|
|
{
|
|
|
|
uint8_t p_val = p[i];
|
|
|
|
if (!p_val)
|
|
|
|
continue;
|
|
|
|
temp_result[i + j] ^= gfMul(p_val, q_val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
product = temp_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gfPolyDiv(const vector<uint8_t> ÷nd, const vector<uint8_t> &divisor, const int bits_needed, vector<uint8_t> "ient)
|
|
|
|
{
|
|
|
|
int dividend_len = (int)dividend.size() - 1;
|
|
|
|
int divisor_len = (int)divisor.size() - 1;
|
|
|
|
vector<uint8_t> temp = dividend;
|
|
|
|
|
|
|
|
int times = dividend_len - divisor_len + 1;
|
|
|
|
for (int i = 0; i < times; i++)
|
|
|
|
{
|
|
|
|
uint8_t dividend_val = temp[dividend_len - i];
|
|
|
|
if(dividend_val != 0)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < divisor_len + 1; j++)
|
|
|
|
{
|
|
|
|
uint8_t divisor_val = divisor[divisor_len - j];
|
|
|
|
if (divisor_val != 0)
|
|
|
|
{
|
|
|
|
temp[dividend_len - i - j] ^= gfMul(divisor_val, dividend_val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
quotient = vector<uint8_t>(temp.begin(), temp.begin() + bits_needed);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void polyGenerator(const int n, vector<uint8_t> &result)
|
|
|
|
{
|
|
|
|
vector<uint8_t> temp(2, 1);
|
|
|
|
result = vector<uint8_t>(1, 1);
|
|
|
|
for (int i = 1; i <= n; i++)
|
|
|
|
{
|
|
|
|
temp[0] = gfPow(2, i - 1);
|
|
|
|
gfPolyMul(result, temp, result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int getBits(const int bits, const vector<uint8_t> &payload, int &pay_index)
|
|
|
|
{
|
|
|
|
int result = 0;
|
|
|
|
for (int i = 0; i < bits; i++)
|
|
|
|
{
|
|
|
|
result = result << 1;
|
|
|
|
result += payload[pay_index++];
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mapSymbol(char c)
|
|
|
|
{
|
|
|
|
if (c >= '0' && c <= '9')
|
|
|
|
return (int)(c - '0');
|
|
|
|
if (c >= 'A' && c <= 'Z')
|
|
|
|
return (int)(c - 'A') + 10;
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case ' ': return 36 + 0;
|
|
|
|
case '$': return 36 + 1;
|
|
|
|
case '%': return 36 + 2;
|
|
|
|
case '*': return 36 + 3;
|
|
|
|
case '+': return 36 + 4;
|
|
|
|
case '-': return 36 + 5;
|
|
|
|
case '.': return 36 + 6;
|
|
|
|
case '/': return 36 + 7;
|
|
|
|
case ':': return 36 + 8;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
QRCodeEncoder::QRCodeEncoder()
|
|
|
|
{
|
|
|
|
// nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
QRCodeEncoder::~QRCodeEncoder()
|
|
|
|
{
|
|
|
|
// nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
QRCodeEncoder::Params::Params()
|
|
|
|
{
|
|
|
|
version = 0;
|
|
|
|
correction_level = CORRECT_LEVEL_L;
|
|
|
|
mode = MODE_AUTO;
|
|
|
|
structure_number = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
class QRCodeEncoderImpl : public QRCodeEncoder
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
QRCodeEncoderImpl(const QRCodeEncoder::Params& parameters) : params(parameters)
|
|
|
|
{
|
|
|
|
version_level = parameters.version;
|
|
|
|
ecc_level = parameters.correction_level;
|
|
|
|
mode_type = parameters.mode;
|
|
|
|
struct_num = parameters.structure_number;
|
|
|
|
version_size = 21;
|
|
|
|
mask_type = 0;
|
|
|
|
parity = 0;
|
|
|
|
sequence_num = 0;
|
|
|
|
total_num = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void encode(const String& encoded_info, OutputArray qrcode) CV_OVERRIDE;
|
|
|
|
void encodeStructuredAppend(const String& input, OutputArrayOfArrays output) CV_OVERRIDE;
|
|
|
|
QRCodeEncoder::Params params;
|
|
|
|
protected:
|
|
|
|
int version_level;
|
|
|
|
CorrectionLevel ecc_level;
|
|
|
|
EncodeMode mode_type;
|
|
|
|
int struct_num;
|
|
|
|
int version_size;
|
|
|
|
int mask_type;
|
|
|
|
vector<uint8_t> format;
|
|
|
|
vector<uint8_t> version_reserved;
|
|
|
|
vector<uint8_t> payload;
|
|
|
|
vector<uint8_t> rearranged_data;
|
|
|
|
Mat original;
|
|
|
|
Mat masked_data;
|
|
|
|
uint8_t parity;
|
|
|
|
uint8_t sequence_num;
|
|
|
|
uint8_t total_num;
|
|
|
|
vector<Mat> final_qrcodes;
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
const VersionInfo* version_info;
|
|
|
|
const BlockParams* cur_ecc_params;
|
2021-11-16 01:15:39 +08:00
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
bool isNumeric(const std::string& input) const;
|
|
|
|
bool isAlphaNumeric(const std::string& input) const;
|
|
|
|
EncodeMode autoEncodeMode(const std::string &input) const ;
|
2021-11-16 01:15:39 +08:00
|
|
|
bool encodeByte(const std::string& input, vector<uint8_t> &output);
|
|
|
|
bool encodeAlpha(const std::string& input, vector<uint8_t> &output);
|
|
|
|
bool encodeNumeric(const std::string& input, vector<uint8_t> &output);
|
|
|
|
bool encodeECI(const std::string& input, vector<uint8_t> &output);
|
|
|
|
bool encodeKanji(const std::string& input, vector<uint8_t> &output);
|
2023-11-16 21:28:28 +08:00
|
|
|
bool encodeAuto(const std::string& input, vector<uint8_t> &output, EncodeMode *mode = nullptr);
|
2021-11-16 01:15:39 +08:00
|
|
|
bool encodeStructure(const std::string& input, vector<uint8_t> &output);
|
|
|
|
int eccLevelToCode(CorrectionLevel level);
|
|
|
|
void padBitStream();
|
|
|
|
bool stringToBits(const std::string& input_info);
|
|
|
|
void eccGenerate(vector<vector<uint8_t> > &data_blocks, vector<vector<uint8_t> > &ecc_blocks);
|
|
|
|
void rearrangeBlocks(const vector<vector<uint8_t> > &data_blocks, const vector<vector<uint8_t> > &ecc_blocks);
|
|
|
|
void writeReservedArea();
|
|
|
|
bool writeBit(int x, int y, bool value);
|
|
|
|
void writeData();
|
|
|
|
void structureFinalMessage();
|
|
|
|
void formatGenerate(const int mask_type_num, vector<uint8_t> &format_array);
|
|
|
|
void versionInfoGenerate(const int version_level_num, vector<uint8_t> &version_array);
|
|
|
|
void fillReserved(const vector<uint8_t> &format_array, Mat &masked);
|
|
|
|
void maskData(const int mask_type_num, Mat &masked);
|
|
|
|
void findAutoMaskType();
|
2023-11-16 21:28:28 +08:00
|
|
|
bool estimateVersion(const int input_length, EncodeMode mode, vector<int> &possible_version);
|
2021-11-16 01:15:39 +08:00
|
|
|
int versionAuto(const std::string &input_str);
|
2023-11-16 21:28:28 +08:00
|
|
|
int findVersionCapacity(const int input_length, const int ecc, const std::vector<int>& possible_versions);
|
2021-11-16 01:15:39 +08:00
|
|
|
void generatingProcess(const std::string& input, Mat &qrcode);
|
|
|
|
void generateQR(const std::string& input);
|
|
|
|
};
|
|
|
|
|
|
|
|
int QRCodeEncoderImpl::eccLevelToCode(CorrectionLevel level)
|
|
|
|
{
|
|
|
|
switch (level)
|
|
|
|
{
|
|
|
|
case CORRECT_LEVEL_L:
|
|
|
|
return 0b01;
|
|
|
|
case CORRECT_LEVEL_M:
|
|
|
|
return 0b00;
|
|
|
|
case CORRECT_LEVEL_Q:
|
|
|
|
return 0b11;
|
|
|
|
case CORRECT_LEVEL_H:
|
|
|
|
return 0b10;
|
|
|
|
}
|
|
|
|
CV_Error( Error::StsBadArg,
|
|
|
|
"Error correction level is incorrect. Available levels are"
|
|
|
|
"CORRECT_LEVEL_L, CORRECT_LEVEL_M, CORRECT_LEVEL_Q, CORRECT_LEVEL_H." );
|
|
|
|
}
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
int QRCodeEncoderImpl::findVersionCapacity(const int input_length, const int ecc, const std::vector<int>& possible_versions)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
int data_codewords, version_index = -1;
|
|
|
|
const int byte_len = 8;
|
|
|
|
version_index = -1;
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
for (int i : possible_versions)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
2023-11-16 21:28:28 +08:00
|
|
|
auto& tmp_ecc_params = version_info_database[i].ecc[ecc];
|
|
|
|
data_codewords = tmp_ecc_params.data_codewords_in_G1 * tmp_ecc_params.num_blocks_in_G1 +
|
|
|
|
tmp_ecc_params.data_codewords_in_G2 * tmp_ecc_params.num_blocks_in_G2;
|
2021-11-16 01:15:39 +08:00
|
|
|
|
|
|
|
if (data_codewords * byte_len >= input_length)
|
|
|
|
{
|
|
|
|
version_index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return version_index;
|
|
|
|
}
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
static inline int getCapacity(int version, QRCodeEncoder::CorrectionLevel ecc_level, QRCodeEncoder::EncodeMode mode) {
|
|
|
|
const int* capacity = version_capacity_database[version].ec_level[ecc_level].encoding_modes;
|
|
|
|
switch (mode) {
|
|
|
|
case QRCodeEncoder::EncodeMode::MODE_NUMERIC:
|
|
|
|
return capacity[0];
|
|
|
|
case QRCodeEncoder::EncodeMode::MODE_ALPHANUMERIC:
|
|
|
|
return capacity[1];
|
|
|
|
case QRCodeEncoder::EncodeMode::MODE_BYTE:
|
|
|
|
return capacity[2];
|
|
|
|
case QRCodeEncoder::EncodeMode::MODE_KANJI:
|
|
|
|
return capacity[3];
|
|
|
|
default:
|
|
|
|
CV_Error(Error::StsNotImplemented, format("Unexpected mode %d", mode));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::estimateVersion(const int input_length, EncodeMode mode, vector<int>& possible_version)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
possible_version.clear();
|
2023-11-16 21:28:28 +08:00
|
|
|
|
|
|
|
CV_Assert(mode != EncodeMode::MODE_AUTO);
|
|
|
|
|
|
|
|
if (input_length > getCapacity(MAX_VERSION, ecc_level, mode))
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
2023-11-16 21:28:28 +08:00
|
|
|
return false;
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
2023-11-16 21:28:28 +08:00
|
|
|
|
|
|
|
int version = MAX_VERSION;
|
|
|
|
|
|
|
|
for (; version > 0; --version)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
2023-11-16 21:28:28 +08:00
|
|
|
if (input_length > getCapacity(version, ecc_level, mode)) {
|
|
|
|
break;
|
|
|
|
}
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
2023-11-16 21:28:28 +08:00
|
|
|
|
|
|
|
if (version < MAX_VERSION)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
2023-11-16 21:28:28 +08:00
|
|
|
version += 1;
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
2023-11-16 21:28:28 +08:00
|
|
|
|
|
|
|
possible_version.push_back(version);
|
|
|
|
|
|
|
|
if (version < MAX_VERSION)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
2023-11-16 21:28:28 +08:00
|
|
|
possible_version.push_back(version + 1);
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
2023-11-16 21:28:28 +08:00
|
|
|
|
2021-11-16 01:15:39 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int QRCodeEncoderImpl::versionAuto(const std::string& input_str)
|
|
|
|
{
|
|
|
|
vector<uint8_t> payload_tmp;
|
2023-11-16 21:28:28 +08:00
|
|
|
EncodeMode mode;
|
|
|
|
encodeAuto(input_str, payload_tmp, &mode);
|
2021-12-02 20:04:04 +08:00
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
vector<int> possible_version;
|
|
|
|
if (!estimateVersion((int)input_str.length(), mode, possible_version)) {
|
|
|
|
return -1;
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
2023-11-16 21:28:28 +08:00
|
|
|
|
|
|
|
const auto tmp_version = findVersionCapacity((int)payload_tmp.size(), ecc_level, possible_version);
|
|
|
|
|
2021-11-16 01:15:39 +08:00
|
|
|
return tmp_version;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::generateQR(const std::string &input)
|
|
|
|
{
|
|
|
|
if (struct_num > 1)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < input.length(); i++)
|
|
|
|
{
|
|
|
|
parity ^= input[i];
|
|
|
|
}
|
|
|
|
if (struct_num > 16)
|
|
|
|
{
|
|
|
|
struct_num = 16;
|
|
|
|
}
|
|
|
|
total_num = (uint8_t) struct_num - 1;
|
|
|
|
}
|
|
|
|
|
2023-01-22 08:55:18 +08:00
|
|
|
auto string_itr = input.begin();
|
|
|
|
for (int i = struct_num; i > 0; --i)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
sequence_num = (uint8_t) i;
|
2023-01-22 08:55:18 +08:00
|
|
|
size_t segment_begin = string_itr - input.begin();
|
|
|
|
size_t segment_end = (input.end() - string_itr) / i;
|
|
|
|
|
|
|
|
std::string input_info = input.substr(segment_begin, segment_end);
|
|
|
|
string_itr += segment_end;
|
2023-11-16 21:28:28 +08:00
|
|
|
|
2021-12-02 20:04:04 +08:00
|
|
|
int detected_version = versionAuto(input_info);
|
2023-11-16 21:28:28 +08:00
|
|
|
int tmp_version_level = version_level;
|
|
|
|
if (detected_version == -1)
|
|
|
|
CV_Error(Error::StsBadArg, "The given input exceeds the maximum capacity of a QR code with the selected encoding mode and error correction level " );
|
|
|
|
else if (tmp_version_level == 0)
|
|
|
|
tmp_version_level = detected_version;
|
|
|
|
else if (tmp_version_level < detected_version)
|
2021-11-16 01:15:39 +08:00
|
|
|
CV_Error(Error::StsBadArg, "The given version is not suitable for the given input string length ");
|
|
|
|
|
|
|
|
payload.clear();
|
|
|
|
payload.reserve(MAX_PAYLOAD_LEN);
|
|
|
|
format = vector<uint8_t> (15, 255);
|
|
|
|
version_reserved = vector<uint8_t> (18, 255);
|
2023-11-16 21:28:28 +08:00
|
|
|
version_size = (21 + (tmp_version_level - 1) * 4);
|
|
|
|
version_info = &version_info_database[tmp_version_level];
|
|
|
|
cur_ecc_params = &version_info->ecc[ecc_level];
|
2021-11-16 01:15:39 +08:00
|
|
|
original = Mat(Size(version_size, version_size), CV_8UC1, Scalar(255));
|
|
|
|
masked_data = original.clone();
|
|
|
|
Mat qrcode = masked_data.clone();
|
|
|
|
generatingProcess(input_info, qrcode);
|
|
|
|
final_qrcodes.push_back(qrcode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::formatGenerate(const int mask_type_num, vector<uint8_t> &format_array)
|
|
|
|
{
|
|
|
|
const int mask_bits_num = 3;
|
|
|
|
const int level_bits_num = 2;
|
|
|
|
|
|
|
|
std::vector<uint8_t> mask_type_bin(mask_bits_num);
|
|
|
|
std::vector<uint8_t> ec_level_bin(level_bits_num);
|
|
|
|
decToBin(mask_type_num, mask_bits_num, mask_type_bin);
|
|
|
|
decToBin(eccLevelToCode(ecc_level), level_bits_num, ec_level_bin);
|
|
|
|
|
|
|
|
std::vector<uint8_t> format_bits;
|
|
|
|
hconcat(ec_level_bin, mask_type_bin, format_bits);
|
|
|
|
std::reverse(format_bits.begin(), format_bits.end());
|
|
|
|
|
|
|
|
const int ecc_info_bits = 10;
|
|
|
|
|
|
|
|
std::vector<uint8_t> shift(ecc_info_bits, 0);
|
|
|
|
std::vector<uint8_t> polynomial;
|
|
|
|
hconcat(shift, format_bits, polynomial);
|
|
|
|
|
|
|
|
const int generator_len = 11;
|
|
|
|
const uint8_t generator_arr[generator_len] = {1, 1, 1, 0, 1, 1, 0, 0, 1, 0, 1};
|
|
|
|
std::vector<uint8_t> format_generator (generator_arr, generator_arr + sizeof(generator_arr) / sizeof(generator_arr[0]));
|
|
|
|
vector<uint8_t> ecc_code;
|
|
|
|
gfPolyDiv(polynomial, format_generator, ecc_info_bits, ecc_code);
|
|
|
|
hconcat(ecc_code, format_bits, format_array);
|
|
|
|
|
|
|
|
const uint8_t mask_arr[MAX_FORMAT_LENGTH] = {0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 1, 0, 1};
|
|
|
|
std::vector<uint8_t> system_mask (mask_arr, mask_arr + sizeof(mask_arr) / sizeof(mask_arr[0]));
|
|
|
|
for(int i = 0; i < MAX_FORMAT_LENGTH; i++)
|
|
|
|
{
|
|
|
|
format_array[i] ^= system_mask[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::versionInfoGenerate(const int version_level_num, vector<uint8_t> &version_array)
|
|
|
|
{
|
|
|
|
const int version_bits_num = 6;
|
|
|
|
std::vector<uint8_t> version_bits(version_bits_num);
|
|
|
|
decToBin(version_level_num, version_bits_num, version_bits);
|
|
|
|
|
|
|
|
std::reverse(version_bits.begin(), version_bits.end());
|
|
|
|
vector<uint8_t> shift(12, 0);
|
|
|
|
vector<uint8_t> polynomial;
|
|
|
|
hconcat(shift, version_bits, polynomial);
|
|
|
|
|
|
|
|
const int generator_len = 13;
|
|
|
|
const uint8_t generator_arr[generator_len] = {1, 0, 1, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1};
|
|
|
|
std::vector<uint8_t> format_mask (generator_arr, generator_arr + sizeof(generator_arr) / sizeof(generator_arr[0]));
|
|
|
|
|
|
|
|
vector<uint8_t> ecc_code;
|
|
|
|
gfPolyDiv(polynomial, format_mask, 12, ecc_code);
|
|
|
|
hconcat(ecc_code, version_bits, version_array);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeAlpha(const std::string& input, vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
writeDecNumber(MODE_ALPHANUMERIC, MODE_BITS_NUM, output);
|
|
|
|
|
|
|
|
int length_bits_num = 13;
|
|
|
|
if (version_level < 10)
|
|
|
|
length_bits_num = 9;
|
|
|
|
else if (version_level < 27)
|
|
|
|
length_bits_num = 11;
|
|
|
|
|
|
|
|
int str_len = int(input.length());
|
|
|
|
writeDecNumber(str_len, length_bits_num, output);
|
|
|
|
|
|
|
|
const int alpha_symbol_bits = 11;
|
|
|
|
const int residual_bits = 6;
|
|
|
|
for (int i = 0; i < str_len - 1; i += 2)
|
|
|
|
{
|
|
|
|
int index_1 = mapSymbol(input[i]);
|
|
|
|
int index_2 = mapSymbol(input[i + 1]);
|
|
|
|
|
|
|
|
if(index_1 == -1 || (index_2 == -1 && i + 1 < str_len))
|
|
|
|
return false;
|
|
|
|
int alpha = index_1 * 45 + index_2;
|
|
|
|
|
|
|
|
writeDecNumber(alpha, alpha_symbol_bits, output);
|
|
|
|
}
|
|
|
|
if (str_len % 2 != 0)
|
|
|
|
{
|
|
|
|
int index_residual_elem = mapSymbol(*input.rbegin());
|
|
|
|
if(index_residual_elem == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
writeDecNumber(index_residual_elem, residual_bits, output);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeByte(const std::string& input, vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
writeDecNumber(MODE_BYTE, MODE_BITS_NUM, output);
|
|
|
|
|
|
|
|
int length_bits_num = 8;
|
|
|
|
if (version_level > 9)
|
|
|
|
length_bits_num = 16;
|
|
|
|
|
|
|
|
int str_len = int(input.length());
|
|
|
|
writeDecNumber(str_len, length_bits_num, output);
|
|
|
|
|
|
|
|
const int byte_symbol_bits = 8;
|
|
|
|
for (int i = 0; i < str_len; i++)
|
|
|
|
{
|
|
|
|
writeDecNumber(uint8_t(input[i]), byte_symbol_bits, output);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeNumeric(const std::string& input,vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
writeDecNumber(MODE_NUMERIC, MODE_BITS_NUM, output);
|
|
|
|
|
|
|
|
int length_bits_num = 10;
|
|
|
|
if (version_level >= 27)
|
|
|
|
length_bits_num = 14;
|
|
|
|
else if (version_level >= 10)
|
|
|
|
length_bits_num = 12;
|
|
|
|
|
|
|
|
int str_len = int(input.length());
|
|
|
|
writeDecNumber(str_len, length_bits_num, output);
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
const int num_symbol_bits = 10;
|
|
|
|
const int residual_bits[2] = {7, 4};
|
|
|
|
while (count + 3 <= str_len)
|
|
|
|
{
|
|
|
|
if (input[count] > '9' || input[count] < '0' ||
|
|
|
|
input[count + 1] > '9' || input[count + 1] < '0' ||
|
|
|
|
input[count + 2] > '9' || input[count + 2] < '0')
|
|
|
|
return false;
|
|
|
|
int num = 100 * (int)(input[count] - '0') +
|
|
|
|
10 * (int)(input[count + 1] - '0') +
|
|
|
|
(int)(input[count + 2] - '0');
|
|
|
|
|
|
|
|
writeDecNumber(num, num_symbol_bits, output);
|
|
|
|
count += 3;
|
|
|
|
}
|
|
|
|
if (count + 2 == str_len)
|
|
|
|
{
|
|
|
|
if (input[count] > '9' || input[count] < '0'||
|
|
|
|
input[count + 1] >'9'|| input[count + 1] < '0')
|
|
|
|
return false;
|
|
|
|
int num = 10 * (int)(input[count] - '0') +
|
|
|
|
(int)(input[count + 1] - '0');
|
|
|
|
|
|
|
|
writeDecNumber(num, residual_bits[0], output);
|
|
|
|
}
|
|
|
|
else if (count + 1 == str_len)
|
|
|
|
{
|
|
|
|
if (input[count] > '9' || input[count] < '0')
|
|
|
|
return false;
|
|
|
|
int num = (int)(input[count] - '0');
|
|
|
|
|
|
|
|
writeDecNumber(num, residual_bits[1], output);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeECI(const std::string& input, vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
writeDecNumber(MODE_ECI, MODE_BITS_NUM, output);
|
|
|
|
const uint32_t assign_value_range[3] = {127, 16383, 999999};
|
|
|
|
|
|
|
|
// by adding other ECI modes `eci_assignment_number` can be moved to algorithm parameters
|
|
|
|
uint32_t eci_assignment_number = ECI_UTF8; // utf-8
|
|
|
|
|
|
|
|
int codewords = 1;
|
|
|
|
if(eci_assignment_number > assign_value_range[2])
|
|
|
|
return false;
|
|
|
|
if (eci_assignment_number > assign_value_range[1])
|
|
|
|
codewords = 3;
|
|
|
|
else if (eci_assignment_number > assign_value_range[0])
|
|
|
|
codewords = 2;
|
|
|
|
|
|
|
|
const int bits = 8;
|
|
|
|
switch (codewords)
|
|
|
|
{
|
|
|
|
case 1:
|
|
|
|
writeDecNumber(0, codewords, output);
|
|
|
|
writeDecNumber(eci_assignment_number, codewords * bits - 1, output);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
writeDecNumber(2, codewords, output);
|
|
|
|
writeDecNumber(eci_assignment_number, codewords * bits - 2, output);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
writeDecNumber(6, codewords, output);
|
|
|
|
writeDecNumber(eci_assignment_number, codewords * bits - 3, output);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeByte(input, output);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeKanji(const std::string& input, vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
writeDecNumber(MODE_KANJI, MODE_BITS_NUM, output);
|
|
|
|
|
|
|
|
int length_bits_num = 8;
|
|
|
|
if (version_level >= 10)
|
|
|
|
length_bits_num = 10;
|
|
|
|
else if (version_level >= 27)
|
|
|
|
length_bits_num = 12;
|
|
|
|
|
|
|
|
int str_len = int(input.length()) / 2;
|
|
|
|
writeDecNumber(str_len, length_bits_num, output);
|
|
|
|
|
|
|
|
const int kanji_symbol_bits = 13;
|
|
|
|
int i = 0;
|
|
|
|
while(i < str_len * 2)
|
|
|
|
{
|
|
|
|
uint16_t high_byte = (uint16_t)(input[i] & 0xff);
|
|
|
|
uint16_t low_byte = (uint16_t)(input[i+1] & 0xff);
|
|
|
|
uint16_t per_char = (high_byte << 8) + (low_byte);
|
|
|
|
|
|
|
|
if(0x8140 <= per_char && per_char <= 0x9FFC)
|
|
|
|
{
|
|
|
|
per_char -= 0x8140;
|
|
|
|
}
|
|
|
|
else if(0xE040 <= per_char && per_char <= 0xEBBF)
|
|
|
|
{
|
|
|
|
per_char -= 0xC140;
|
|
|
|
}
|
|
|
|
uint16_t new_high = per_char >> 8;
|
|
|
|
uint16_t result = new_high * 0xC0;
|
|
|
|
result += (per_char & 0xFF);
|
|
|
|
|
|
|
|
writeDecNumber(result, kanji_symbol_bits, output);
|
|
|
|
i += 2;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeStructure(const std::string& input, vector<uint8_t>& output)
|
|
|
|
{
|
|
|
|
const int num_field = 4;
|
|
|
|
const int checksum_field = 8;
|
|
|
|
writeDecNumber(MODE_STRUCTURED_APPEND, MODE_BITS_NUM, output);
|
|
|
|
writeDecNumber(sequence_num, num_field, output);
|
|
|
|
writeDecNumber(total_num, num_field, output);
|
|
|
|
writeDecNumber(parity, checksum_field, output);
|
|
|
|
|
|
|
|
return encodeAuto(input, output);
|
|
|
|
}
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
bool QRCodeEncoderImpl::isNumeric(const std::string& input) const
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
for (size_t i = 0; i < input.length(); i++)
|
|
|
|
{
|
|
|
|
if (input[i] < '0' || input[i] > '9')
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
bool QRCodeEncoderImpl::isAlphaNumeric(const std::string& input) const
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
for (size_t i = 0; i < input.length(); i++)
|
|
|
|
{
|
|
|
|
if (mapSymbol(input[i]) == -1)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-11-16 21:28:28 +08:00
|
|
|
QRCodeEncoder::EncodeMode QRCodeEncoderImpl::autoEncodeMode(const std::string &input) const
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
if (isNumeric(input))
|
2023-11-16 21:28:28 +08:00
|
|
|
{
|
|
|
|
return EncodeMode::MODE_NUMERIC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isAlphaNumeric(input))
|
|
|
|
{
|
|
|
|
return EncodeMode::MODE_ALPHANUMERIC;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EncodeMode::MODE_BYTE;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::encodeAuto(const std::string& input, vector<uint8_t>& output, EncodeMode *mode)
|
|
|
|
{
|
|
|
|
const auto selected_mode = autoEncodeMode(input);
|
|
|
|
|
|
|
|
CV_Assert(selected_mode != EncodeMode::MODE_AUTO);
|
|
|
|
|
|
|
|
switch (selected_mode)
|
|
|
|
{
|
|
|
|
case EncodeMode::MODE_NUMERIC:
|
|
|
|
encodeNumeric(input, output);
|
|
|
|
break;
|
|
|
|
case EncodeMode::MODE_ALPHANUMERIC:
|
|
|
|
encodeAlpha(input, output);
|
|
|
|
break;
|
|
|
|
case EncodeMode::MODE_STRUCTURED_APPEND:
|
|
|
|
encodeByte(input, output);
|
|
|
|
break;
|
|
|
|
case EncodeMode::MODE_BYTE:
|
|
|
|
encodeByte(input, output);
|
|
|
|
break;
|
|
|
|
case EncodeMode::MODE_KANJI:
|
|
|
|
encodeKanji(input, output);
|
|
|
|
break;
|
|
|
|
case EncodeMode::MODE_ECI:
|
|
|
|
encodeECI(input, output);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode != nullptr)
|
|
|
|
{
|
|
|
|
*mode = selected_mode;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:39 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::padBitStream()
|
|
|
|
{
|
|
|
|
int total_data = version_info->total_codewords -
|
|
|
|
cur_ecc_params->ecc_codewords * (cur_ecc_params->num_blocks_in_G1 + cur_ecc_params->num_blocks_in_G2);
|
|
|
|
const int bits = 8;
|
|
|
|
total_data *= bits;
|
|
|
|
int pad_num = total_data - (int)payload.size();
|
|
|
|
|
|
|
|
if (pad_num <= 0)
|
|
|
|
return;
|
|
|
|
else if (pad_num <= 4)
|
|
|
|
{
|
|
|
|
int payload_size = (int)payload.size();
|
|
|
|
writeDecNumber(0, payload_size, payload);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
writeDecNumber(0, 4, payload);
|
|
|
|
|
|
|
|
int i = payload.size() % bits;
|
|
|
|
|
|
|
|
if (i != 0)
|
|
|
|
{
|
|
|
|
writeDecNumber(0, bits - i, payload);
|
|
|
|
}
|
|
|
|
pad_num = total_data - (int)payload.size();
|
|
|
|
|
|
|
|
if (pad_num > 0)
|
|
|
|
{
|
|
|
|
const int pad_patterns[2] = {236, 17};
|
|
|
|
int num = pad_num / bits;
|
|
|
|
const int pattern_size = 8;
|
|
|
|
for (int j = 0; j < num; j++)
|
|
|
|
{
|
|
|
|
writeDecNumber(pad_patterns[j % 2], pattern_size, payload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::stringToBits(const std::string& input_info)
|
|
|
|
{
|
|
|
|
switch (mode_type)
|
|
|
|
{
|
|
|
|
case MODE_NUMERIC:
|
|
|
|
return encodeNumeric(input_info, payload);
|
|
|
|
case MODE_ALPHANUMERIC:
|
|
|
|
return encodeAlpha(input_info, payload);
|
|
|
|
case MODE_STRUCTURED_APPEND:
|
|
|
|
return encodeStructure(input_info, payload);
|
|
|
|
case MODE_BYTE:
|
|
|
|
return encodeByte(input_info, payload);
|
|
|
|
case MODE_ECI:
|
|
|
|
return encodeECI(input_info, payload);
|
|
|
|
case MODE_KANJI:
|
|
|
|
return encodeKanji(input_info, payload);
|
|
|
|
default:
|
|
|
|
return encodeAuto(input_info, payload);
|
|
|
|
}
|
2023-10-06 18:33:21 +08:00
|
|
|
}
|
2021-11-16 01:15:39 +08:00
|
|
|
|
|
|
|
void QRCodeEncoderImpl::eccGenerate(vector<vector<uint8_t> > &data_blocks, vector<vector<uint8_t> > &ecc_blocks)
|
|
|
|
{
|
|
|
|
const int ec_codewords = cur_ecc_params->ecc_codewords;
|
|
|
|
int pay_index = 0;
|
|
|
|
vector<uint8_t> g_x;
|
|
|
|
polyGenerator(ec_codewords, g_x);
|
|
|
|
int blocks = cur_ecc_params->num_blocks_in_G2 + cur_ecc_params->num_blocks_in_G1;
|
|
|
|
for (int i = 0; i < blocks; i++)
|
|
|
|
{
|
|
|
|
int block_len = 0;
|
|
|
|
|
|
|
|
if (i < cur_ecc_params->num_blocks_in_G1)
|
|
|
|
{
|
|
|
|
block_len = cur_ecc_params->data_codewords_in_G1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
block_len = cur_ecc_params->data_codewords_in_G2;
|
|
|
|
}
|
|
|
|
vector<uint8_t> block_i (block_len, 0);
|
|
|
|
|
|
|
|
for (int j = 0; j < block_len; j++)
|
|
|
|
{
|
|
|
|
block_i[block_len - 1 - j] = (uchar)getBits(8, payload, pay_index);
|
|
|
|
}
|
|
|
|
vector<uint8_t> dividend;
|
|
|
|
vector<uint8_t> shift (ec_codewords, 0);
|
|
|
|
hconcat(shift, block_i, dividend);
|
|
|
|
vector<uint8_t> ecc_i;
|
|
|
|
gfPolyDiv(dividend, g_x, ec_codewords, ecc_i);
|
|
|
|
data_blocks.push_back(block_i);
|
|
|
|
ecc_blocks.push_back(ecc_i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::rearrangeBlocks(const vector<vector<uint8_t> > &data_blocks, const vector<vector<uint8_t> > &ecc_blocks)
|
|
|
|
{
|
|
|
|
rearranged_data.clear();
|
|
|
|
int blocks = cur_ecc_params->num_blocks_in_G2 + cur_ecc_params->num_blocks_in_G1;
|
|
|
|
int col_border = max(cur_ecc_params->data_codewords_in_G2, cur_ecc_params->data_codewords_in_G1);
|
|
|
|
int total_codeword_num = version_info->total_codewords;
|
|
|
|
int is_not_equal = cur_ecc_params->data_codewords_in_G2 - cur_ecc_params->data_codewords_in_G1;
|
2021-12-02 20:04:04 +08:00
|
|
|
int add_steps = cur_ecc_params->data_codewords_in_G2 > cur_ecc_params->data_codewords_in_G1 ?
|
|
|
|
(cur_ecc_params->data_codewords_in_G2 - cur_ecc_params->data_codewords_in_G1) * cur_ecc_params->num_blocks_in_G1 : 0;
|
|
|
|
rearranged_data.reserve(total_codeword_num + add_steps);
|
|
|
|
for (int i = 0; i < total_codeword_num + add_steps; i++)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
int cur_col = i / blocks;
|
|
|
|
int cur_row = i % blocks;
|
|
|
|
int data_col = (int)data_blocks[cur_row].size() - 1;
|
|
|
|
int ecc_col = (int)ecc_blocks[cur_row].size() - 1;
|
|
|
|
uint8_t tmp = 0;
|
|
|
|
|
|
|
|
if (cur_col < col_border)
|
|
|
|
{
|
|
|
|
if (is_not_equal && cur_col == cur_ecc_params->data_codewords_in_G2 - 1 && cur_row < cur_ecc_params->num_blocks_in_G1)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tmp = data_blocks[cur_row][data_col - cur_col];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int index = ecc_col - (cur_col - col_border);
|
|
|
|
tmp = ecc_blocks[cur_row][index];
|
|
|
|
}
|
|
|
|
rearranged_data.push_back(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::findAutoMaskType()
|
|
|
|
{
|
|
|
|
int best_index = 0;
|
|
|
|
int lowest_penalty = INT_MAX;
|
|
|
|
int penalty_two_value = 3, penalty_three_value = 40;
|
|
|
|
for (int cur_type = 0; cur_type < 8; cur_type++)
|
|
|
|
{
|
|
|
|
Mat test_result = masked_data.clone();
|
|
|
|
vector<uint8_t> test_format = format;
|
|
|
|
maskData(cur_type, test_result);
|
|
|
|
formatGenerate(cur_type, test_format);
|
|
|
|
fillReserved(test_format, test_result);
|
|
|
|
int continued_num = 0;
|
|
|
|
int penalty_one = 0, penalty_two = 0, penalty_three = 0, penalty_four = 0, penalty_total = 0;
|
|
|
|
int current_color = -1;
|
|
|
|
|
|
|
|
for (int direction = 0; direction < 2; direction++)
|
|
|
|
{
|
|
|
|
if (direction != 0)
|
|
|
|
{
|
|
|
|
test_result = test_result.t();
|
|
|
|
}
|
|
|
|
for (int i = 0; i < version_size; i++)
|
|
|
|
{
|
|
|
|
int per_row = 0;
|
|
|
|
for (int j = 0; j < version_size; j++)
|
|
|
|
{
|
|
|
|
if (j == 0)
|
|
|
|
{
|
|
|
|
current_color = test_result.at<uint8_t>(i, j);
|
|
|
|
continued_num = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (current_color == test_result.at<uint8_t>(i, j))
|
|
|
|
{
|
|
|
|
continued_num += 1;
|
|
|
|
}
|
|
|
|
if (current_color != test_result.at<uint8_t>(i, j) || j + 1 == version_size)
|
|
|
|
{
|
|
|
|
current_color = test_result.at<uint8_t>(i, j);
|
|
|
|
if (continued_num >= 5)
|
|
|
|
{
|
|
|
|
per_row += 3 + continued_num - 5;
|
|
|
|
}
|
|
|
|
continued_num = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
penalty_one += per_row;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < version_size - 1; i++)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < version_size - 1; j++)
|
|
|
|
{
|
|
|
|
uint8_t color = test_result.at<uint8_t>(i, j);
|
|
|
|
if (color == test_result.at<uint8_t>(i, j + 1) &&
|
|
|
|
color == test_result.at<uint8_t>(i + 1, j + 1) &&
|
|
|
|
color == test_result.at<uint8_t>(i + 1, j))
|
|
|
|
{
|
|
|
|
penalty_two += penalty_two_value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Mat penalty_pattern[2];
|
|
|
|
penalty_pattern[0] = (Mat_<uint8_t >(1, 11) << 255, 255, 255, 255, 0, 255, 0, 0, 0, 255, 0);
|
|
|
|
penalty_pattern[1] = (Mat_<uint8_t >(1, 11) << 0, 255, 0, 0, 0, 255, 0, 255, 255, 255, 255);
|
|
|
|
for (int direction = 0; direction < 2; direction++)
|
|
|
|
{
|
|
|
|
if (direction != 0)
|
|
|
|
{
|
|
|
|
test_result = test_result.t();
|
|
|
|
}
|
|
|
|
for (int i = 0; i < version_size; i++)
|
|
|
|
{
|
|
|
|
int per_row = 0;
|
|
|
|
for (int j = 0; j < version_size - 10; j++)
|
|
|
|
{
|
|
|
|
Mat cur_test = test_result(Range(i, i + 1), Range(j, j + 11));
|
|
|
|
for (int pattern_index = 0; pattern_index < 2; pattern_index++)
|
|
|
|
{
|
|
|
|
Mat diff = (penalty_pattern[pattern_index] != cur_test);
|
|
|
|
bool equal = (countNonZero(diff) == 0);
|
|
|
|
if (equal)
|
|
|
|
{
|
|
|
|
per_row += penalty_three_value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
penalty_three += per_row;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int dark_modules = 0;
|
|
|
|
int total_modules = 0;
|
|
|
|
for (int i = 0; i < version_size; i++)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < version_size; j++)
|
|
|
|
{
|
|
|
|
if (test_result.at<uint8_t>(i, j) == 0)
|
|
|
|
{
|
|
|
|
dark_modules += 1;
|
|
|
|
}
|
|
|
|
total_modules += 1;
|
|
|
|
}
|
|
|
|
}
|
2021-12-15 17:49:13 +08:00
|
|
|
if (total_modules == 0)
|
|
|
|
continue; // TODO: refactor, extract functions to reduce complexity
|
2021-11-16 01:15:39 +08:00
|
|
|
int modules_percent = dark_modules * 100 / total_modules;
|
|
|
|
int lower_bound = 45;
|
|
|
|
int upper_bound = 55;
|
|
|
|
int diff = min(abs(modules_percent - lower_bound), abs(modules_percent - upper_bound));
|
|
|
|
penalty_four = (diff / 5) * 10;
|
|
|
|
penalty_total = penalty_one + penalty_two + penalty_three + penalty_four;
|
|
|
|
if (penalty_total < lowest_penalty)
|
|
|
|
{
|
|
|
|
best_index = cur_type;
|
|
|
|
lowest_penalty = penalty_total;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mask_type = best_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::maskData(const int mask_type_num, Mat& masked)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < version_size; i++)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < version_size; j++)
|
|
|
|
{
|
|
|
|
if (original.at<uint8_t>(i, j) == INVALID_REGION_VALUE)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if((mask_type_num == 0 && !((i + j) % 2)) ||
|
|
|
|
(mask_type_num == 1 && !(i % 2)) ||
|
|
|
|
(mask_type_num == 2 && !(j % 3)) ||
|
|
|
|
(mask_type_num == 3 && !((i + j) % 3)) ||
|
|
|
|
(mask_type_num == 4 && !(((i / 2) + (j / 3)) % 2)) ||
|
|
|
|
(mask_type_num == 5 && !((i * j) % 2 + (i * j) % 3))||
|
|
|
|
(mask_type_num == 6 && !(((i * j) % 2 + (i * j) % 3) % 2))||
|
|
|
|
((mask_type_num == 7 && !(((i * j) % 3 + (i + j) % 2) % 2))))
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(i, j) = original.at<uint8_t>(i, j) ^ 255;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::writeReservedArea()
|
|
|
|
{
|
|
|
|
vector<Rect> finder_pattern(3);
|
|
|
|
finder_pattern[0] = Rect(Point(0, 0), Point(9, 9));
|
|
|
|
finder_pattern[1] = Rect(Point(0, (unsigned)version_size - 8), Point(9, version_size));
|
|
|
|
finder_pattern[2] = Rect(Point((unsigned)version_size - 8, 0), Point(version_size, 9));
|
|
|
|
const int coordinates_num = 2;
|
|
|
|
int locator_position[coordinates_num] = {3, version_size - 1 - 3};
|
|
|
|
|
|
|
|
for (int first_coordinate = 0; first_coordinate < coordinates_num; first_coordinate++)
|
|
|
|
{
|
|
|
|
for (int second_coordinate = 0; second_coordinate < coordinates_num; second_coordinate++)
|
|
|
|
{
|
|
|
|
if (first_coordinate == 1 && second_coordinate == 1)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
int x = locator_position[first_coordinate];
|
|
|
|
int y = locator_position[second_coordinate];
|
|
|
|
for (int i = -5; i <= 5; i++)
|
|
|
|
{
|
|
|
|
for (int j = -5; j <= 5; j++)
|
|
|
|
{
|
|
|
|
if (x + i < 0 || x + i >= version_size || y + j < 0 || y + j >= version_size)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!(abs(j) == 2 && abs(i) <= 2) &&
|
|
|
|
!(abs(j) <= 2 && abs(i) == 2) &&
|
|
|
|
!(abs(i) == 4) && !(abs(j) == 4))
|
|
|
|
{
|
|
|
|
masked_data.at<uint8_t>(x + i, y + j) = 0;
|
|
|
|
}
|
|
|
|
if ((y == locator_position[1] && j == -5) || (x == locator_position[1] && i == -5))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
original.at<uint8_t>(x + i, y + j) = INVALID_REGION_VALUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int x = locator_position[1] - 4;
|
|
|
|
int y = locator_position[0] + 5;
|
|
|
|
masked_data.at<uint8_t>(x, y) = 0;
|
|
|
|
original.at<uint8_t>(x, y) = INVALID_REGION_VALUE;
|
|
|
|
if (version_level >= 7)
|
|
|
|
{
|
2022-05-31 18:58:17 +08:00
|
|
|
for (int i = 0; i <= 5; i++)
|
2021-11-16 01:15:39 +08:00
|
|
|
{
|
|
|
|
for (int j = version_size - 11; j <= version_size - 8; j++)
|
|
|
|
{
|
|
|
|
original.at<uint8_t>(i, j) = INVALID_REGION_VALUE;
|
|
|
|
original.at<uint8_t>(j, i) = INVALID_REGION_VALUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < version_size; i++)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < version_size; j++)
|
|
|
|
{
|
|
|
|
if (original.at<uint8_t>(i, j) == INVALID_REGION_VALUE)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ((i == 6 || j == 6))
|
|
|
|
{
|
|
|
|
original.at<uint8_t>(i, j) = INVALID_REGION_VALUE;
|
|
|
|
if (!((i == 6) && (j - 7) % 2 == 0) &&
|
|
|
|
!((j == 6) && ((i - 7) % 2 == 0)))
|
|
|
|
{
|
|
|
|
masked_data.at<uint8_t>(i, j) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int first_coord = 0; first_coord < MAX_ALIGNMENT && version_info->alignment_pattern[first_coord]; first_coord++)
|
|
|
|
{
|
|
|
|
for (int second_coord = 0; second_coord < MAX_ALIGNMENT && version_info->alignment_pattern[second_coord]; second_coord++)
|
|
|
|
{
|
|
|
|
x = version_info->alignment_pattern[first_coord];
|
|
|
|
y = version_info->alignment_pattern[second_coord];
|
|
|
|
bool is_in_finder = false;
|
|
|
|
for (size_t i = 0; i < finder_pattern.size(); i++)
|
|
|
|
{
|
|
|
|
Rect rect = finder_pattern[i];
|
|
|
|
if (x >= rect.tl().x && x <= rect.br().x
|
|
|
|
&&
|
|
|
|
y >= rect.tl().y && y <= rect.br().y)
|
|
|
|
{
|
|
|
|
is_in_finder = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!is_in_finder)
|
|
|
|
{
|
|
|
|
for (int i = -2; i <= 2; i++)
|
|
|
|
{
|
|
|
|
for (int j = -2; j <= 2; j++)
|
|
|
|
{
|
|
|
|
original.at<uint8_t>(x + i, y + j) = INVALID_REGION_VALUE;
|
|
|
|
if ((j == 0 && i == 0) || (abs(j) == 2) || abs(i) == 2)
|
|
|
|
{
|
|
|
|
masked_data.at<uint8_t>(x + i, y + j) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QRCodeEncoderImpl::writeBit(int x, int y, bool value)
|
|
|
|
{
|
|
|
|
if (original.at<uint8_t>(y, x) == INVALID_REGION_VALUE)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!value)
|
|
|
|
{
|
|
|
|
original.at<uint8_t>(y, x) = 0;
|
|
|
|
masked_data.at<uint8_t>(y, x) = 0;
|
|
|
|
}
|
|
|
|
original.at<uint8_t>(y, x) = static_cast<uint8_t>(255 * value);
|
|
|
|
masked_data.at<uint8_t>(y, x) = static_cast<uint8_t>(255 * value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::writeData()
|
|
|
|
{
|
|
|
|
int y = version_size - 1;
|
|
|
|
int x = version_size - 1;
|
|
|
|
int dir = -1;
|
|
|
|
int count = 0;
|
|
|
|
int codeword_value = rearranged_data[0];
|
2021-12-02 20:04:04 +08:00
|
|
|
const int limit_bits = (int)rearranged_data.size() * 8;
|
|
|
|
bool limit_reached = false;
|
2021-11-16 01:15:39 +08:00
|
|
|
while (x > 0)
|
|
|
|
{
|
|
|
|
if (x == 6)
|
|
|
|
{
|
|
|
|
x --;
|
|
|
|
}
|
|
|
|
for(int i = 0; i <= 1; i++)
|
|
|
|
{
|
|
|
|
bool bit_value = (codeword_value & (0x80 >> count % 8)) == 0;
|
|
|
|
bool success = writeBit(x - i, y, bit_value);
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
count++;
|
2021-12-02 20:04:04 +08:00
|
|
|
if (count == limit_bits)
|
|
|
|
{
|
|
|
|
limit_reached = true;
|
|
|
|
break;
|
|
|
|
}
|
2021-11-16 01:15:39 +08:00
|
|
|
if (count % 8 == 0)
|
|
|
|
{
|
|
|
|
codeword_value = rearranged_data[count / 8];
|
|
|
|
}
|
|
|
|
}
|
2021-12-02 20:04:04 +08:00
|
|
|
if (limit_reached)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2021-11-16 01:15:39 +08:00
|
|
|
y += dir;
|
|
|
|
if (y < 0 || y >= version_size)
|
|
|
|
{
|
|
|
|
dir = -dir;
|
|
|
|
x -= 2;
|
|
|
|
y += dir;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::fillReserved(const vector<uint8_t> &format_array, Mat &masked)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 7; i++)
|
|
|
|
{
|
|
|
|
if (format_array[MAX_FORMAT_LENGTH - 1 - i] == 0)
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(version_size - 1 - i, 8) = 255;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(version_size - 1 - i, 8) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
|
|
{
|
|
|
|
if (format_array[MAX_FORMAT_LENGTH - 1 - (7 + i)] == 0)
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(8, version_size - 8 + i) = 255;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(8, version_size - 8 + i) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static const int xs_format[MAX_FORMAT_LENGTH] = {
|
|
|
|
8, 8, 8, 8, 8, 8, 8, 8, 7, 5, 4, 3, 2, 1, 0
|
|
|
|
};
|
|
|
|
static const int ys_format[MAX_FORMAT_LENGTH] = {
|
|
|
|
0, 1, 2, 3, 4, 5, 7, 8, 8, 8, 8, 8, 8, 8, 8
|
|
|
|
};
|
|
|
|
for (int i = MAX_FORMAT_LENGTH - 1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
if (format_array[i] == 0)
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(ys_format[i], xs_format[i]) = 255;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(ys_format[i], xs_format[i]) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (version_level >= 7)
|
|
|
|
{
|
|
|
|
const int max_size = version_size;
|
|
|
|
const int version_block_width = 2;
|
|
|
|
const int xs_version[version_block_width][MAX_VERSION_LENGTH] = {
|
|
|
|
{ 5, 5, 5,
|
|
|
|
4, 4, 4,
|
|
|
|
3, 3, 3,
|
|
|
|
2, 2, 2,
|
|
|
|
1, 1, 1,
|
|
|
|
0, 0, 0
|
|
|
|
},
|
|
|
|
{ max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const int ys_version[version_block_width][MAX_VERSION_LENGTH] = {
|
|
|
|
{ max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11,
|
|
|
|
max_size - 9, max_size - 10, max_size - 11
|
|
|
|
},
|
|
|
|
{ 5, 5, 5,
|
|
|
|
4, 4, 4,
|
|
|
|
3, 3, 3,
|
|
|
|
2, 2, 2,
|
|
|
|
1, 1, 1,
|
|
|
|
0, 0, 0,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
for (int i = 0; i < version_block_width; i++)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < MAX_VERSION_LENGTH; j++)
|
|
|
|
{
|
|
|
|
if (version_reserved[MAX_VERSION_LENGTH - j - 1] == 0)
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(ys_version[i][j], xs_version[i][j]) = 255;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
masked.at<uint8_t>(ys_version[i][j], xs_version[i][j]) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::structureFinalMessage()
|
|
|
|
{
|
|
|
|
writeReservedArea();
|
|
|
|
writeData();
|
|
|
|
findAutoMaskType();
|
|
|
|
maskData(mask_type, masked_data);
|
|
|
|
formatGenerate(mask_type, format);
|
|
|
|
versionInfoGenerate(version_level, version_reserved);
|
|
|
|
fillReserved(format, masked_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::generatingProcess(const std::string& input, Mat& final_result)
|
|
|
|
{
|
|
|
|
vector<vector<uint8_t> > data_blocks, ecc_blocks;
|
|
|
|
if (!stringToBits(input))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
padBitStream();
|
|
|
|
eccGenerate(data_blocks, ecc_blocks);
|
|
|
|
rearrangeBlocks(data_blocks, ecc_blocks);
|
|
|
|
structureFinalMessage();
|
|
|
|
final_result = masked_data.clone();
|
|
|
|
const int border = 2;
|
|
|
|
copyMakeBorder(final_result, final_result, border, border, border, border, BORDER_CONSTANT, Scalar(255));
|
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::encode(const String& input, OutputArray output)
|
|
|
|
{
|
|
|
|
if (output.kind() != _InputArray::MAT)
|
|
|
|
CV_Error(Error::StsBadArg, "Output should be cv::Mat");
|
|
|
|
CV_Check((int)mode_type, mode_type != MODE_STRUCTURED_APPEND, "For structured append mode please call encodeStructuredAppend() method");
|
|
|
|
CV_Check(struct_num, struct_num == 1, "For structured append mode please call encodeStructuredAppend() method");
|
|
|
|
generateQR(input);
|
|
|
|
CV_Assert(!final_qrcodes.empty());
|
|
|
|
output.assign(final_qrcodes[0]);
|
2023-01-22 08:55:18 +08:00
|
|
|
final_qrcodes.clear();
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void QRCodeEncoderImpl::encodeStructuredAppend(const String& input, OutputArrayOfArrays output)
|
|
|
|
{
|
|
|
|
if (output.kind() != _InputArray::STD_VECTOR_MAT)
|
|
|
|
CV_Error(Error::StsBadArg, "Output should be vector of cv::Mat");
|
|
|
|
mode_type = MODE_STRUCTURED_APPEND;
|
|
|
|
generateQR(input);
|
|
|
|
CV_Assert(!final_qrcodes.empty());
|
|
|
|
output.create((int)final_qrcodes.size(), 1, final_qrcodes[0].type());
|
|
|
|
vector<Mat> dst;
|
|
|
|
output.getMatVector(dst);
|
|
|
|
for (int i = 0; i < (int)final_qrcodes.size(); i++)
|
|
|
|
{
|
|
|
|
output.getMatRef(i) = final_qrcodes[i];
|
|
|
|
}
|
2023-01-22 08:55:18 +08:00
|
|
|
final_qrcodes.clear();
|
2021-11-16 01:15:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Ptr<QRCodeEncoder> QRCodeEncoder::create(const QRCodeEncoder::Params& parameters)
|
|
|
|
{
|
|
|
|
return makePtr<QRCodeEncoderImpl>(parameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|