1407 lines
41 KiB
PHP
1407 lines
41 KiB
PHP
<?php
|
|
|
|
/**
|
|
* QRCode - QR 2D barcode generator
|
|
*
|
|
* QRCode - QR 2D barcode generator
|
|
*
|
|
* PHP version 5
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this library in the file LICENSE.LGPL; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
* 02111-1307 USA
|
|
*
|
|
* From the original code by Y.Swetake:
|
|
* THIS SOFTWARE IS PROVIDED BY Y.Swetake ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
* IN NO EVENT SHALL Y.Swetake OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
|
|
* USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* @category Images
|
|
* @package Image_QRCode
|
|
* @author Y.Swetake <swe@venus.dti.ne.jp>
|
|
* @author Rich Sage <rich.sage@gmail.com>
|
|
* @copyright 2009 Y.Swetake
|
|
* @copyright 2009 R.Sage
|
|
* @license http://www.gnu.org/copyleft/lesser.html LGPL License 2.1
|
|
* @version SVN: $Id$
|
|
* @link http://code.google.com/p/pearqrcode/
|
|
*/
|
|
|
|
/**
|
|
* Exception handling
|
|
*/
|
|
|
|
require_once 'QRCode/Exception.php';
|
|
|
|
/**
|
|
* QRCode
|
|
*
|
|
* @category Images
|
|
* @package Image_QRCode
|
|
* @author Y.Swetake <swe@venus.dti.ne.jp>
|
|
* @author Rich Sage <rich.sage@gmail.com>
|
|
* @license http://www.gnu.org/licenses/lgpl-2.1.txt GNU LGPL
|
|
* @version Release: 0.1
|
|
* @link http://code.google.com/p/pearqrcode/
|
|
*/
|
|
|
|
class Image_QRCode
|
|
{
|
|
|
|
/**
|
|
* path to the data directory
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $path;
|
|
|
|
/**
|
|
* path to the image directory
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $image_path;
|
|
|
|
/**
|
|
* upper limit for version
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $version_ul;
|
|
|
|
/**
|
|
* string/data to encode
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $data_string;
|
|
|
|
/**
|
|
* output type
|
|
* - return: return raw PHP GD image object
|
|
* - display: output to browser
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $output_type;
|
|
|
|
/**
|
|
* image_type
|
|
* - png/jpeg
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $image_type;
|
|
|
|
/**
|
|
* base_image
|
|
*
|
|
* @var gd_image
|
|
*/
|
|
protected $base_image;
|
|
|
|
/**
|
|
* output image
|
|
*
|
|
* @var gd_image
|
|
*/
|
|
protected $output_image;
|
|
|
|
/**
|
|
* total number of data bits
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $total_data_bits;
|
|
|
|
/**
|
|
* Error correction indicator
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $error_correct;
|
|
|
|
/**
|
|
* lookup table for alphanumeric characters
|
|
* Note: the '$' index is in single quotes as otherwise
|
|
* PHP tries to parse it as a variable (see PEAR bug #17321)
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $alphanumeric_hash = array(
|
|
"0"=>0,"1"=>1,"2"=>2,"3"=>3,"4"=>4,
|
|
"5"=>5,"6"=>6,"7"=>7,"8"=>8,"9"=>9,"A"=>10,"B"=>11,"C"=>12,"D"=>13,"E"=>14,
|
|
"F"=>15,"G"=>16,"H"=>17,"I"=>18,"J"=>19,"K"=>20,"L"=>21,"M"=>22,"N"=>23,
|
|
"O"=>24,"P"=>25,"Q"=>26,"R"=>27,"S"=>28,"T"=>29,"U"=>30,"V"=>31,
|
|
"W"=>32,"X"=>33,"Y"=>34,"Z"=>35," "=>36,'$'=>37,"%"=>38,"*"=>39,
|
|
"+"=>40,"-"=>41,"."=>42,"/"=>43,":"=>44
|
|
);
|
|
|
|
/**
|
|
* Maximum data bits lookup
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $max_data_bits_array = array(
|
|
0,128,224,352,512,688,864,992,1232,1456,1728,
|
|
2032,2320,2672,2920,3320,3624,4056,4504,5016,5352,
|
|
5712,6256,6880,7312,8000,8496,9024,9544,10136,10984,
|
|
11640,12328,13048,13800,14496,15312,15936,16816,17728,18672,
|
|
|
|
152,272,440,640,864,1088,1248,1552,1856,2192,
|
|
2592,2960,3424,3688,4184,4712,5176,5768,6360,6888,
|
|
7456,8048,8752,9392,10208,10960,11744,12248,13048,13880,
|
|
14744,15640,16568,17528,18448,19472,20528,21616,22496,23648,
|
|
|
|
72,128,208,288,368,480,528,688,800,976,
|
|
1120,1264,1440,1576,1784,2024,2264,2504,2728,3080,
|
|
3248,3536,3712,4112,4304,4768,5024,5288,5608,5960,
|
|
6344,6760,7208,7688,7888,8432,8768,9136,9776,10208,
|
|
|
|
104,176,272,384,496,608,704,880,1056,1232,
|
|
1440,1648,1952,2088,2360,2600,2936,3176,3560,3880,
|
|
4096,4544,4912,5312,5744,6032,6464,6968,7288,7880,
|
|
8264,8920,9368,9848,10288,10832,11408,12016,12656,13328
|
|
);
|
|
|
|
/**
|
|
* Maximum number of codewords, dependant on barcode version
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $max_codewords_array = array(
|
|
0,26,44,70,100,134,172,196,242,
|
|
292,346,404,466,532,581,655,733,815,901,991,1085,1156,
|
|
1258,1364,1474,1588,1706,1828,1921,2051,2185,2323,2465,
|
|
2611,2761,2876,3034,3196,3362,3532,3706
|
|
);
|
|
|
|
/**
|
|
* Formatting data for the final barcode
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $format_array = array(
|
|
"101010000010010","101000100100101",
|
|
"101111001111100","101101101001011","100010111111001","100000011001110",
|
|
"100111110010111","100101010100000","111011111000100","111001011110011",
|
|
"111110110101010","111100010011101","110011000101111","110001100011000",
|
|
"110110001000001","110100101110110","001011010001001","001001110111110",
|
|
"001110011100111","001100111010000","000011101100010","000001001010101",
|
|
"000110100001100","000100000111011","011010101011111","011000001101000",
|
|
"011111100110001","011101000000110","010010010110100","010000110000011",
|
|
"010111011011010","010101111101101"
|
|
);
|
|
|
|
/**
|
|
* Maximum number of data bits
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $max_data_bits;
|
|
|
|
/**
|
|
* Size of the module
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $module_size;
|
|
|
|
/**
|
|
* Data values
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $data_value;
|
|
|
|
/**
|
|
* Incremental counter, pointer into the data arrays
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $data_counter;
|
|
|
|
/**
|
|
* Data bits array
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $data_bits;
|
|
|
|
/**
|
|
* Codeword pointer, incremental
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $codeword_num_counter_value;
|
|
|
|
/**
|
|
* Codeword details
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $codeword_num_plus;
|
|
|
|
/**
|
|
* RS-ECC codewords
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $rs_ecc_codewords;
|
|
|
|
/**
|
|
* Codewords calculated prior to matrix generation
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $codewords;
|
|
|
|
/**
|
|
* Maximum number of data codewords
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $max_data_codewords;
|
|
|
|
/**
|
|
* Final matrix details for plotting barcode
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $matrix_content;
|
|
|
|
/**
|
|
* Matrix X array
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $matrix_x_array;
|
|
|
|
/**
|
|
* Matrix Y array
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $matrix_y_array;
|
|
|
|
/**
|
|
* Masking array - used in final matrix generation
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $mask_array;
|
|
|
|
/**
|
|
* RS Block order array
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $rs_block_order;
|
|
|
|
/**
|
|
* RS Calculation table array
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $rs_cal_table_array;
|
|
|
|
/**
|
|
* Byte number counter
|
|
*
|
|
* @var integer
|
|
*/
|
|
protected $byte_num;
|
|
|
|
/**
|
|
* Class constructor
|
|
*
|
|
* @param array $options An array of options for the class
|
|
*/
|
|
public function __construct($options = array())
|
|
{
|
|
$this->path = "@data_dir@" . DIRECTORY_SEPARATOR . "Image_QRCode" .
|
|
DIRECTORY_SEPARATOR . "data";
|
|
if ("@data_dir@" == "@" . "data_dir@") {
|
|
// development path
|
|
$this->path = dirname(__FILE__) . "/../data";
|
|
}
|
|
|
|
$this->image_path = "@data_dir@" . DIRECTORY_SEPARATOR . "Image_QRCode" .
|
|
DIRECTORY_SEPARATOR . "image";
|
|
if ("@data_dir@" == "@" . "data_dir@") {
|
|
// development path
|
|
$this->image_path = dirname(__FILE__) . "/../image";
|
|
}
|
|
|
|
$this->version_ul = 40;
|
|
|
|
$this->data_string = "";
|
|
|
|
$this->output_type = "display";
|
|
$this->image_type = "png";
|
|
|
|
$this->total_data_bits = 0;
|
|
$this->error_correct = "M";
|
|
$this->module_size = 0;
|
|
|
|
$this->matrix_content = array();
|
|
|
|
// handle options
|
|
if (array_key_exists("version", $options)) {
|
|
$this->setVersion($options["version"]);
|
|
}
|
|
if (array_key_exists("path", $options)) {
|
|
$this->path = $options["path"];
|
|
}
|
|
if (array_key_exists("image_path", $options)) {
|
|
$this->image_path = $options["image_path"];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets the data to encode
|
|
*
|
|
* @param string $str the data to encode into the barcode
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setData($str)
|
|
{
|
|
$str = trim($str);
|
|
|
|
if (strlen($str) == 0) {
|
|
throw new Image_QRCode_Exception("Data cannot be empty");
|
|
}
|
|
$this->data_string = $str;
|
|
$this->data_length = strlen($str);
|
|
}
|
|
|
|
/**
|
|
* Sets the error correction level.
|
|
* One of:
|
|
* L: 7% error level
|
|
* M: 15% error level
|
|
* Q: 25% error level
|
|
* H: 30% error level
|
|
*
|
|
* @param char $e the error level to use
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setErrorCorrect($e)
|
|
{
|
|
$e = strtoupper(trim($e));
|
|
if (!in_array($e, array("L", "M", "Q", "H"))) {
|
|
throw new Image_QRCode_Exception("Error correction level not supported");
|
|
}
|
|
$this->error_correct = $e;
|
|
}
|
|
|
|
/**
|
|
* Sets the default module size
|
|
* Defaults are PNG: 4, JPEG: 8
|
|
*
|
|
* @param integer $size the module size to use
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setModuleSize($size)
|
|
{
|
|
$size = intval($size);
|
|
if ($size < 1) {
|
|
throw new Image_QRCode_Exception("Module size is invalid");
|
|
}
|
|
$this->module_size = $size;
|
|
}
|
|
|
|
/**
|
|
* Sets the version number to use
|
|
* (between 1 and 40)
|
|
*
|
|
* Version 1 is 21*21 matrix
|
|
* and 4 modules increases whenever 1 version increases.
|
|
* So version 40 is 177*177 matrix.
|
|
*
|
|
* @param integer $v the version number to use for this barcode
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setVersion($v)
|
|
{
|
|
$v = intval($v);
|
|
if ($v < 1 || $v > $this->version_ul) {
|
|
throw new Image_QRCode_Exception("Version is invalid");
|
|
}
|
|
$this->version = $v;
|
|
}
|
|
|
|
/**
|
|
* Defines the return image type
|
|
* (from 'jpeg', or 'png')
|
|
*
|
|
* Default is PNG format
|
|
*
|
|
* @param string $t the image type to use
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setImageType($t)
|
|
{
|
|
$t = strtolower($t);
|
|
if (!in_array($t, array("jpeg", "png"))) {
|
|
throw new Image_QRCode_Exception("Image format not supported");
|
|
}
|
|
$this->image_type = $t;
|
|
}
|
|
|
|
/**
|
|
* Sets what to do once the code has been generated
|
|
* ('display' or 'return')
|
|
*
|
|
* Default is to display the image, complete with headers.
|
|
*
|
|
* @param string $t what to do with the image once it's been created
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function setOutputType($t)
|
|
{
|
|
$t = strtolower($t);
|
|
if (!in_array($t, array("display", "return"))) {
|
|
throw new Image_QRCode_Exception("Output type not supported");
|
|
}
|
|
$this->output_type = $t;
|
|
}
|
|
|
|
/**
|
|
* Sets the data to append to the code (experimental)
|
|
*
|
|
* @param array $data an array of elements (n, m, parity, originaldata)
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function setStructureAppend($data = array())
|
|
{
|
|
if (!array_key_exists("n", $data)
|
|
&& !array_key_exists("m", $data)
|
|
&& !array_key_exists("parity", $data)
|
|
&& !array_key_exists("originaldata", $data)
|
|
) {
|
|
throw new Image_QRCode_Exception("Appended structure data is not valid");
|
|
}
|
|
|
|
$this->structureappend_n = $data["n"];
|
|
$this->structureappend_m = $data["m"];
|
|
$this->structureappend_parity = $data["parity"];
|
|
$this->structureappend_originaldata = $data["originaldata"];
|
|
}
|
|
|
|
/**
|
|
* Performs all necessary calculations and returns/outputs an image
|
|
* as defined by the configuration
|
|
*
|
|
* @param string $data the string/data to encode into the barcode
|
|
* @param array $options an array of options for the barcode
|
|
* @param array $append additional data to append (experimental)
|
|
*
|
|
* @return resource
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
public function makeCode($data = "", $options = array(), $append = array())
|
|
{
|
|
// apply any passed options
|
|
if (array_key_exists("image_type", $options)) {
|
|
$this->setImageType($options["image_type"]);
|
|
}
|
|
if (array_key_exists("output_type", $options)) {
|
|
$this->setOutputType($options["output_type"]);
|
|
}
|
|
if (array_key_exists("error_correct", $options)) {
|
|
$this->setErrorCorrect($options["error_correct"]);
|
|
}
|
|
if (array_key_exists("module_size", $options)) {
|
|
$this->setModuleSize($options["module_size"]);
|
|
}
|
|
|
|
// set our data value
|
|
$this->setData($data);
|
|
|
|
// if we have an extra set of data to append, set this now
|
|
if (!empty($append)) {
|
|
$this->setStructureAppend($append);
|
|
}
|
|
|
|
// some checks to start with
|
|
if ($this->module_size <= 0) {
|
|
$this->module_size = 4;
|
|
}
|
|
|
|
$this->data_counter = 0;
|
|
|
|
// apply parity and m/n structure append data
|
|
$this->applyStructureAppend();
|
|
|
|
$this->data_bits[$this->data_counter] = 4;
|
|
|
|
// Determine the encoding mode, based on the input data
|
|
$this->determineEncoding();
|
|
|
|
if (@$this->data_bits[$this->data_counter] > 0) {
|
|
$this->data_counter++;
|
|
}
|
|
$i = 0;
|
|
$this->total_data_bits = 0;
|
|
while ($i < $this->data_counter) {
|
|
$this->total_data_bits += $this->data_bits[$i];
|
|
$i++;
|
|
}
|
|
|
|
$ecc_character_hash = array(
|
|
"L"=>"1",
|
|
"l"=>"1",
|
|
"M"=>"0",
|
|
"m"=>"0",
|
|
"Q"=>"3",
|
|
"q"=>"3",
|
|
"H"=>"2",
|
|
"h"=>"2"
|
|
);
|
|
|
|
$this->ec = @$ecc_character_hash[$this->error_correct];
|
|
|
|
if (!$this->ec) {
|
|
$this->ec = 0;
|
|
}
|
|
|
|
// Calculate the version of the code we're generating
|
|
$this->checkVersion();
|
|
|
|
$this->total_data_bits += $this->codeword_num_plus[$this->version];
|
|
$this->data_bits[$this->codeword_num_counter_value]
|
|
+= $this->codeword_num_plus[$this->version];
|
|
|
|
$max_codewords = $this->max_codewords_array[$this->version];
|
|
$max_modules_1side = 17 + ($this->version << 2);
|
|
|
|
$matrix_remain_bit = array(
|
|
0,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
|
|
);
|
|
|
|
/* Create our base clean matrix data, ready for our own data */
|
|
$this->emptyMatrix($max_modules_1side);
|
|
|
|
/* ECC -this is the heart of the codeword calculations */
|
|
$format_info = $this->performECCOperation(
|
|
$matrix_remain_bit,
|
|
$max_codewords
|
|
);
|
|
|
|
/* Attach calculated codeword data to the matrix */
|
|
$this->attachCodewordData($max_codewords, $matrix_remain_bit);
|
|
|
|
/* Mask selection */
|
|
$mask_number = $this->maskSelection($max_modules_1side);
|
|
$mask_content = 1 << $mask_number;
|
|
|
|
/* Calculate our format information */
|
|
$this->calculateFormatInformation($format_info, $mask_number);
|
|
|
|
/* Create base image based on calculated size */
|
|
$mib = $this->createBaseImage($max_modules_1side);
|
|
|
|
// Add actual matrix data to the base image
|
|
$this->addMatrixToImage($max_modules_1side, $mask_content);
|
|
|
|
// create output image, saved in $this->output_image
|
|
$this->createOutputImage($mib);
|
|
|
|
// what are we doing with the created output image?
|
|
if ($this->output_type == "return") {
|
|
return $this->output_image;
|
|
} else {
|
|
switch ($this->image_type)
|
|
{
|
|
case "jpeg":
|
|
header("Content-type: image/jpeg");
|
|
imagejpeg($this->output_image);
|
|
break;
|
|
|
|
case "png":
|
|
header("Content-type: image/png");
|
|
imagepng($this->output_image);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Performs version checking/calculation
|
|
*
|
|
* @return void
|
|
* @throws Image_QRCode_Exception
|
|
*/
|
|
protected function checkVersion()
|
|
{
|
|
if (!is_numeric($this->version)) {
|
|
$this->version = 0;
|
|
}
|
|
|
|
if (!$this->version) {
|
|
$i = 1 + 40 * $this->ec;
|
|
$j = $i + 39;
|
|
$this->version = 1;
|
|
while ($i <= $j) {
|
|
$cw = $this->codeword_num_plus[$this->version];
|
|
|
|
$max = $this->max_data_bits_array[$i];
|
|
$tdb = $this->total_data_bits + $cw;
|
|
|
|
if ($max >= $tdb) {
|
|
$this->max_data_bits = $this->max_data_bits_array[$i];
|
|
break;
|
|
}
|
|
$i++;
|
|
$this->version++;
|
|
}
|
|
} else {
|
|
$b = $this->max_data_bits_array[$this->version + 40 * $this->ec];
|
|
$this->max_data_bits = $b;
|
|
}
|
|
if ($this->version > $this->version_ul) {
|
|
throw new Image_QRCode_Exception("Version number is too large");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Determines the encoding needed for the data provided
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function determineEncoding()
|
|
{
|
|
if (preg_match("/[^0-9]/", $this->data_string) != 0) {
|
|
|
|
$expr = "/[^0-9A-Z \$\*\%\+\.\/\:\-]/";
|
|
if (preg_match($expr, $this->data_string) != 0) {
|
|
|
|
/* 8-bit byte mode */
|
|
|
|
$this->codeword_num_plus = array(
|
|
0,0,0,0,0,0,0,0,0,0,
|
|
8,8,8,8,8,8,8,8,8,8,8,8,8,8,8,8,8,
|
|
8,8,8,8,8,8,8,8,8,8,8,8,8,8
|
|
);
|
|
|
|
$this->data_value[$this->data_counter] = 4;
|
|
$this->data_counter++;
|
|
$this->data_value[$this->data_counter] = $this->data_length;
|
|
$this->data_bits[$this->data_counter] = 8; /* version 1-9 */
|
|
$this->codeword_num_counter_value = $this->data_counter;
|
|
|
|
$this->data_counter++;
|
|
$i = 0;
|
|
while ($i < $this->data_length) {
|
|
$this->data_value[$this->data_counter] = ord(
|
|
substr(
|
|
$this->data_string,
|
|
$i,
|
|
1
|
|
)
|
|
);
|
|
$this->data_bits[$this->data_counter] = 8;
|
|
$this->data_counter++;
|
|
$i++;
|
|
}
|
|
} else {
|
|
|
|
/* Alphanumeric mode */
|
|
|
|
$this->codeword_num_plus = array(
|
|
0,0,0,0,0,0,0,0,0,0,
|
|
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
4,4,4,4,4,4,4,4,4,4,4,4,4,4
|
|
);
|
|
|
|
$this->data_value[$this->data_counter] = 2;
|
|
$this->data_counter++;
|
|
$this->data_value[$this->data_counter] = $this->data_length;
|
|
$this->data_bits[$this->data_counter] = 9; /* version 1-9 */
|
|
$this->codeword_num_counter_value = $this->data_counter;
|
|
|
|
$i = 0;
|
|
$this->data_counter++;
|
|
while ($i < $this->data_length) {
|
|
if (($i % 2) == 0) {
|
|
$c = substr($this->data_string, $i, 1);
|
|
$h = $this->alphanumeric_hash[$c];
|
|
$this->data_value[$this->data_counter] = $h;
|
|
$this->data_bits[$this->data_counter] = 6;
|
|
} else {
|
|
$c = substr($this->data_string, $i, 1);
|
|
$h = $this->alphanumeric_hash[$c];
|
|
|
|
$this->data_value[$this->data_counter]
|
|
= $this->data_value[$this->data_counter] * 45 + $h;
|
|
$this->data_bits[$this->data_counter] = 11;
|
|
$this->data_counter++;
|
|
}
|
|
$i++;
|
|
}
|
|
}
|
|
} else {
|
|
|
|
/* Numeric mode */
|
|
|
|
$this->codeword_num_plus = array(
|
|
0,0,0,0,0,0,0,0,0,0,
|
|
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
4,4,4,4,4,4,4,4,4,4,4,4,4,4
|
|
);
|
|
|
|
$this->data_value[$this->data_counter] = 1;
|
|
$this->data_counter++;
|
|
$this->data_value[$this->data_counter] = $this->data_length;
|
|
$this->data_bits[$this->data_counter] = 10; /* #version 1-9 */
|
|
$this->codeword_num_counter_value = $this->data_counter;
|
|
|
|
$i = 0;
|
|
$this->data_counter++;
|
|
while ($i < $this->data_length) {
|
|
if (($i % 3) == 0) {
|
|
$this->data_value[$this->data_counter]
|
|
= substr($this->data_string, $i, 1);
|
|
$this->data_bits[$this->data_counter] = 4;
|
|
} else {
|
|
$c = substr($this->data_string, $i, 1);
|
|
$this->data_value[$this->data_counter]
|
|
= $this->data_value[$this->data_counter] * 10 + $c;
|
|
|
|
if (($i % 3) == 1) {
|
|
$this->data_bits[$this->data_counter] = 7;
|
|
} else {
|
|
$this->data_bits[$this->data_counter] = 10;
|
|
$this->data_counter++;
|
|
}
|
|
}
|
|
$i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Performs the error correction and code generation operations
|
|
*
|
|
* @param array $matrix_remain_bit array of remaining bits for data matrix
|
|
* @param integer $max_codewords number of maximum codewords
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function performECCOperation($matrix_remain_bit, $max_codewords)
|
|
{
|
|
// get our ECC data in from the relevant file
|
|
$format_info = $this->readECCData($matrix_remain_bit, $max_codewords);
|
|
|
|
$codewordsCounter = $this->divideBy8Bit();
|
|
$codewordsCounter = $this->setPaddingCharacter($codewordsCounter);
|
|
|
|
// Do the actual RS-ECC magic
|
|
$this->doRSCalculations();
|
|
|
|
return $format_info; // used in main makeCode()
|
|
}
|
|
|
|
/**
|
|
* Read in correct baseline data from ECC/RS files
|
|
*
|
|
* @param array $matrix_remain_bit array of remaining bits for data matrix
|
|
* @param integer $max_codewords number of maximum codewords
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function readECCData($matrix_remain_bit, $max_codewords)
|
|
{
|
|
$this->byte_num = $matrix_remain_bit[$this->version]
|
|
+ ($max_codewords << 3);
|
|
$filename = $this->path
|
|
. "/qrv"
|
|
. $this->version
|
|
. "_"
|
|
. $this->ec
|
|
. ".dat";
|
|
if (!file_exists($filename)) {
|
|
throw new Image_QRCode_Exception("Can't open ECC data file");
|
|
}
|
|
$fp1 = fopen($filename, "rb");
|
|
|
|
$matx = fread($fp1, $this->byte_num);
|
|
$maty = fread($fp1, $this->byte_num);
|
|
$masks = fread($fp1, $this->byte_num);
|
|
$fi_x = fread($fp1, 15);
|
|
$fi_y = fread($fp1, 15);
|
|
$this->rs_ecc_codewords = ord(fread($fp1, 1));
|
|
$rso = fread($fp1, 128);
|
|
fclose($fp1);
|
|
|
|
$this->matrix_x_array = unpack("C*", $matx);
|
|
$this->matrix_y_array = unpack("C*", $maty);
|
|
$this->mask_array = unpack("C*", $masks);
|
|
|
|
$this->rs_block_order = unpack("C*", $rso);
|
|
|
|
$format_information_x2 = unpack("C*", $fi_x);
|
|
$format_information_y2 = unpack("C*", $fi_y);
|
|
|
|
$format_information_x1 = array(0,1,2,3,4,5,7,8,8,8,8,8,8,8,8);
|
|
$format_information_y1 = array(8,8,8,8,8,8,8,8,7,5,4,3,2,1,0);
|
|
|
|
$format_info = array(
|
|
"x1" => $format_information_x1,
|
|
"x2" => $format_information_x2,
|
|
"y1" => $format_information_y1,
|
|
"y2" => $format_information_y2
|
|
);
|
|
|
|
$this->max_data_codewords = ($this->max_data_bits >> 3);
|
|
|
|
$filename = $this->path . "/rsc" . $this->rs_ecc_codewords . ".dat";
|
|
if (!file_exists($filename)) {
|
|
throw new Image_QRCode_Exception("Can't open rsc data file");
|
|
}
|
|
$fp0 = fopen($filename, "rb");
|
|
|
|
$i = 0;
|
|
while ($i < 256) {
|
|
$this->rs_cal_table_array[$i] = fread($fp0, $this->rs_ecc_codewords);
|
|
$i++;
|
|
}
|
|
fclose($fp0);
|
|
|
|
/* Set terminator for data */
|
|
|
|
if ($this->total_data_bits <= $this->max_data_bits-4) {
|
|
$this->data_value[$this->data_counter] = 0;
|
|
$this->data_bits[$this->data_counter] = 4;
|
|
} else {
|
|
if ($this->total_data_bits < $this->max_data_bits) {
|
|
$this->data_value[$this->data_counter] = 0;
|
|
$this->data_bits[$this->data_counter]
|
|
= $this->max_data_bits-$this->total_data_bits;
|
|
} else {
|
|
if ($this->total_data_bits > $this->max_data_bits) {
|
|
throw new Image_QRCode_Exception("Overflow exception");
|
|
}
|
|
}
|
|
}
|
|
|
|
return $format_info;
|
|
}
|
|
|
|
/**
|
|
* Gets data into 8-bit format
|
|
*
|
|
* @return integer
|
|
*/
|
|
protected function divideBy8Bit()
|
|
{
|
|
$i = 0;
|
|
$codewords_counter = 0;
|
|
$this->codewords[0] = 0;
|
|
$remaining_bits = 8;
|
|
|
|
while ($i <= $this->data_counter) {
|
|
|
|
$buffer = @$this->data_value[$i];
|
|
$buffer_bits = @$this->data_bits[$i];
|
|
|
|
$flag = 1;
|
|
while ($flag) {
|
|
|
|
if ($remaining_bits > $buffer_bits) {
|
|
$this->codewords[$codewords_counter]
|
|
= ((@$this->codewords[$codewords_counter] << $buffer_bits)
|
|
| $buffer);
|
|
$remaining_bits -= $buffer_bits;
|
|
$flag = 0;
|
|
} else {
|
|
$buffer_bits -= $remaining_bits;
|
|
$this->codewords[$codewords_counter]
|
|
= (($this->codewords[$codewords_counter] << $remaining_bits)
|
|
| ($buffer >> $buffer_bits));
|
|
|
|
if ($buffer_bits == 0) {
|
|
$flag = 0;
|
|
} else {
|
|
$buffer = ($buffer & ((1 << $buffer_bits)-1));
|
|
$flag = 1;
|
|
}
|
|
|
|
$codewords_counter++;
|
|
if ($codewords_counter < $this->max_data_codewords-1) {
|
|
$this->codewords[$codewords_counter] = 0;
|
|
}
|
|
$remaining_bits = 8;
|
|
}
|
|
}
|
|
$i++;
|
|
}
|
|
|
|
if ($remaining_bits != 8) {
|
|
$this->codewords[$codewords_counter]
|
|
= $this->codewords[$codewords_counter] << $remaining_bits;
|
|
} else {
|
|
$codewords_counter--;
|
|
}
|
|
|
|
return $codewords_counter;
|
|
}
|
|
|
|
/**
|
|
* Sets the padding character to pad out data
|
|
*
|
|
* @param integer $codewordsCounter counter for current no. of codewords
|
|
*
|
|
* @return integer
|
|
*/
|
|
protected function setPaddingCharacter($codewordsCounter)
|
|
{
|
|
if ($codewordsCounter < $this->max_data_codewords-1) {
|
|
$flag = 1;
|
|
while ($codewordsCounter < $this->max_data_codewords-1) {
|
|
|
|
$codewordsCounter++;
|
|
if ($flag == 1) {
|
|
$this->codewords[$codewordsCounter] = 236;
|
|
} else {
|
|
$this->codewords[$codewordsCounter] = 17;
|
|
}
|
|
$flag = $flag * -1;
|
|
}
|
|
}
|
|
return $codewordsCounter;
|
|
}
|
|
|
|
/**
|
|
* Calculates the format information for the barcode
|
|
*
|
|
* @param array $format_info format information from data files
|
|
* @param integer $mask_number mask integer to use
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function calculateFormatInformation($format_info, $mask_number)
|
|
{
|
|
$format_value = (($this->ec << 3) | $mask_number);
|
|
$i = 0;
|
|
while ($i < 15) {
|
|
$content = substr($this->format_array[$format_value], $i, 1);
|
|
|
|
$x = $format_info["x1"][$i];
|
|
$y = $format_info["y1"][$i];
|
|
$this->matrix_content[$x][$y] = $content * 255;
|
|
|
|
$x = $format_info["x2"][$i+1];
|
|
$y = $format_info["y2"][$i+1];
|
|
$this->matrix_content[$x][$y] = $content * 255;
|
|
|
|
$i++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Perform the actual RS calculations on the data
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function doRSCalculations()
|
|
{
|
|
// Preparation
|
|
$i = 0;
|
|
$j = 0;
|
|
$rs_block_number = 0;
|
|
$rs_temp[0] = "";
|
|
|
|
while ($i < $this->max_data_codewords) {
|
|
$rs_temp[$rs_block_number] .= chr($this->codewords[$i]);
|
|
$j++;
|
|
|
|
$v = $this->rs_block_order[$rs_block_number+1]
|
|
- $this->rs_ecc_codewords;
|
|
if ($j >= $v) {
|
|
$j = 0;
|
|
$rs_block_number++;
|
|
$rs_temp[$rs_block_number] = "";
|
|
}
|
|
$i++;
|
|
}
|
|
|
|
// RS-ECC main calculation
|
|
$rs_block_number = 0;
|
|
$this->rs_block_order_num = count($this->rs_block_order);
|
|
|
|
while ($rs_block_number < $this->rs_block_order_num) {
|
|
$rs_codewords = $this->rs_block_order[$rs_block_number+1];
|
|
$rs_data_codewords = $rs_codewords-$this->rs_ecc_codewords;
|
|
|
|
$rstemp = $rs_temp[$rs_block_number]
|
|
. str_repeat(chr(0), $this->rs_ecc_codewords);
|
|
$padding_data = str_repeat(chr(0), $rs_data_codewords);
|
|
|
|
$j = $rs_data_codewords;
|
|
while ($j > 0) {
|
|
$first = ord(substr($rstemp, 0, 1));
|
|
|
|
if ($first) {
|
|
$left_chr = substr($rstemp, 1);
|
|
$cal = $this->rs_cal_table_array[$first] . $padding_data;
|
|
$rstemp = $left_chr ^ $cal;
|
|
} else {
|
|
$rstemp=substr($rstemp, 1);
|
|
}
|
|
|
|
$j--;
|
|
}
|
|
|
|
$this->codewords = array_merge(
|
|
$this->codewords,
|
|
unpack("C*", $rstemp)
|
|
);
|
|
|
|
$rs_block_number++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates the final GD output image block
|
|
* ready for sending wherever the user desires
|
|
*
|
|
* @param integer $mib destination image width/height
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function createOutputImage($mib)
|
|
{
|
|
imagecopyresized(
|
|
$this->output_image,
|
|
$this->base_image,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
$this->image_size,
|
|
$this->image_size,
|
|
$mib,
|
|
$mib
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Adds m/n and parity data for structure append (if supplied)
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function applyStructureAppend()
|
|
{
|
|
if ($this->structureappend_n > 1
|
|
&& $this->structureappend_n <= 16
|
|
&& $this->structureappend_m > 0
|
|
&& $this->structureappend_m <= 16
|
|
) {
|
|
|
|
$this->data_value[0] = 3;
|
|
$this->data_bits[0] = 4;
|
|
|
|
$this->data_value[1] = $this->structureappend_m - 1;
|
|
$this->data_bits[1] = 4;
|
|
|
|
$this->data_value[2] = $this->structureappend_n - 1;
|
|
$this->data_bits[2] = 4;
|
|
|
|
|
|
$originaldata_length = strlen($this->structureappend_originaldata);
|
|
if ($originaldata_length > 1) {
|
|
$this->structureappend_parity = 0;
|
|
$i = 0;
|
|
while ($i < $originaldata_length) {
|
|
$ord = ord(
|
|
substr(
|
|
$this->structureappend_originaldata,
|
|
$i,
|
|
1
|
|
)
|
|
);
|
|
$this->structureappend_parity
|
|
= $this->structureappend_parity ^ $ord;
|
|
$i++;
|
|
}
|
|
}
|
|
|
|
$this->data_value[3] = $this->structureappend_parity;
|
|
$this->data_bits[3] = 8;
|
|
|
|
$this->data_counter = 4;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Attaches the calculated codeword data to the data matrix
|
|
*
|
|
* @param integer $max_codewords Maximum number of codewords for this code
|
|
* @param integer $matrix_remain_bit Dependant on the version of the code
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function attachCodewordData($max_codewords, $matrix_remain_bit)
|
|
{
|
|
$i = 0;
|
|
while ($i < $max_codewords) {
|
|
$codeword_i=$this->codewords[$i];
|
|
$j = 8;
|
|
while ($j >= 1) {
|
|
$codeword_bits_number = ($i << 3) + $j;
|
|
|
|
$x = $this->matrix_x_array[$codeword_bits_number];
|
|
$y = $this->matrix_y_array[$codeword_bits_number];
|
|
|
|
$this->matrix_content[$x][$y]
|
|
= ((255*($codeword_i & 1))
|
|
^ $this->mask_array[$codeword_bits_number]);
|
|
|
|
$codeword_i= $codeword_i >> 1;
|
|
$j--;
|
|
}
|
|
$i++;
|
|
}
|
|
|
|
$matrix_remain = $matrix_remain_bit[$this->version];
|
|
while ($matrix_remain) {
|
|
$remain_bit_temp = $matrix_remain + ( $max_codewords << 3);
|
|
|
|
$x = $this->matrix_x_array[$remain_bit_temp];
|
|
$y = $this->matrix_y_array[$remain_bit_temp];
|
|
|
|
$this->matrix_content[$x][$y]
|
|
= (255 ^ $this->mask_array[$remain_bit_temp]);
|
|
$matrix_remain--;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Selects the mask to use
|
|
*
|
|
* @param integer $max_modules_1side Max number of modules (single side)
|
|
*
|
|
* @return integer
|
|
*/
|
|
protected function maskSelection($max_modules_1side)
|
|
{
|
|
$min_demerit_score = 0;
|
|
$hor_master = "";
|
|
$ver_master = "";
|
|
$k = 0;
|
|
while ($k < $max_modules_1side) {
|
|
$l = 0;
|
|
while ($l < $max_modules_1side) {
|
|
$hor_master = $hor_master . chr($this->matrix_content[$l][$k]);
|
|
$ver_master = $ver_master . chr($this->matrix_content[$k][$l]);
|
|
$l++;
|
|
}
|
|
$k++;
|
|
}
|
|
$i = 0;
|
|
$all_matrix = $max_modules_1side * $max_modules_1side;
|
|
while ($i < 8) {
|
|
$demerit_n1 = 0;
|
|
$ptn_temp = array();
|
|
$bit = 1 << $i;
|
|
$bit_r = (~$bit) & 255;
|
|
$bit_mask = str_repeat(chr($bit), $all_matrix);
|
|
$hor = $hor_master & $bit_mask;
|
|
$ver = $ver_master & $bit_mask;
|
|
|
|
$ver_shift1 = $ver . str_repeat(chr(170), $max_modules_1side);
|
|
$ver_shift2 = str_repeat(chr(170), $max_modules_1side) . $ver;
|
|
$ver_shift1_0 = $ver . str_repeat(chr(0), $max_modules_1side);
|
|
$ver_shift2_0 = str_repeat(chr(0), $max_modules_1side) . $ver;
|
|
$ver_or = chunk_split(
|
|
~($ver_shift1 | $ver_shift2),
|
|
$max_modules_1side, chr(170)
|
|
);
|
|
$ver_and = chunk_split(
|
|
~($ver_shift1_0 & $ver_shift2_0),
|
|
$max_modules_1side, chr(170)
|
|
);
|
|
|
|
$hor = chunk_split(~$hor, $max_modules_1side, chr(170));
|
|
$ver = chunk_split(~$ver, $max_modules_1side, chr(170));
|
|
$hor = $hor . chr(170) . $ver;
|
|
|
|
$n1_search = "/"
|
|
. str_repeat(chr(255), 5)
|
|
. "+|"
|
|
. str_repeat(chr($bit_r), 5)
|
|
. "+/";
|
|
$n3_search = chr($bit_r)
|
|
. chr(255)
|
|
. chr($bit_r)
|
|
. chr($bit_r)
|
|
. chr($bit_r)
|
|
. chr(255)
|
|
. chr($bit_r);
|
|
|
|
$demerit_n3 = substr_count($hor, $n3_search) * 40;
|
|
|
|
$sc = substr_count($ver, chr($bit_r));
|
|
$demerit_n4 = floor(abs(((100 * ($sc / ($this->byte_num))) - 50) / 5));
|
|
$demerit_n4 *= 10;
|
|
|
|
$n2_search1 = "/" . chr($bit_r) . chr($bit_r) . "+/";
|
|
$n2_search2 = "/" . chr(255) . chr(255) . "+/";
|
|
$demerit_n2 = 0;
|
|
|
|
preg_match_all($n2_search1, $ver_and, $ptn_temp);
|
|
foreach ($ptn_temp[0] as $str_temp) {
|
|
$demerit_n2 += (strlen($str_temp) - 1);
|
|
}
|
|
|
|
$ptn_temp = array();
|
|
preg_match_all($n2_search2, $ver_or, $ptn_temp);
|
|
foreach ($ptn_temp[0] as $str_temp) {
|
|
$demerit_n2 += (strlen($str_temp) - 1);
|
|
}
|
|
$demerit_n2*=3;
|
|
|
|
$ptn_temp=array();
|
|
preg_match_all($n1_search, $hor, $ptn_temp);
|
|
foreach ($ptn_temp[0] as $str_temp) {
|
|
$demerit_n1 += (strlen($str_temp) - 2);
|
|
}
|
|
|
|
$demerit_score = $demerit_n1 + $demerit_n2;
|
|
$demerit_score += $demerit_n3 + $demerit_n4;
|
|
|
|
if ($demerit_score <= $min_demerit_score || $i == 0) {
|
|
$mask_number = $i;
|
|
$min_demerit_score = $demerit_score;
|
|
}
|
|
|
|
$i++;
|
|
}
|
|
|
|
return $mask_number;
|
|
}
|
|
|
|
/**
|
|
* Clears the output matrix array, ready for new data
|
|
*
|
|
* @param integer $max_modules_1side Max number of modules (single side)
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function emptyMatrix($max_modules_1side)
|
|
{
|
|
$i = 0;
|
|
while ($i < $max_modules_1side) {
|
|
$j = 0;
|
|
while ($j < $max_modules_1side) {
|
|
$this->matrix_content[$j][$i] = 0;
|
|
$j++;
|
|
}
|
|
$i++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates the base square image, based on the module size
|
|
*
|
|
* @param integer $max_modules_1side Max number of modules (single side)
|
|
*
|
|
* @return integer
|
|
*/
|
|
protected function createBaseImage($max_modules_1side)
|
|
{
|
|
$mib = $max_modules_1side+8;
|
|
$this->image_size = $mib * $this->module_size;
|
|
if ($this->image_size > 1480) {
|
|
throw new Image_QRCode_Exception("Image size is too large");
|
|
}
|
|
|
|
// image is square
|
|
$this->output_image = imagecreate(
|
|
$this->image_size,
|
|
$this->image_size
|
|
);
|
|
|
|
// load our base image in
|
|
$image_path = $this->image_path . "/qrv" . $this->version . ".png";
|
|
if (!file_exists($image_path)) {
|
|
throw new Image_QRCode_Exception("Base image not found");
|
|
}
|
|
$this->base_image = imagecreatefrompng($image_path);
|
|
if (!$this->base_image) {
|
|
throw new Image_QRCode_Exception("Couldn't load base image");
|
|
}
|
|
|
|
return $mib;
|
|
}
|
|
|
|
/**
|
|
* Adds the calculated matrix data to the base image.
|
|
*
|
|
* @param integer $max_modules_1side Max number of modules (single side)
|
|
* @param integer $mask_content Mask based on original data
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function addMatrixToImage($max_modules_1side, $mask_content)
|
|
{
|
|
// Create colours that we need to use
|
|
// NB if you want your barcode to be in a different colour,
|
|
// simply changing these colours will not work, as the base image
|
|
// is in black and white to start with.
|
|
$col[1] = imagecolorallocate($this->base_image, 0, 0, 0);
|
|
$col[0] = imagecolorallocate($this->base_image, 255, 255, 255);
|
|
|
|
$i = 4;
|
|
$mxe = 4 + $max_modules_1side;
|
|
$ii = 0;
|
|
while ($i < $mxe) {
|
|
$j = 4;
|
|
$jj = 0;
|
|
while ($j < $mxe) {
|
|
if ($this->matrix_content[$ii][$jj] & $mask_content) {
|
|
imagesetpixel($this->base_image, $i, $j, $col[1]);
|
|
}
|
|
$j++;
|
|
$jj++;
|
|
}
|
|
$i++;
|
|
$ii++;
|
|
}
|
|
}
|
|
}
|
|
?>
|