, * Lukas Bestle * @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 implements Stringable { 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); $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 = static 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 `` element * * @param int|string|null $size Optional CSS width of the `` 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 '' . '' . '' . ''; } 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 `` 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( message: '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(message: 'Invalid QR mode') // @codeCoverageIgnore }; $code = [...$code, ...array_fill(0, 4, 0)]; if ($remainder = count($code) % 8) { $code = [...$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(message: '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] ]; }