| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502 | <?php/* * PHP QR Code encoder * * Main encoder classes. * * Based on libqrencode C library distributed under LGPL 2.1 * Copyright (C) 2006, 2007, 2008, 2009 Kentaro Fukuchi <fukuchi@megaui.net> * * PHP QR Code is distributed under LGPL 3 * Copyright (C) 2010 Dominik Dzienia <deltalab at poczta dot fm> * * 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 3 of the License, or 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; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */     class QRrsblock {        public $dataLength;        public $data = array();        public $eccLength;        public $ecc = array();                public function __construct($dl, $data, $el, &$ecc, QRrsItem $rs)        {            $rs->encode_rs_char($data, $ecc);                    $this->dataLength = $dl;            $this->data = $data;            $this->eccLength = $el;            $this->ecc = $ecc;        }    };        //##########################################################################    class QRrawcode {        public $version;        public $datacode = array();        public $ecccode = array();        public $blocks;        public $rsblocks = array(); //of RSblock        public $count;        public $dataLength;        public $eccLength;        public $b1;                //----------------------------------------------------------------------        public function __construct(QRinput $input)        {            $spec = array(0,0,0,0,0);                        $this->datacode = $input->getByteStream();            if(is_null($this->datacode)) {                throw new Exception('null imput string');            }            QRspec::getEccSpec($input->getVersion(), $input->getErrorCorrectionLevel(), $spec);            $this->version = $input->getVersion();            $this->b1 = QRspec::rsBlockNum1($spec);            $this->dataLength = QRspec::rsDataLength($spec);            $this->eccLength = QRspec::rsEccLength($spec);            $this->ecccode = array_fill(0, $this->eccLength, 0);            $this->blocks = QRspec::rsBlockNum($spec);                        $ret = $this->init($spec);            if($ret < 0) {                throw new Exception('block alloc error');                return null;            }            $this->count = 0;        }                //----------------------------------------------------------------------        public function init(array $spec)        {            $dl = QRspec::rsDataCodes1($spec);            $el = QRspec::rsEccCodes1($spec);            $rs = QRrs::init_rs(8, 0x11d, 0, 1, $el, 255 - $dl - $el);                        $blockNo = 0;            $dataPos = 0;            $eccPos = 0;            for($i=0; $i<QRspec::rsBlockNum1($spec); $i++) {                $ecc = array_slice($this->ecccode,$eccPos);                $this->rsblocks[$blockNo] = new QRrsblock($dl, array_slice($this->datacode, $dataPos), $el,  $ecc, $rs);                $this->ecccode = array_merge(array_slice($this->ecccode,0, $eccPos), $ecc);                                $dataPos += $dl;                $eccPos += $el;                $blockNo++;            }            if(QRspec::rsBlockNum2($spec) == 0)                return 0;            $dl = QRspec::rsDataCodes2($spec);            $el = QRspec::rsEccCodes2($spec);            $rs = QRrs::init_rs(8, 0x11d, 0, 1, $el, 255 - $dl - $el);                        if($rs == NULL) return -1;                        for($i=0; $i<QRspec::rsBlockNum2($spec); $i++) {                $ecc = array_slice($this->ecccode,$eccPos);                $this->rsblocks[$blockNo] = new QRrsblock($dl, array_slice($this->datacode, $dataPos), $el, $ecc, $rs);                $this->ecccode = array_merge(array_slice($this->ecccode,0, $eccPos), $ecc);                                $dataPos += $dl;                $eccPos += $el;                $blockNo++;            }            return 0;        }                //----------------------------------------------------------------------        public function getCode()        {            $ret;            if($this->count < $this->dataLength) {                $row = $this->count % $this->blocks;                $col = $this->count / $this->blocks;                if($col >= $this->rsblocks[0]->dataLength) {                    $row += $this->b1;                }                $ret = $this->rsblocks[$row]->data[$col];            } else if($this->count < $this->dataLength + $this->eccLength) {                $row = ($this->count - $this->dataLength) % $this->blocks;                $col = ($this->count - $this->dataLength) / $this->blocks;                $ret = $this->rsblocks[$row]->ecc[$col];            } else {                return 0;            }            $this->count++;                        return $ret;        }    }    //##########################################################################        class QRcode {            public $version;        public $width;        public $data;                 //----------------------------------------------------------------------        public function encodeMask(QRinput $input, $mask)        {            if($input->getVersion() < 0 || $input->getVersion() > QRSPEC_VERSION_MAX) {                throw new Exception('wrong version');            }            if($input->getErrorCorrectionLevel() > QR_ECLEVEL_H) {                throw new Exception('wrong level');            }            $raw = new QRrawcode($input);                        QRtools::markTime('after_raw');                        $version = $raw->version;            $width = QRspec::getWidth($version);            $frame = QRspec::newFrame($version);                        $filler = new FrameFiller($width, $frame);            if(is_null($filler)) {                return NULL;            }            // inteleaved data and ecc codes            for($i=0; $i<$raw->dataLength + $raw->eccLength; $i++) {                $code = $raw->getCode();                $bit = 0x80;                for($j=0; $j<8; $j++) {                    $addr = $filler->next();                    $filler->setFrameAt($addr, 0x02 | (($bit & $code) != 0));                    $bit = $bit >> 1;                }            }                        QRtools::markTime('after_filler');                        unset($raw);                        // remainder bits            $j = QRspec::getRemainder($version);            for($i=0; $i<$j; $i++) {                $addr = $filler->next();                $filler->setFrameAt($addr, 0x02);            }                        $frame = $filler->frame;            unset($filler);                                    // masking            $maskObj = new QRmask();            if($mask < 0) {                            if (QR_FIND_BEST_MASK) {                    $masked = $maskObj->mask($width, $frame, $input->getErrorCorrectionLevel());                } else {                    $masked = $maskObj->makeMask($width, $frame, (intval(QR_DEFAULT_MASK) % 8), $input->getErrorCorrectionLevel());                }            } else {                $masked = $maskObj->makeMask($width, $frame, $mask, $input->getErrorCorrectionLevel());            }                        if($masked == NULL) {                return NULL;            }                        QRtools::markTime('after_mask');                        $this->version = $version;            $this->width = $width;            $this->data = $masked;                        return $this;        }            //----------------------------------------------------------------------        public function encodeInput(QRinput $input)        {            return $this->encodeMask($input, -1);        }                //----------------------------------------------------------------------        public function encodeString8bit($string, $version, $level)        {            if(string == NULL) {                throw new Exception('empty string!');                return NULL;            }            $input = new QRinput($version, $level);            if($input == NULL) return NULL;            $ret = $input->append($input, QR_MODE_8, strlen($string), str_split($string));            if($ret < 0) {                unset($input);                return NULL;            }            return $this->encodeInput($input);        }        //----------------------------------------------------------------------        public function encodeString($string, $version, $level, $hint, $casesensitive)        {            if($hint != QR_MODE_8 && $hint != QR_MODE_KANJI) {                throw new Exception('bad hint');                return NULL;            }            $input = new QRinput($version, $level);            if($input == NULL) return NULL;            $ret = QRsplit::splitStringToQRinput($string, $input, $hint, $casesensitive);            if($ret < 0) {                return NULL;            }            return $this->encodeInput($input);        }                //----------------------------------------------------------------------        public static function png($text, $outfile = false, $level = QR_ECLEVEL_L, $size = 3, $margin = 4, $saveandprint=false)         {            $enc = QRencode::factory($level, $size, $margin);            return $enc->encodePNG($text, $outfile, $saveandprint=false);        }        //----------------------------------------------------------------------        public static function text($text, $outfile = false, $level = QR_ECLEVEL_L, $size = 3, $margin = 4)         {            $enc = QRencode::factory($level, $size, $margin);            return $enc->encode($text, $outfile);        }        //----------------------------------------------------------------------        public static function raw($text, $outfile = false, $level = QR_ECLEVEL_L, $size = 3, $margin = 4)         {            $enc = QRencode::factory($level, $size, $margin);            return $enc->encodeRAW($text, $outfile);        }    }        //##########################################################################        class FrameFiller {            public $width;        public $frame;        public $x;        public $y;        public $dir;        public $bit;                //----------------------------------------------------------------------        public function __construct($width, &$frame)        {            $this->width = $width;            $this->frame = $frame;            $this->x = $width - 1;            $this->y = $width - 1;            $this->dir = -1;            $this->bit = -1;        }                //----------------------------------------------------------------------        public function setFrameAt($at, $val)        {            $this->frame[$at['y']][$at['x']] = chr($val);        }                //----------------------------------------------------------------------        public function getFrameAt($at)        {            return ord($this->frame[$at['y']][$at['x']]);        }                //----------------------------------------------------------------------        public function next()        {            do {                            if($this->bit == -1) {                    $this->bit = 0;                    return array('x'=>$this->x, 'y'=>$this->y);                }                $x = $this->x;                $y = $this->y;                $w = $this->width;                if($this->bit == 0) {                    $x--;                    $this->bit++;                } else {                    $x++;                    $y += $this->dir;                    $this->bit--;                }                if($this->dir < 0) {                    if($y < 0) {                        $y = 0;                        $x -= 2;                        $this->dir = 1;                        if($x == 6) {                            $x--;                            $y = 9;                        }                    }                } else {                    if($y == $w) {                        $y = $w - 1;                        $x -= 2;                        $this->dir = -1;                        if($x == 6) {                            $x--;                            $y -= 8;                        }                    }                }                if($x < 0 || $y < 0) return null;                $this->x = $x;                $this->y = $y;            } while(ord($this->frame[$y][$x]) & 0x80);                                    return array('x'=>$x, 'y'=>$y);        }            } ;        //##########################################################################            class QRencode {            public $casesensitive = true;        public $eightbit = false;                public $version = 0;        public $size = 3;        public $margin = 4;                public $structured = 0; // not supported yet                public $level = QR_ECLEVEL_L;        public $hint = QR_MODE_8;                //----------------------------------------------------------------------        public static function factory($level = QR_ECLEVEL_L, $size = 3, $margin = 4)        {            $enc = new QRencode();            $enc->size = $size;            $enc->margin = $margin;                        switch ($level.'') {                case '0':                case '1':                case '2':                case '3':                        $enc->level = $level;                    break;                case 'l':                case 'L':                        $enc->level = QR_ECLEVEL_L;                    break;                case 'm':                case 'M':                        $enc->level = QR_ECLEVEL_M;                    break;                case 'q':                case 'Q':                        $enc->level = QR_ECLEVEL_Q;                    break;                case 'h':                case 'H':                        $enc->level = QR_ECLEVEL_H;                    break;            }                        return $enc;        }                //----------------------------------------------------------------------        public function encodeRAW($intext, $outfile = false)         {            $code = new QRcode();            if($this->eightbit) {                $code->encodeString8bit($intext, $this->version, $this->level);            } else {                $code->encodeString($intext, $this->version, $this->level, $this->hint, $this->casesensitive);            }                        return $code->data;        }        //----------------------------------------------------------------------        public function encode($intext, $outfile = false)         {            $code = new QRcode();            if($this->eightbit) {                $code->encodeString8bit($intext, $this->version, $this->level);            } else {                $code->encodeString($intext, $this->version, $this->level, $this->hint, $this->casesensitive);            }                        QRtools::markTime('after_encode');                        if ($outfile!== false) {                file_put_contents($outfile, join("\n", QRtools::binarize($code->data)));            } else {                return QRtools::binarize($code->data);            }        }                //----------------------------------------------------------------------        public function encodePNG($intext, $outfile = false,$saveandprint=false)         {            try {                            ob_start();                $tab = $this->encode($intext);                $err = ob_get_contents();                ob_end_clean();                                if ($err != '')                    QRtools::log($outfile, $err);                                $maxSize = (int)(QR_PNG_MAXIMUM_SIZE / (count($tab)+2*$this->margin));                                QRimage::png($tab, $outfile, min(max(1, $this->size), $maxSize), $this->margin,$saveandprint);                        } catch (Exception $e) {                            QRtools::log($outfile, $e->getMessage());                        }        }    }
 |