designtopack/public/kirby/src/Image/QrCode.php

1612 lines
43 KiB
PHP
Raw Normal View History

2024-07-10 16:10:33 +02:00
<?php
namespace Kirby\Image;
use Closure;
use GdImage;
use Kirby\Exception\InvalidArgumentException;
use Kirby\Exception\LogicException;
use Kirby\Filesystem\F;
/**
* Creates a QR code
* @since 4.0.0
*
* @package Kirby Image
* @author Nico Hoffmann <nico@getkirby.com>,
* Lukas Bestle <lukas@getkirby.com>
* @link https://getkirby.com
* @copyright Bastian Allgeier
* @license https://opensource.org/licenses/MIT
*
* QR Code® is a registered trademark of DENSO WAVE INCORPORATED.
*
* The code of this class is based on:
* https://github.com/psyon/php-qrcode
*
* qrcode.php - Generate QR Codes. MIT license.
*
* Copyright for portions of this project are held by Kreative Software, 2016-2018.
* All other copyright for the project are held by Donald Becker, 2019
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*/
class QrCode
{
public function __construct(public string $data)
{
}
/**
* Returns the QR code as a PNG data URI
*
* @param int|null $size Image width/height in pixels, defaults to a size per module of 4x4
* @param string $color Foreground color in hex format
* @param string $back Background color in hex format
* @param int $border Border size in number of modules
*/
public function toDataUri(
int|null $size = null,
string $color = '#000000',
string $back = '#ffffff',
int $border = 4
): string {
$image = $this->toImage($size, $color, $back, $border);
ob_start();
imagepng($image);
imagedestroy($image);
$data = ob_get_contents();
ob_end_clean();
return 'data:image/png;base64,' . base64_encode($data);
}
/**
* Returns the QR code as a GdImage object
*
* @param int|null $size Image width/height in pixels, defaults to a size per module of 4x4
* @param string $color Foreground color in hex format
* @param string $back Background color in hex format
* @param int $border Border size in number of modules
*/
public function toImage(
int|null $size = null,
string $color = '#000000',
string $back = '#ffffff',
int $border = 4
): GdImage {
// get code and size measurements
$code = $this->encode($border);
[$width, $height] = $this->measure($code);
$size ??= ceil($width * 4);
$ws = $size / $width;
$hs = $size / $height;
// create image baseplate
$image = imagecreatetruecolor($size, $size);
$allocateColor = function (string $hex) use ($image) {
$hex = preg_replace('/[^0-9A-Fa-f]/', '', $hex);
$r = hexdec(substr($hex, 0, 2));
$g = hexdec(substr($hex, 2, 2));
$b = hexdec(substr($hex, 4, 2));
return imagecolorallocate($image, $r, $g, $b);
};
$back = $allocateColor($back);
$color = $allocateColor($color);
imagefill($image, 0, 0, $back);
// paint square for each module
$this->eachModuleGroup(
$code,
fn ($x, $y, $width, $height) => imagefilledrectangle(
$image,
floor($x * $ws),
floor($y * $hs),
floor($x * $ws + $ws * $width) - 1,
floor($y * $hs + $hs * $height) - 1,
$color
)
);
return $image;
}
/**
* Returns the QR code as `<svg>` element
*
* @param int|string|null $size Optional CSS width of the `<svg>` element
* @param string $color Foreground color in hex format
* @param string $back Background color in hex format
* @param int $border Border size in number of modules
*/
public function toSvg(
int|string|null $size = null,
string $color = '#000000',
string $back = '#ffffff',
int $border = 4
): string {
$code = $this->encode($border);
[$vbw, $vbh] = $this->measure($code);
$modules = $this->eachModuleGroup(
$code,
fn ($x, $y, $width, $height) => 'M' . $x . ',' . $y . 'h' . $width . 'v' . $height . 'h-' . $width . 'z'
);
$size = $size ? ' style="width: ' . $size . '"' : '';
return '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 ' . $vbw . ' ' . $vbh . '" stroke="none"' . $size . '>' .
'<rect width="100%" height="100%" fill="' . $back . '"/>' .
'<path d="' . implode(' ', $modules) . '" fill="' . $color . '"/>' .
'</svg>';
}
public function __toString(): string
{
return $this->toSvg();
}
/**
* Saves the QR code to a file.
* Supported formats: gif, jpg, jpeg, png, svg, webp
*
* @param string $file Path to the output file with one of the supported file extensions
* @param int|string|null $size Optional image width/height in pixels (defaults to a size per module of 4x4) or CSS width of the `<svg>` element
* @param string $color Foreground color in hex format
* @param string $back Background color in hex format
* @param int $border Border size in number of modules
*/
public function write(
string $file,
int|string|null $size = null,
string $color = '#000000',
string $back = '#ffffff',
int $border = 4
): void {
$format = F::extension($file);
$args = [$size, $color, $back, $border];
match ($format) {
'gif' => imagegif($this->toImage(...$args), $file),
'jpg',
'jpeg' => imagejpeg($this->toImage(...$args), $file),
'png' => imagepng($this->toImage(...$args), $file),
'svg' => F::write($file, $this->toSvg(...$args)),
'webp' => imagewebp($this->toImage(...$args), $file),
default => throw new InvalidArgumentException('Cannot write QR code as ' . $format)
};
}
protected function applyMask(array $matrix, int $size, int $mask): array
{
for ($i = 0; $i < $size; $i++) {
for ($j = 0; $j < $size; $j++) {
if ($matrix[$i][$j] >= 4 && $this->mask($mask, $i, $j)) {
$matrix[$i][$j] ^= 1;
}
}
}
return $matrix;
}
protected function applyBestMask(array $matrix, int $size): array
{
$mask = 0;
$mmatrix = $this->applyMask($matrix, $size, $mask);
$penalty = $this->penalty($mmatrix, $size);
for ($tmask = 1; $tmask < 8; $tmask++) {
$tmatrix = $this->applyMask($matrix, $size, $tmask);
$tpenalty = $this->penalty($tmatrix, $size);
if ($tpenalty < $penalty) {
$mask = $tmask;
$mmatrix = $tmatrix;
$penalty = $tpenalty;
}
}
return [$mask, $mmatrix];
}
protected function createMatrix(int $version, array $data): array
{
$size = $version * 4 + 17;
$matrix = [];
$row = array_fill(0, $size, 0);
for ($i = 0; $i < $size; $i++) {
$matrix[] = $row;
}
// finder patterns
for ($i = 0; $i < 8; $i++) {
for ($j = 0; $j < 8; $j++) {
$m = (($i == 7 || $j == 7) ? 2 :
(($i == 0 || $j == 0 || $i == 6 || $j == 6) ? 3 :
(($i == 1 || $j == 1 || $i == 5 || $j == 5) ? 2 : 3)));
$matrix[$i][$j] = $m;
$matrix[$size - $i - 1][$j] = $m;
$matrix[$i][$size - $j - 1] = $m;
}
}
// alignment patterns
if ($version >= 2) {
$alignment = static::ALIGNMENT_PATTERNS[$version - 2];
foreach ($alignment as $i) {
foreach ($alignment as $j) {
if (!$matrix[$i][$j]) {
for ($ii = -2; $ii <= 2; $ii++) {
for ($jj = -2; $jj <= 2; $jj++) {
$m = (max(abs($ii), abs($jj)) & 1) ^ 3;
$matrix[$i + $ii][$j + $jj] = $m;
}
}
}
}
}
}
// timing patterns
for ($i = $size - 9; $i >= 8; $i--) {
$matrix[$i][6] = ($i & 1) ^ 3;
$matrix[6][$i] = ($i & 1) ^ 3;
}
// dark module such an ominous name for such an innocuous thing
$matrix[$size - 8][8] = 3;
// format information area
for ($i = 0; $i <= 8; $i++) {
if (!$matrix[$i][8]) {
$matrix[$i][8] = 1;
}
if (!$matrix[8][$i]) {
$matrix[8][$i] = 1;
}
if ($i && !$matrix[$size - $i][8]) {
$matrix[$size - $i][8] = 1;
}
if ($i && !$matrix[8][$size - $i]) {
$matrix[8][$size - $i] = 1;
}
}
// version information area
if ($version >= 7) {
for ($i = 9; $i < 12; $i++) {
for ($j = 0; $j < 6; $j++) {
$matrix[$size - $i][$j] = 1;
$matrix[$j][$size - $i] = 1;
}
}
}
// data
$col = $size - 1;
$row = $size - 1;
$dir = -1;
$offset = 0;
$length = count($data);
while ($col > 0 && $offset < $length) {
if (!$matrix[$row][$col]) {
$matrix[$row][$col] = $data[$offset] ? 5 : 4;
$offset++;
}
if (!$matrix[$row][$col - 1]) {
$matrix[$row][$col - 1] = $data[$offset] ? 5 : 4;
$offset++;
}
$row += $dir;
if ($row < 0 || $row >= $size) {
$dir = -$dir;
$row += $dir;
$col -= 2;
if ($col == 6) {
$col--;
}
}
}
return [$size, $matrix];
}
/**
* Loops over every row and column, finds all modules that can
* be grouped as rectangle (starting at the top left corner)
* and applies the given action to each active module group
*/
protected function eachModuleGroup(array $code, Closure $action): array
{
$result = [];
$xStart = $code['q'][3];
$yStart = $code['q'][0];
// generate empty matrix to track what modules have been covered
$covered = array_fill(0, count($code['bits']), array_fill(0, count($code['bits'][0]), 0));
foreach ($code['bits'] as $by => $row) {
foreach ($row as $bx => $module) {
// skip if module is inactive or already covered
if ($module === 0 || $covered[$by][$bx] === 1) {
continue;
}
$width = 0;
$height = 0;
$rowLength = count($row);
$colLength = count($code['bits']);
// extend to the right as long as the modules are active
// and use this to determine the width of the group
for ($x = $bx; $x < $rowLength; $x++) {
if ($row[$x] === 0) {
break;
}
$width++;
$covered[$by][$x] = 1;
}
// extend downwards as long as all the modules
// at the same width range are active;
// use this to determine the height of the group
for ($y = $by; $y < $colLength; $y++) {
$below = array_slice($code['bits'][$y], $bx, $width);
// if the sum is less than the width,
// there is at least one inactive module
if (array_sum($below) < $width) {
break;
}
$height++;
for ($x = $bx; $x < $bx + $width; $x++) {
$covered[$y][$x] = 1;
}
}
$result[] = $action(
$xStart + $bx,
$yStart + $by,
$width,
$height
);
}
}
return $result;
}
protected function encode(int $q = 4): array
{
[$data, $version, $ecl, $ec] = $this->encodeData();
$data = $this->encodeErrorCorrection($data, $ec, $version);
[$size, $mtx] = $this->createMatrix($version, $data);
[$mask, $mtx] = $this->applyBestMask($mtx, $size);
$mtx = $this->finalizeMatrix($mtx, $size, $ecl, $mask, $version);
return [
'q' => [$q, $q, $q, $q],
'size' => [$size, $size],
'bits' => $mtx
];
}
protected function encodeData(): array
{
$mode = $this->mode();
[$version, $ecl] = $this->version($mode);
$group = match (true) {
$version >= 27 => 2,
$version >= 10 => 1,
default => 0
};
$ec = static::EC_PARAMS[($version - 1) * 4 + $ecl];
// don't cut off mid-character if exceeding capacity
$max_chars = static::CAPACITY[$version - 1][$ecl][$mode];
if ($mode == 3) {
$max_chars <<= 1;
}
$data = substr($this->data, 0, $max_chars);
// convert from character level to bit level
$code = match ($mode) {
0 => $this->encodeNumeric($data, $group),
1 => $this->encodeAlphanum($data, $group),
2 => $this->encodeBinary($data, $group),
default => throw new LogicException('Invalid QR mode') // @codeCoverageIgnore
};
$code = array_merge($code, array_fill(0, 4, 0));
if ($remainder = count($code) % 8) {
$code = array_merge($code, array_fill(0, 8 - $remainder, 0));
}
// convert from bit level to byte level
$data = [];
for ($i = 0, $n = count($code); $i < $n; $i += 8) {
$byte = 0;
if ($code[$i + 0]) {
$byte |= 0x80;
}
if ($code[$i + 1]) {
$byte |= 0x40;
}
if ($code[$i + 2]) {
$byte |= 0x20;
}
if ($code[$i + 3]) {
$byte |= 0x10;
}
if ($code[$i + 4]) {
$byte |= 0x08;
}
if ($code[$i + 5]) {
$byte |= 0x04;
}
if ($code[$i + 6]) {
$byte |= 0x02;
}
if ($code[$i + 7]) {
$byte |= 0x01;
}
$data[] = $byte;
}
for (
$i = count($data),
$a = 1,
$n = $ec[0];
$i < $n;
$i++,
$a ^= 1
) {
$data[] = $a ? 236 : 17;
}
return [
$data,
$version,
$ecl,
$ec
];
}
protected function encodeNumeric($data, $version_group): array
{
$code = [0, 0, 0, 1];
$length = strlen($data);
switch ($version_group) {
case 2: // 27 - 40
$code[] = $length & 0x2000;
$code[] = $length & 0x1000;
// no break
case 1: // 10 - 26
$code[] = $length & 0x0800;
$code[] = $length & 0x0400;
// no break
case 0: // 1 - 9
$code[] = $length & 0x0200;
$code[] = $length & 0x0100;
$code[] = $length & 0x0080;
$code[] = $length & 0x0040;
$code[] = $length & 0x0020;
$code[] = $length & 0x0010;
$code[] = $length & 0x0008;
$code[] = $length & 0x0004;
$code[] = $length & 0x0002;
$code[] = $length & 0x0001;
}
for ($i = 0; $i < $length; $i += 3) {
$group = substr($data, $i, 3);
switch (strlen($group)) {
case 3:
$code[] = $group & 0x200;
$code[] = $group & 0x100;
$code[] = $group & 0x080;
// no break
case 2:
$code[] = $group & 0x040;
$code[] = $group & 0x020;
$code[] = $group & 0x010;
// no break
case 1:
$code[] = $group & 0x008;
$code[] = $group & 0x004;
$code[] = $group & 0x002;
$code[] = $group & 0x001;
}
}
return $code;
}
protected function encodeAlphanum($data, $version_group): array
{
$alphabet = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ $%*+-./:';
$code = [0, 0, 1, 0];
$length = strlen($data);
switch ($version_group) {
case 2: // 27 - 40
$code[] = $length & 0x1000;
$code[] = $length & 0x0800;
// no break
case 1: // 10 - 26
$code[] = $length & 0x0400;
$code[] = $length & 0x0200;
// no break
case 0: // 1 - 9
$code[] = $length & 0x0100;
$code[] = $length & 0x0080;
$code[] = $length & 0x0040;
$code[] = $length & 0x0020;
$code[] = $length & 0x0010;
$code[] = $length & 0x0008;
$code[] = $length & 0x0004;
$code[] = $length & 0x0002;
$code[] = $length & 0x0001;
}
for ($i = 0; $i < $length; $i += 2) {
$group = substr($data, $i, 2);
if (strlen($group) > 1) {
$c1 = strpos($alphabet, substr($group, 0, 1));
$c2 = strpos($alphabet, substr($group, 1, 1));
$ch = $c1 * 45 + $c2;
$code[] = $ch & 0x400;
$code[] = $ch & 0x200;
$code[] = $ch & 0x100;
$code[] = $ch & 0x080;
$code[] = $ch & 0x040;
$code[] = $ch & 0x020;
$code[] = $ch & 0x010;
$code[] = $ch & 0x008;
$code[] = $ch & 0x004;
$code[] = $ch & 0x002;
$code[] = $ch & 0x001;
} else {
$ch = strpos($alphabet, $group);
$code[] = $ch & 0x020;
$code[] = $ch & 0x010;
$code[] = $ch & 0x008;
$code[] = $ch & 0x004;
$code[] = $ch & 0x002;
$code[] = $ch & 0x001;
}
}
return $code;
}
protected function encodeBinary(string $data, int $version_group): array
{
$code = [0, 1, 0, 0];
$length = strlen($data);
switch ($version_group) {
case 2: // 27 - 40
case 1: // 10 - 26
$code[] = $length & 0x8000;
$code[] = $length & 0x4000;
$code[] = $length & 0x2000;
$code[] = $length & 0x1000;
$code[] = $length & 0x0800;
$code[] = $length & 0x0400;
$code[] = $length & 0x0200;
$code[] = $length & 0x0100;
// no break
case 0: // 1 - 9
$code[] = $length & 0x0080;
$code[] = $length & 0x0040;
$code[] = $length & 0x0020;
$code[] = $length & 0x0010;
$code[] = $length & 0x0008;
$code[] = $length & 0x0004;
$code[] = $length & 0x0002;
$code[] = $length & 0x0001;
}
for ($i = 0; $i < $length; $i++) {
$ch = ord(substr($data, $i, 1));
$code[] = $ch & 0x80;
$code[] = $ch & 0x40;
$code[] = $ch & 0x20;
$code[] = $ch & 0x10;
$code[] = $ch & 0x08;
$code[] = $ch & 0x04;
$code[] = $ch & 0x02;
$code[] = $ch & 0x01;
}
return $code;
}
protected function encodeErrorCorrection(
array $data,
array $ec_params,
int $version
): array {
$blocks = $this->errorCorrectionSplit($data, $ec_params);
$ec_blocks = [];
for ($i = 0, $n = count($blocks); $i < $n; $i++) {
$ec_blocks[] = $this->errorCorrectionDivide($blocks[$i], $ec_params);
}
$data = $this->errorCorrectionInterleave($blocks);
$ec_data = $this->errorCorrectionInterleave($ec_blocks);
$code = [];
foreach ($data as $ch) {
$code[] = $ch & 0x80;
$code[] = $ch & 0x40;
$code[] = $ch & 0x20;
$code[] = $ch & 0x10;
$code[] = $ch & 0x08;
$code[] = $ch & 0x04;
$code[] = $ch & 0x02;
$code[] = $ch & 0x01;
}
foreach ($ec_data as $ch) {
$code[] = $ch & 0x80;
$code[] = $ch & 0x40;
$code[] = $ch & 0x20;
$code[] = $ch & 0x10;
$code[] = $ch & 0x08;
$code[] = $ch & 0x04;
$code[] = $ch & 0x02;
$code[] = $ch & 0x01;
}
for ($n = static::REMAINER_BITS[$version - 1]; $n > 0; $n--) {
$code[] = 0;
}
return $code;
}
protected function errorCorrectionSplit(array $data, array $ec): array
{
$blocks = [];
$offset = 0;
for ($i = $ec[2], $length = $ec[3]; $i > 0; $i--) {
$blocks[] = array_slice($data, $offset, $length);
$offset += $length;
}
for ($i = $ec[4], $length = $ec[5]; $i > 0; $i--) {
$blocks[] = array_slice($data, $offset, $length);
$offset += $length;
}
return $blocks;
}
protected function errorCorrectionDivide(array $data, array $ec): array
{
$num_data = count($data);
$num_error = $ec[1];
$generator = static::EC_POLYNOMIALS[$num_error];
$message = $data;
for ($i = 0; $i < $num_error; $i++) {
$message[] = 0;
}
for ($i = 0; $i < $num_data; $i++) {
if ($message[$i]) {
$leadterm = static::LOG[$message[$i]];
for ($j = 0; $j <= $num_error; $j++) {
$term = ($generator[$j] + $leadterm) % 255;
$message[$i + $j] ^= static::EXP[$term];
}
}
}
return array_slice($message, $num_data, $num_error);
}
protected function errorCorrectionInterleave(array $blocks): array
{
$data = [];
$num_blocks = count($blocks);
for ($offset = 0; true; $offset++) {
$break = true;
for ($i = 0; $i < $num_blocks; $i++) {
if (isset($blocks[$i][$offset]) === true) {
$data[] = $blocks[$i][$offset];
$break = false;
}
}
if ($break) {
break;
}
}
return $data;
}
protected function finalizeMatrix(
array $matrix,
int $size,
int $ecl,
int $mask,
int $version
): array {
// Format info
$format = static::FORMAT_INFO[$ecl * 8 + $mask];
$matrix[8][0] = $format[0];
$matrix[8][1] = $format[1];
$matrix[8][2] = $format[2];
$matrix[8][3] = $format[3];
$matrix[8][4] = $format[4];
$matrix[8][5] = $format[5];
$matrix[8][7] = $format[6];
$matrix[8][8] = $format[7];
$matrix[7][8] = $format[8];
$matrix[5][8] = $format[9];
$matrix[4][8] = $format[10];
$matrix[3][8] = $format[11];
$matrix[2][8] = $format[12];
$matrix[1][8] = $format[13];
$matrix[0][8] = $format[14];
$matrix[$size - 1][8] = $format[0];
$matrix[$size - 2][8] = $format[1];
$matrix[$size - 3][8] = $format[2];
$matrix[$size - 4][8] = $format[3];
$matrix[$size - 5][8] = $format[4];
$matrix[$size - 6][8] = $format[5];
$matrix[$size - 7][8] = $format[6];
$matrix[8][$size - 8] = $format[7];
$matrix[8][$size - 7] = $format[8];
$matrix[8][$size - 6] = $format[9];
$matrix[8][$size - 5] = $format[10];
$matrix[8][$size - 4] = $format[11];
$matrix[8][$size - 3] = $format[12];
$matrix[8][$size - 2] = $format[13];
$matrix[8][$size - 1] = $format[14];
// version info
if ($version >= 7) {
$version = static::VERSION_INFO[$version - 7];
for ($i = 0; $i < 18; $i++) {
$r = $size - 9 - ($i % 3);
$c = 5 - floor($i / 3);
$matrix[$r][$c] = $version[$i];
$matrix[$c][$r] = $version[$i];
}
}
// patterns and data
for ($i = 0; $i < $size; $i++) {
for ($j = 0; $j < $size; $j++) {
$matrix[$i][$j] &= 1;
}
}
return $matrix;
}
protected function mask(int $mask, int $row, int $column): int
{
return match ($mask) {
0 => !(($row + $column) % 2),
1 => !($row % 2),
2 => !($column % 3),
3 => !(($row + $column) % 3),
4 => !((floor($row / 2) + floor($column / 3)) % 2),
5 => !(((($row * $column) % 2) + (($row * $column) % 3))),
6 => !(((($row * $column) % 2) + (($row * $column) % 3)) % 2),
7 => !(((($row + $column) % 2) + (($row * $column) % 3)) % 2),
default => throw new LogicException('Invalid QR mask') // @codeCoverageIgnore
};
}
/**
* Returns width and height based on the
* generated modules and quiet zone
*/
protected function measure($code): array
{
return [
$code['q'][3] + $code['size'][0] + $code['q'][1],
$code['q'][0] + $code['size'][1] + $code['q'][2]
];
}
/**
* Detect what encoding mode (numeric, alphanumeric, binary)
* can be used
*/
protected function mode(): int
{
// numeric
if (preg_match('/^[0-9]*$/', $this->data)) {
return 0;
}
// alphanumeric
if (preg_match('/^[0-9A-Z .\/:$%*+-]*$/', $this->data)) {
return 1;
}
return 2;
}
protected function penalty(array &$matrix, int $size): int
{
$score = $this->penalty1($matrix, $size);
$score += $this->penalty2($matrix, $size);
$score += $this->penalty3($matrix, $size);
$score += $this->penalty4($matrix, $size);
return $score;
}
protected function penalty1(array &$matrix, int $size): int
{
$score = 0;
for ($i = 0; $i < $size; $i++) {
$rowvalue = 0;
$rowcount = 0;
$colvalue = 0;
$colcount = 0;
for ($j = 0; $j < $size; $j++) {
$rv = ($matrix[$i][$j] == 5 || $matrix[$i][$j] == 3) ? 1 : 0;
$cv = ($matrix[$j][$i] == 5 || $matrix[$j][$i] == 3) ? 1 : 0;
if ($rv == $rowvalue) {
$rowcount++;
} else {
if ($rowcount >= 5) {
$score += $rowcount - 2;
}
$rowvalue = $rv;
$rowcount = 1;
}
if ($cv == $colvalue) {
$colcount++;
} else {
if ($colcount >= 5) {
$score += $colcount - 2;
}
$colvalue = $cv;
$colcount = 1;
}
}
if ($rowcount >= 5) {
$score += $rowcount - 2;
}
if ($colcount >= 5) {
$score += $colcount - 2;
}
}
return $score;
}
protected function penalty2(array &$matrix, int $size): int
{
$score = 0;
for ($i = 1; $i < $size; $i++) {
for ($j = 1; $j < $size; $j++) {
$v1 = $matrix[$i - 1][$j - 1];
$v2 = $matrix[$i - 1][$j ];
$v3 = $matrix[$i ][$j - 1];
$v4 = $matrix[$i ][$j ];
$v1 = ($v1 == 5 || $v1 == 3) ? 1 : 0;
$v2 = ($v2 == 5 || $v2 == 3) ? 1 : 0;
$v3 = ($v3 == 5 || $v3 == 3) ? 1 : 0;
$v4 = ($v4 == 5 || $v4 == 3) ? 1 : 0;
if ($v1 == $v2 && $v2 == $v3 && $v3 == $v4) {
$score += 3;
}
}
}
return $score;
}
protected function penalty3(array &$matrix, int $size): int
{
$score = 0;
for ($i = 0; $i < $size; $i++) {
$rowvalue = 0;
$colvalue = 0;
for ($j = 0; $j < 11; $j++) {
$rv = ($matrix[$i][$j] == 5 || $matrix[$i][$j] == 3) ? 1 : 0;
$cv = ($matrix[$j][$i] == 5 || $matrix[$j][$i] == 3) ? 1 : 0;
$rowvalue = (($rowvalue << 1) & 0x7FF) | $rv;
$colvalue = (($colvalue << 1) & 0x7FF) | $cv;
}
if ($rowvalue == 0x5D0 || $rowvalue == 0x5D) {
$score += 40;
}
if ($colvalue == 0x5D0 || $colvalue == 0x5D) {
$score += 40;
}
for ($j = 11; $j < $size; $j++) {
$rv = ($matrix[$i][$j] == 5 || $matrix[$i][$j] == 3) ? 1 : 0;
$cv = ($matrix[$j][$i] == 5 || $matrix[$j][$i] == 3) ? 1 : 0;
$rowvalue = (($rowvalue << 1) & 0x7FF) | $rv;
$colvalue = (($colvalue << 1) & 0x7FF) | $cv;
if ($rowvalue == 0x5D0 || $rowvalue == 0x5D) {
$score += 40;
}
if ($colvalue == 0x5D0 || $colvalue == 0x5D) {
$score += 40;
}
}
}
return $score;
}
protected function penalty4(array &$matrix, int $size): int
{
$dark = 0;
for ($i = 0; $i < $size; $i++) {
for ($j = 0; $j < $size; $j++) {
if ($matrix[$i][$j] == 5 || $matrix[$i][$j] == 3) {
$dark++;
}
}
}
$dark *= 20;
$dark /= $size * $size;
$a = abs(floor($dark) - 10);
$b = abs(ceil($dark) - 10);
return min($a, $b) * 10;
}
/**
* Detect what version needs to be used by
* trying to maximize the error correction level
*/
protected function version(int $mode): array
{
$length = strlen($this->data);
if ($mode == 3) {
$length >>= 1;
}
$ecl = 0;
// first try to find the minimum version
// that can contain the data
for ($version = 1; $version <= 40; $version++) {
if ($length <= static::CAPACITY[$version - 1][$ecl][$mode]) {
break;
}
}
// with the version in place, try to raise
// the error correction level as long as
// the data still fits
for ($newEcl = 1; $newEcl <= 3; $newEcl++) {
if ($length <= static::CAPACITY[$version - 1][$newEcl][$mode]) {
$ecl = $newEcl;
}
}
return [$version, $ecl];
}
/**
* maximum encodable characters = $qr_capacity [ (version - 1) ]
* [ (0 for L, 1 for M, 2 for Q, 3 for H) ]
* [ (0 for numeric, 1 for alpha, 2 for binary) ]
*/
protected const CAPACITY = [
[
[ 41, 25, 17],
[ 34, 20, 14],
[ 27, 16, 11],
[ 17, 10, 7]
],
[
[ 77, 47, 32],
[ 63, 38, 26],
[ 48, 29, 20],
[ 34, 20, 14]
],
[
[ 127, 77, 53],
[ 101, 61, 42],
[ 77, 47, 32],
[ 58, 35, 24]
],
[
[ 187, 114, 78],
[ 149, 90, 62],
[ 111, 67, 46],
[ 82, 50, 34]
],
[
[ 255, 154, 106],
[ 202, 122, 84],
[ 144, 87, 60],
[ 106, 64, 44]
],
[
[ 322, 195, 134],
[ 255, 154, 106],
[ 178, 108, 74],
[ 139, 84, 58]
],
[
[ 370, 224, 154],
[ 293, 178, 122],
[ 207, 125, 86],
[ 154, 93, 64]
],
[
[ 461, 279, 192],
[ 365, 221, 152],
[ 259, 157, 108],
[ 202, 122, 84]
],
[
[ 552, 335, 230],
[ 432, 262, 180],
[ 312, 189, 130],
[ 235, 143, 98]],
[
[ 652, 395, 271],
[ 513, 311, 213],
[ 364, 221, 151],
[ 288, 174, 119]
],
[
[ 772, 468, 321],
[ 604, 366, 251],
[ 427, 259, 177],
[ 331, 200, 137]
],
[
[ 883, 535, 367],
[ 691, 419, 287],
[ 489, 296, 203],
[ 374, 227, 155]
],
[
[1022, 619, 425],
[ 796, 483, 331],
[ 580, 352, 241],
[ 427, 259, 177]
],
[
[1101, 667, 458],
[ 871, 528, 362],
[ 621, 376, 258],
[ 468, 283, 194]
],
[
[1250, 758, 520],
[ 991, 600, 412],
[ 703, 426, 292],
[ 530, 321, 220]
],
[
[1408, 854, 586],
[1082, 656, 450],
[ 775, 470, 322],
[ 602, 365, 250]
],
[
[1548, 938, 644],
[1212, 734, 504],
[ 876, 531, 364],
[ 674, 408, 280]
],
[
[1725, 1046, 718],
[1346, 816, 560],
[ 948, 574, 394],
[ 746, 452, 310]
],
[
[1903, 1153, 792],
[1500, 909, 624],
[1063, 644, 442],
[ 813, 493, 338]
],
[
[2061, 1249, 858],
[1600, 970, 666],
[1159, 702, 482],
[ 919, 557, 382]
],
[
[2232, 1352, 929],
[1708, 1035, 711],
[1224, 742, 509],
[ 969, 587, 403]
],
[
[2409, 1460, 1003],
[1872, 1134, 779],
[1358, 823, 565],
[1056, 640, 439]
],
[
[2620, 1588, 1091],
[2059, 1248, 857],
[1468, 890, 611],
[1108, 672, 461]
],
[
[2812, 1704, 1171],
[2188, 1326, 911],
[1588, 963, 661],
[1228, 744, 511]
],
[
[3057, 1853, 1273],
[2395, 1451, 997],
[1718, 1041, 715],
[1286, 779, 535]
],
[
[3283, 1990, 1367],
[2544, 1542, 1059],
[1804, 1094, 751],
[1425, 864, 593]
],
[
[3517, 2132, 1465],
[2701, 1637, 1125],
[1933, 1172, 805],
[1501, 910, 625]
],
[
[3669, 2223, 1528],
[2857, 1732, 1190],
[2085, 1263, 868],
[1581, 958, 658]
],
[
[3909, 2369, 1628],
[3035, 1839, 1264],
[2181, 1322, 908],
[1677, 1016, 698]
],
[
[4158, 2520, 1732],
[3289, 1994, 1370],
[2358, 1429, 982],
[1782, 1080, 742]
],
[
[4417, 2677, 1840],
[3486, 2113, 1452],
[2473, 1499, 1030],
[1897, 1150, 790]
],
[
[4686, 2840, 1952],
[3693, 2238, 1538],
[2670, 1618, 1112],
[2022, 1226, 842]
],
[
[4965, 3009, 2068],
[3909, 2369, 1628],
[2805, 1700, 1168],
[2157, 1307, 898]
],
[
[5253, 3183, 2188],
[4134, 2506, 1722],
[2949, 1787, 1228],
[2301, 1394, 958]
],
[
[5529, 3351, 2303],
[4343, 2632, 1809],
[3081, 1867, 1283],
[2361, 1431, 983]
],
[
[5836, 3537, 2431],
[4588, 2780, 1911],
[3244, 1966, 1351],
[2524, 1530, 1051]
],
[
[6153, 3729, 2563],
[4775, 2894, 1989],
[3417, 2071, 1423],
[2625, 1591, 1093]
],
[
[6479, 3927, 2699],
[5039, 3054, 2099],
[3599, 2181, 1499],
[2735, 1658, 1139]
],
[
[6743, 4087, 2809],
[5313, 3220, 2213],
[3791, 2298, 1579],
[2927, 1774, 1219]
],
[
[7089, 4296, 2953],
[5596, 3391, 2331],
[3993, 2420, 1663],
[3057, 1852, 1273]
],
];
/**
* $qr_ec_params[
* 4 * (version - 1) + (0 for L, 1 for M, 2 for Q, 3 for H)
* ] = [
* total number of data codewords,
* number of error correction codewords per block,
* number of blocks in first group,
* number of data codewords per block in first group,
* number of blocks in second group,
* number of data codewords per block in second group
* );
*/
protected const EC_PARAMS = [
[ 19, 7, 1, 19, 0, 0],
[ 16, 10, 1, 16, 0, 0],
[ 13, 13, 1, 13, 0, 0],
[ 9, 17, 1, 9, 0, 0],
[ 34, 10, 1, 34, 0, 0],
[ 28, 16, 1, 28, 0, 0],
[ 22, 22, 1, 22, 0, 0],
[ 16, 28, 1, 16, 0, 0],
[ 55, 15, 1, 55, 0, 0],
[ 44, 26, 1, 44, 0, 0],
[ 34, 18, 2, 17, 0, 0],
[ 26, 22, 2, 13, 0, 0],
[ 80, 20, 1, 80, 0, 0],
[ 64, 18, 2, 32, 0, 0],
[ 48, 26, 2, 24, 0, 0],
[ 36, 16, 4, 9, 0, 0],
[ 108, 26, 1, 108, 0, 0],
[ 86, 24, 2, 43, 0, 0],
[ 62, 18, 2, 15, 2, 16],
[ 46, 22, 2, 11, 2, 12],
[ 136, 18, 2, 68, 0, 0],
[ 108, 16, 4, 27, 0, 0],
[ 76, 24, 4, 19, 0, 0],
[ 60, 28, 4, 15, 0, 0],
[ 156, 20, 2, 78, 0, 0],
[ 124, 18, 4, 31, 0, 0],
[ 88, 18, 2, 14, 4, 15],
[ 66, 26, 4, 13, 1, 14],
[ 194, 24, 2, 97, 0, 0],
[ 154, 22, 2, 38, 2, 39],
[ 110, 22, 4, 18, 2, 19],
[ 86, 26, 4, 14, 2, 15],
[ 232, 30, 2, 116, 0, 0],
[ 182, 22, 3, 36, 2, 37],
[ 132, 20, 4, 16, 4, 17],
[ 100, 24, 4, 12, 4, 13],
[ 274, 18, 2, 68, 2, 69],
[ 216, 26, 4, 43, 1, 44],
[ 154, 24, 6, 19, 2, 20],
[ 122, 28, 6, 15, 2, 16],
[ 324, 20, 4, 81, 0, 0],
[ 254, 30, 1, 50, 4, 51],
[ 180, 28, 4, 22, 4, 23],
[ 140, 24, 3, 12, 8, 13],
[ 370, 24, 2, 92, 2, 93],
[ 290, 22, 6, 36, 2, 37],
[ 206, 26, 4, 20, 6, 21],
[ 158, 28, 7, 14, 4, 15],
[ 428, 26, 4, 107, 0, 0],
[ 334, 22, 8, 37, 1, 38],
[ 244, 24, 8, 20, 4, 21],
[ 180, 22, 12, 11, 4, 12],
[ 461, 30, 3, 115, 1, 116],
[ 365, 24, 4, 40, 5, 41],
[ 261, 20, 11, 16, 5, 17],
[ 197, 24, 11, 12, 5, 13],
[ 523, 22, 5, 87, 1, 88],
[ 415, 24, 5, 41, 5, 42],
[ 295, 30, 5, 24, 7, 25],
[ 223, 24, 11, 12, 7, 13],
[ 589, 24, 5, 98, 1, 99],
[ 453, 28, 7, 45, 3, 46],
[ 325, 24, 15, 19, 2, 20],
[ 253, 30, 3, 15, 13, 16],
[ 647, 28, 1, 107, 5, 108],
[ 507, 28, 10, 46, 1, 47],
[ 367, 28, 1, 22, 15, 23],
[ 283, 28, 2, 14, 17, 15],
[ 721, 30, 5, 120, 1, 121],
[ 563, 26, 9, 43, 4, 44],
[ 397, 28, 17, 22, 1, 23],
[ 313, 28, 2, 14, 19, 15],
[ 795, 28, 3, 113, 4, 114],
[ 627, 26, 3, 44, 11, 45],
[ 445, 26, 17, 21, 4, 22],
[ 341, 26, 9, 13, 16, 14],
[ 861, 28, 3, 107, 5, 108],
[ 669, 26, 3, 41, 13, 42],
[ 485, 30, 15, 24, 5, 25],
[ 385, 28, 15, 15, 10, 16],
[ 932, 28, 4, 116, 4, 117],
[ 714, 26, 17, 42, 0, 0],
[ 512, 28, 17, 22, 6, 23],
[ 406, 30, 19, 16, 6, 17],
[1006, 28, 2, 111, 7, 112],
[ 782, 28, 17, 46, 0, 0],
[ 568, 30, 7, 24, 16, 25],
[ 442, 24, 34, 13, 0, 0],
[1094, 30, 4, 121, 5, 122],
[ 860, 28, 4, 47, 14, 48],
[ 614, 30, 11, 24, 14, 25],
[ 464, 30, 16, 15, 14, 16],
[1174, 30, 6, 117, 4, 118],
[ 914, 28, 6, 45, 14, 46],
[ 664, 30, 11, 24, 16, 25],
[ 514, 30, 30, 16, 2, 17],
[1276, 26, 8, 106, 4, 107],
[1000, 28, 8, 47, 13, 48],
[ 718, 30, 7, 24, 22, 25],
[ 538, 30, 22, 15, 13, 16],
[1370, 28, 10, 114, 2, 115],
[1062, 28, 19, 46, 4, 47],
[ 754, 28, 28, 22, 6, 23],
[ 596, 30, 33, 16, 4, 17],
[1468, 30, 8, 122, 4, 123],
[1128, 28, 22, 45, 3, 46],
[ 808, 30, 8, 23, 26, 24],
[ 628, 30, 12, 15, 28, 16],
[1531, 30, 3, 117, 10, 118],
[1193, 28, 3, 45, 23, 46],
[ 871, 30, 4, 24, 31, 25],
[ 661, 30, 11, 15, 31, 16],
[1631, 30, 7, 116, 7, 117],
[1267, 28, 21, 45, 7, 46],
[ 911, 30, 1, 23, 37, 24],
[ 701, 30, 19, 15, 26, 16],
[1735, 30, 5, 115, 10, 116],
[1373, 28, 19, 47, 10, 48],
[ 985, 30, 15, 24, 25, 25],
[ 745, 30, 23, 15, 25, 16],
[1843, 30, 13, 115, 3, 116],
[1455, 28, 2, 46, 29, 47],
[1033, 30, 42, 24, 1, 25],
[ 793, 30, 23, 15, 28, 16],
[1955, 30, 17, 115, 0, 0],
[1541, 28, 10, 46, 23, 47],
[1115, 30, 10, 24, 35, 25],
[ 845, 30, 19, 15, 35, 16],
[2071, 30, 17, 115, 1, 116],
[1631, 28, 14, 46, 21, 47],
[1171, 30, 29, 24, 19, 25],
[ 901, 30, 11, 15, 46, 16],
[2191, 30, 13, 115, 6, 116],
[1725, 28, 14, 46, 23, 47],
[1231, 30, 44, 24, 7, 25],
[ 961, 30, 59, 16, 1, 17],
[2306, 30, 12, 121, 7, 122],
[1812, 28, 12, 47, 26, 48],
[1286, 30, 39, 24, 14, 25],
[ 986, 30, 22, 15, 41, 16],
[2434, 30, 6, 121, 14, 122],
[1914, 28, 6, 47, 34, 48],
[1354, 30, 46, 24, 10, 25],
[1054, 30, 2, 15, 64, 16],
[2566, 30, 17, 122, 4, 123],
[1992, 28, 29, 46, 14, 47],
[1426, 30, 49, 24, 10, 25],
[1096, 30, 24, 15, 46, 16],
[2702, 30, 4, 122, 18, 123],
[2102, 28, 13, 46, 32, 47],
[1502, 30, 48, 24, 14, 25],
[1142, 30, 42, 15, 32, 16],
[2812, 30, 20, 117, 4, 118],
[2216, 28, 40, 47, 7, 48],
[1582, 30, 43, 24, 22, 25],
[1222, 30, 10, 15, 67, 16],
[2956, 30, 19, 118, 6, 119],
[2334, 28, 18, 47, 31, 48],
[1666, 30, 34, 24, 34, 25],
[1276, 30, 20, 15, 61, 16],
];
protected const EC_POLYNOMIALS = [
7 => [0, 87, 229, 146, 149, 238, 102, 21],
10 => [0, 251, 67, 46, 61, 118, 70, 64, 94, 32, 45],
13 => [0, 74, 152, 176, 100, 86, 100, 106, 104, 130, 218, 206, 140, 78],
15 => [0, 8, 183, 61, 91, 202, 37, 51, 58, 58, 237, 140, 124, 5, 99, 105],
16 => [0, 120, 104, 107, 109, 102, 161, 76, 3, 91, 191, 147, 169, 182, 194, 225, 120],
17 => [0, 43, 139, 206, 78, 43, 239, 123, 206, 214, 147, 24, 99, 150, 39, 243, 163, 136],
18 => [0, 215, 234, 158, 94, 184, 97, 118, 170, 79, 187, 152, 148, 252, 179, 5, 98, 96, 153],
20 => [0, 17, 60, 79, 50, 61, 163, 26, 187, 202, 180, 221, 225, 83, 239, 156, 164, 212, 212, 188, 190],
22 => [0, 210, 171, 247, 242, 93, 230, 14, 109, 221, 53, 200, 74, 8, 172, 98, 80, 219, 134, 160, 105, 165, 231],
24 => [0, 229, 121, 135, 48, 211, 117, 251, 126, 159, 180, 169, 152, 192, 226, 228, 218, 111, 0, 117, 232, 87, 96, 227, 21],
26 => [0, 173, 125, 158, 2, 103, 182, 118, 17, 145, 201, 111, 28, 165, 53, 161, 21, 245, 142, 13, 102, 48, 227, 153, 145, 218, 70],
28 => [0, 168, 223, 200, 104, 224, 234, 108, 180, 110, 190, 195, 147, 205, 27, 232, 201, 21, 43, 245, 87, 42, 195, 212, 119, 242, 37, 9, 123],
30 => [0, 41, 173, 145, 152, 216, 31, 179, 182, 50, 48, 110, 86, 239, 96, 222, 125, 42, 173, 226, 193, 224, 130, 156, 37, 251, 216, 238, 40, 192, 180],
];
protected const LOG = [0, 0, 1, 25, 2, 50, 26, 198, 3, 223, 51, 238, 27, 104, 199, 75, 4, 100, 224, 14, 52, 141, 239, 129, 28, 193, 105, 248, 200, 8, 76, 113, 5, 138, 101, 47, 225, 36, 15, 33, 53, 147, 142, 218, 240, 18, 130, 69, 29, 181, 194, 125, 106, 39, 249, 185, 201, 154, 9, 120, 77, 228, 114, 166, 6, 191, 139, 98, 102, 221, 48, 253, 226, 152, 37, 179, 16, 145, 34, 136, 54, 208, 148, 206, 143, 150, 219, 189, 241, 210, 19, 92, 131, 56, 70, 64, 30, 66, 182, 163, 195, 72, 126, 110, 107, 58, 40, 84, 250, 133, 186, 61, 202, 94, 155, 159, 10, 21, 121, 43, 78, 212, 229, 172, 115, 243, 167, 87, 7, 112, 192, 247, 140, 128, 99, 13, 103, 74, 222, 237, 49, 197, 254, 24, 227, 165, 153, 119, 38, 184, 180, 124, 17, 68, 146, 217, 35, 32, 137, 46, 55, 63, 209, 91, 149, 188, 207, 205, 144, 135, 151, 178, 220, 252, 190, 97, 242, 86, 211, 171, 20, 42, 93, 158, 132, 60, 57, 83, 71, 109, 65, 162, 31, 45, 67, 216, 183, 123, 164, 118, 196, 23, 73, 236, 127, 12, 111, 246, 108, 161, 59, 82, 41, 157, 85, 170, 251, 96, 134, 177, 187, 204, 62, 90, 203, 89, 95, 176, 156, 169, 160, 81, 11, 245, 22, 235, 122, 117, 44, 215, 79, 174, 213, 233, 230, 231, 173, 232, 116, 214, 244, 234, 168, 80, 88, 175];
protected const EXP = [1, 2, 4, 8, 16, 32, 64, 128, 29, 58, 116, 232, 205, 135, 19, 38, 76, 152, 45, 90, 180, 117, 234, 201, 143, 3, 6, 12, 24, 48, 96, 192, 157, 39, 78, 156, 37, 74, 148, 53, 106, 212, 181, 119, 238, 193, 159, 35, 70, 140, 5, 10, 20, 40, 80, 160, 93, 186, 105, 210, 185, 111, 222, 161, 95, 190, 97, 194, 153, 47, 94, 188, 101, 202, 137, 15, 30, 60, 120, 240, 253, 231, 211, 187, 107, 214, 177, 127, 254, 225, 223, 163, 91, 182, 113, 226, 217, 175, 67, 134, 17, 34, 68, 136, 13, 26, 52, 104, 208, 189, 103, 206, 129, 31, 62, 124, 248, 237, 199, 147, 59, 118, 236, 197, 151, 51, 102, 204, 133, 23, 46, 92, 184, 109, 218, 169, 79, 158, 33, 66, 132, 21, 42, 84, 168, 77, 154, 41, 82, 164, 85, 170, 73, 146, 57, 114, 228, 213, 183, 115, 230, 209, 191, 99, 198, 145, 63, 126, 252, 229, 215, 179, 123, 246, 241, 255, 227, 219, 171, 75, 150, 49, 98, 196, 149, 55, 110, 220, 165, 87, 174, 65, 130, 25, 50, 100, 200, 141, 7, 14, 28, 56, 112, 224, 221, 167, 83, 166, 81, 162, 89, 178, 121, 242, 249, 239, 195, 155, 43, 86, 172, 69, 138, 9, 18, 36, 72, 144, 61, 122, 244, 245, 247, 243, 251, 235, 203, 139, 11, 22, 44, 88, 176, 125, 250, 233, 207, 131, 27, 54, 108, 216, 173, 71, 142, 1];
protected const REMAINER_BITS = [0, 7, 7, 7, 7, 7, 0, 0, 0, 0, 0, 0, 0, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 3, 3, 3, 3, 3, 3, 3, 0, 0, 0, 0, 0, 0];
protected const ALIGNMENT_PATTERNS = [
[6, 18],
[6, 22],
[6, 26],
[6, 30],
[6, 34],
[6, 22, 38],
[6, 24, 42],
[6, 26, 46],
[6, 28, 50],
[6, 30, 54],
[6, 32, 58],
[6, 34, 62],
[6, 26, 46, 66],
[6, 26, 48, 70],
[6, 26, 50, 74],
[6, 30, 54, 78],
[6, 30, 56, 82],
[6, 30, 58, 86],
[6, 34, 62, 90],
[6, 28, 50, 72, 94],
[6, 26, 50, 74, 98],
[6, 30, 54, 78, 102],
[6, 28, 54, 80, 106],
[6, 32, 58, 84, 110],
[6, 30, 58, 86, 114],
[6, 34, 62, 90, 118],
[6, 26, 50, 74, 98, 122],
[6, 30, 54, 78, 102, 126],
[6, 26, 52, 78, 104, 130],
[6, 30, 56, 82, 108, 134],
[6, 34, 60, 86, 112, 138],
[6, 30, 58, 86, 114, 142],
[6, 34, 62, 90, 118, 146],
[6, 30, 54, 78, 102, 126, 150],
[6, 24, 50, 76, 102, 128, 154],
[6, 28, 54, 80, 106, 132, 158],
[6, 32, 58, 84, 110, 136, 162],
[6, 26, 54, 82, 110, 138, 166],
[6, 30, 58, 86, 114, 142, 170],
];
/**
* format info string = $qr_format_info[
* (0 for L, 8 for M, 16 for Q, 24 for H) + mask
*];
*/
protected const FORMAT_INFO = [
[1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 0, 0, 1, 0, 0],
[1, 1, 1, 0, 0, 1, 0, 1, 1, 1, 1, 0, 0, 1, 1],
[1, 1, 1, 1, 1, 0, 1, 1, 0, 1, 0, 1, 0, 1, 0],
[1, 1, 1, 1, 0, 0, 0, 1, 0, 0, 1, 1, 1, 0, 1],
[1, 1, 0, 0, 1, 1, 0, 0, 0, 1, 0, 1, 1, 1, 1],
[1, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 1, 0, 0, 0],
[1, 1, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1],
[1, 1, 0, 1, 0, 0, 1, 0, 1, 1, 1, 0, 1, 1, 0],
[1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0],
[1, 0, 1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 1],
[1, 0, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1, 1, 0, 0],
[1, 0, 1, 1, 0, 1, 1, 0, 1, 0, 0, 1, 0, 1, 1],
[1, 0, 0, 0, 1, 0, 1, 1, 1, 1, 1, 1, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0],
[1, 0, 0, 1, 1, 1, 1, 1, 0, 0, 1, 0, 1, 1, 1],
[1, 0, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0],
[0, 1, 1, 0, 1, 0, 1, 0, 1, 0, 1, 1, 1, 1, 1],
[0, 1, 1, 0, 0, 0, 0, 0, 1, 1, 0, 1, 0, 0, 0],
[0, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 0, 0, 1],
[0, 1, 1, 1, 0, 1, 0, 0, 0, 0, 0, 0, 1, 1, 0],
[0, 1, 0, 0, 1, 0, 0, 1, 0, 1, 1, 0, 1, 0, 0],
[0, 1, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 1, 1],
[0, 1, 0, 1, 1, 1, 0, 1, 1, 0, 1, 1, 0, 1, 0],
[0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 0, 1, 1, 0, 1],
[0, 0, 1, 0, 1, 1, 0, 1, 0, 0, 0, 1, 0, 0, 1],
[0, 0, 1, 0, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0],
[0, 0, 1, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 1],
[0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 1, 0, 0, 0, 0],
[0, 0, 0, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 1, 0],
[0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 1],
[0, 0, 0, 1, 1, 0, 1, 0, 0, 0, 0, 1, 1, 0, 0],
[0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 1, 1, 0, 1, 1]
];
/**
* version info string = $qr_version_info[ (version - 7) ]
*/
protected const VERSION_INFO = [
[0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 1, 0, 1, 0, 0],
[0, 0, 1, 0, 0, 0, 0, 1, 0, 1, 1, 0, 1, 1, 1, 1, 0, 0],
[0, 0, 1, 0, 0, 1, 1, 0, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1],
[0, 0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 1, 0, 1, 0, 0, 1, 1],
[0, 0, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 0],
[0, 0, 1, 1, 0, 0, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 1, 0],
[0, 0, 1, 1, 0, 1, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1, 1, 1],
[0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 1, 1, 0, 1],
[0, 0, 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 1, 0, 1, 0, 0, 0],
[0, 1, 0, 0, 0, 0, 1, 0, 1, 1, 0, 1, 1, 1, 1, 0, 0, 0],
[0, 1, 0, 0, 0, 1, 0, 1, 0, 0, 0, 1, 0, 1, 1, 1, 0, 1],
[0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 1, 0, 1, 1, 1],
[0, 1, 0, 0, 1, 1, 0, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0],
[0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 1, 0, 1, 0, 0, 1, 1, 0],
[0, 1, 0, 1, 0, 1, 0, 1, 1, 0, 1, 0, 0, 0, 0, 0, 1, 1],
[0, 1, 0, 1, 1, 0, 1, 0, 0, 0, 1, 1, 0, 0, 1, 0, 0, 1],
[0, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 0, 0],
[0, 1, 1, 0, 0, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 1, 0, 0],
[0, 1, 1, 0, 0, 1, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 1],
[0, 1, 1, 0, 1, 0, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0, 1, 1],
[0, 1, 1, 0, 1, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1, 1, 1, 0],
[0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 1, 1, 0, 1, 0],
[0, 1, 1, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 1, 1, 1],
[0, 1, 1, 1, 1, 0, 1, 1, 0, 1, 0, 1, 1, 1, 0, 1, 0, 1],
[0, 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 1, 0, 1, 0, 0, 0, 0],
[1, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1, 1, 0, 1, 0, 1, 0, 1],
[1, 0, 0, 0, 0, 1, 0, 1, 1, 0, 1, 1, 1, 1, 0, 0, 0, 0],
[1, 0, 0, 0, 1, 0, 1, 0, 0, 0, 1, 0, 1, 1, 1, 0, 1, 0],
[1, 0, 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1, 1],
[1, 0, 0, 1, 0, 0, 1, 0, 1, 1, 0, 0, 0, 0, 1, 0, 1, 1],
[1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 1, 0, 1, 1, 1, 0],
[1, 0, 0, 1, 1, 0, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0],
[1, 0, 0, 1, 1, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 1],
[1, 0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 1, 0, 1, 0, 0, 1]
];
}