diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 00000000..0d38a9a2 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,10 @@ +# ignore directories in the git-generated distributed .zip archive +/.github export-ignore +/tests export-ignore +.gitattributes export-ignore +.gitignore export-ignore +generate-examples.php export-ignore +generate-verified-files.php export-ignore +phpunit.xml export-ignore +examples.md export-ignore +phpstan.neon.dist export-ignore diff --git a/.github/workflows/phpstan.yml b/.github/workflows/phpstan.yml new file mode 100644 index 00000000..20fe4d11 --- /dev/null +++ b/.github/workflows/phpstan.yml @@ -0,0 +1,27 @@ +name: Static analysis (phpstan) + +on: + push: + branches: + - main + pull_request: + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Setup PHP + uses: shivammathur/setup-php@v2 + with: + php-version: 8.3 + extensions: mbstring, gd, bcmath, imagick + + - name: Install dependencies + run: composer install --prefer-dist --no-progress --no-interaction + + - name: Run analysis + run: vendor/bin/phpstan --error-format=github --no-progress diff --git a/.github/workflows/phpunit.yml b/.github/workflows/phpunit.yml new file mode 100644 index 00000000..cc4a0d57 --- /dev/null +++ b/.github/workflows/phpunit.yml @@ -0,0 +1,34 @@ +name: Unit tests (phpunit) + +on: + push: + branches: + - main + pull_request: + +jobs: + build: + runs-on: ubuntu-latest + + strategy: + matrix: + php-versions: ['8.2', '8.3', '8.4'] + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Setup PHP + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php-versions }} + extensions: mbstring, gd, bcmath, imagick + + - name: Validate composer.json + run: composer validate + + - name: Install dependencies + run: composer install --prefer-dist --no-progress --no-interaction + + - name: Run test suite + run: composer run-script test diff --git a/.gitignore b/.gitignore index 2e3beb6f..71981a6c 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ -.idea -.DS_Store vendor -composer.lock \ No newline at end of file +composer.lock +composer.phar +.phpunit.result.cache +.DS_Store diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 93e4bfa8..00000000 --- a/.travis.yml +++ /dev/null @@ -1,12 +0,0 @@ -language: php - -php: - - 5.6 - - 7.0 - - hhvm - -sudo: false - -install: composer install --no-interaction --prefer-source - -script: vendor/bin/phpunit \ No newline at end of file diff --git a/Readme.md b/Readme.md index 0dccb4df..99652ca5 100644 --- a/Readme.md +++ b/Readme.md @@ -1,9 +1,15 @@ -# PHP Barcode Generator [![Build Status](https://travis-ci.org/picqer/php-barcode-generator.svg?branch=master)](https://travis-ci.org/picqer/php-barcode-generator) [![Total Downloads](https://poser.pugx.org/picqer/php-barcode-generator/downloads)](https://packagist.org/packages/picqer/php-barcode-generator) -This is an easy to use, non-bloated, framework independent, barcode generator in PHP. +# PHP Barcode Generator +Build Status +Total Downloads +Latest Stable Version + +This is an easy to use, non-bloated, framework independent, barcode generator in PHP. It uses zero(!) composer dependencies and is only a handful of files. Probably the reason that this is the most downloaded barcode generator for PHP on Packagist. ;) It creates SVG, PNG, JPG and HTML images, from the most used 1D barcode standards. -*The codebase is largely from the [TCPDF barcode generator](https://github.com/tecnickcom/TCPDF) by Nicola Asuni. This code is therefor licensed under LGPLv3. It is still a bit of a mess, bit I will clean it in the future. I do not expect the interface of this class will change during the clean ups.* +## No support for... +- No support for any **2D** barcodes, like QR codes. +- We only generate the 'bars' part of a barcode, without text below the barcode. If you want text of the code below the barcode, you could add it later to the output of this package. ## Installation Install through [composer](https://getcomposer.org/doc/00-intro.md): @@ -12,32 +18,115 @@ Install through [composer](https://getcomposer.org/doc/00-intro.md): composer require picqer/php-barcode-generator ``` -If you want to generate PNG or JPG images, you need the GD library or Imagick installed on your system as well. +If you want to generate PNG or JPG images, you need the GD library or Imagick installed on your system as well. For SVG or HTML renders, there are no dependencies. ## Usage -Initiate the barcode generator for the output you want, then call the ->getBarcode() routine as many times as you want. +You want a barcode for a specific "type" (for example Code 128 or UPC) in a specific image format (for example PNG or SVG). + +- First, encode the string you want the barcode of into a `Barcode` object with one of the barcode types. +- Then, use one of the renderers to render the image of the bars in the `Barcode` object. + +> The "type" is a standard that defines which characters you can encode and which bars represent which character. The most used types are [code 128](https://en.wikipedia.org/wiki/Code_128) and [EAN/UPC](https://en.wikipedia.org/wiki/International_Article_Number). Not all characters can be encoded into each barcode type, and not all barcode scanners can read all types. ```php -$generator = new Picqer\Barcode\BarcodeGeneratorHTML(); -echo $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); +getBarcode('081231723897'); + +// Output the barcode as HTML in the browser with a HTML Renderer +$renderer = new Picqer\Barcode\Renderers\HtmlRenderer(); +echo $renderer->render($barcode); ``` -The ->getBarcode() routine accepts the following: -- $code Data for the barcode -- $type Type of barcode, use the constants defined in the class -- $widthFactor Width is based on the length of the data, with this factor you can make the barcode bars wider than default -- $totalHeight The total height of the barcode -- $color Hex code of the foreground color +Will result in this beauty:
+![Barcode 081231723897 as Code 128](tests/verified-files/081231723897-ean13.svg) -## Image types +Each renderer has their own options. For example, you can set the height, width and color of a PNG: ```php -$generatorSVG = new Picqer\Barcode\BarcodeGeneratorSVG(); -$generatorPNG = new Picqer\Barcode\BarcodeGeneratorPNG(); -$generatorJPG = new Picqer\Barcode\BarcodeGeneratorJPG(); -$generatorHTML = new Picqer\Barcode\BarcodeGeneratorHTML(); +getBarcode('081231723897'); +$renderer = new Picqer\Barcode\Renderers\PngRenderer(); +$renderer->setForegroundColor($colorRed); + +// Save PNG to the filesystem, with widthFactor 3 (width of the barcode x 3) and height of 50 pixels +file_put_contents('barcode.png', $renderer->render($barcode, $barcode->getWidth() * 3, 50)); ``` -## Accepted types +## Image renderers +Available image renderers: SVG, PNG, JPG and HTML. + +They all conform to the RendererInterface and have the same `render()` method. Some renderers have extra options as well, via set*() methods. + +### Widths +The render() method needs the Barcode object, the width and height. **For JPG/PNG images**, you only get a valid barcode if you give a width that is a factor of the width of the Barcode object. That is why the examples show `$barcode->getWidth() * 2` to make the image 2 times wider in pixels then the width of the barcode data. You *can* give an arbitrary number as width and the image will be scaled as best as possible, but without anti-aliasing, it will not be perfectly valid. + +HTML and SVG renderers can handle any width and height, even floats. + +Here are all the options for each renderer: + +### SVG +A vector based SVG image. Gives the best quality to print. +```php +$renderer = new Picqer\Barcode\Renderers\SvgRenderer(); +$renderer->setForegroundColor([255, 0, 0]); // Give a color red for the bars, default is black. Give it as 3 times 0-255 values for red, green and blue. +$renderer->setBackgroundColor([0, 0, 255]); // Give a color blue for the background, default is transparent. Give it as 3 times 0-255 values for red, green and blue. +$renderer->setSvgType($renderer::TYPE_SVG_INLINE); // Changes the output to be used inline inside HTML documents, instead of a standalone SVG image (default) +$renderer->setSvgType($renderer::TYPE_SVG_STANDALONE); // If you want to force the default, create a stand alone SVG image + +$renderer->render($barcode, 450.20, 75); // Width and height support floats +```` + +### PNG + JPG +All options for PNG and JPG are the same. +```php +$renderer = new Picqer\Barcode\Renderers\PngRenderer(); +$renderer->setForegroundColor([255, 0, 0]); // Give a color for the bars, default is black. Give it as 3 times 0-255 values for red, green and blue. +$renderer->setBackgroundColor([0, 255, 255]); // Give a color for the background, default is transparent (in PNG) or white (in JPG). Give it as 3 times 0-255 values for red, green and blue. +$renderer->useGd(); // If you have Imagick and GD installed, but want to use GD +$renderer->useImagick(); // If you have Imagick and GD installed, but want to use Imagick + +$renderer->render($barcode, 5, 40); // Width factor (how many pixel wide every bar is), and the height in pixels +```` + +### HTML +Gives HTML to use inline in a full HTML document. +```php +$renderer = new Picqer\Barcode\Renderers\HtmlRenderer(); +$renderer->setForegroundColor([255, 0, 0]); // Give a color red for the bars, default is black. Give it as 3 times 0-255 values for red, green and blue. +$renderer->setBackgroundColor([0, 0, 255]); // Give a color blue for the background, default is transparent. Give it as 3 times 0-255 values for red, green and blue. + +$renderer->render($barcode, 450.20, 75); // Width and height support floats +```` + +### Dynamic HTML +Give HTML here the barcode is using the full width and height, to put inside a container/div that has a fixed size. +```php +$renderer = new Picqer\Barcode\Renderers\DynamicHtmlRenderer(); +$renderer->setForegroundColor([255, 0, 0]); // Give a color red for the bars, default is black. Give it as 3 times 0-255 values for red, green and blue. +$renderer->setBackgroundColor([0, 0, 255]); // Give a color blue for the background, default is transparent. Give it as 3 times 0-255 values for red, green and blue. + +$renderer->render($barcode); +```` + +You can put the rendered HTML inside a div like this: +```html +
+``` + +## Accepted barcode types +These barcode types are supported. All types support different character sets and some have mandatory lengths. Please see wikipedia for supported chars and lengths per type. + +You can find all supported types in the [src/Types](src/Types) folder. + +Most used types are TYPE_CODE_128 and TYPE_CODE_39. Because of the best scanner support, variable length and most chars supported. + +- TYPE_CODE_32 (italian pharmaceutical code 'MINSAN') - TYPE_CODE_39 - TYPE_CODE_39_CHECKSUM - TYPE_CODE_39E @@ -55,6 +144,7 @@ $generatorHTML = new Picqer\Barcode\BarcodeGeneratorHTML(); - TYPE_EAN_5 - TYPE_EAN_8 - TYPE_EAN_13 +- TYPE_ITF14 (Also known as GTIN-14) - TYPE_UPC_A - TYPE_UPC_E - TYPE_MSI @@ -69,10 +159,127 @@ $generatorHTML = new Picqer\Barcode\BarcodeGeneratorHTML(); - TYPE_PHARMA_CODE - TYPE_PHARMA_CODE_TWO_TRACKS +[See example images for all supported barcode types](examples.md) + +## A note about PNG and JPG images +If you want to use PNG or JPG images, you need to install [Imagick](https://www.php.net/manual/en/intro.imagick.php) or the [GD library](https://www.php.net/manual/en/intro.image.php). This package will use Imagick if that is installed, or fall back to GD. If you have both installed, but you want a specific method, you can use `$renderer->useGd()` or `$renderer->useImagick()` to force your preference. + ## Examples -Embedded PNG image in HTML: +### Embedded PNG image in HTML +```php +$barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); +$renderer = new Picqer\Barcode\Renderers\PngRenderer(); +echo ''; +``` + +### Save JPG barcode to disk ```php -$generator = new \Picqer\Barcode\BarcodeGeneratorPNG(); +$barcode = (new Picqer\Barcode\Types\TypeCodabar())->getBarcode('081231723897'); +$renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + +file_put_contents('barcode.jpg', $renderer->render($barcode, $barcode->getWidth() * 2)); +``` + +### Oneliner SVG output to disk +```php +file_put_contents('barcode.svg', (new Picqer\Barcode\Renderers\SvgRenderer())->render((new Picqer\Barcode\Types\TypeKix())->getBarcode('6825ME601'))); +``` + +## Upgrading to v3 +There is no need to change anything when upgrading from v2 to v3. Above you find the new preferred way of using this library since v3. But the old style still works. + +To give the renderers the same interface, setting colors is now always with an array of RGB colors. If you use the old BarcodeGenerator* classes and use colors with names ('red') or hex codes (#3399ef), these will be converted using the ColorHelper. All hexcodes are supported, but for names of colors only the basic colors are supported. + +If you want to convert to the new style, here is an example: +```php +// Old style +$generator = new Picqer\Barcode\BarcodeGeneratorSVG(); +echo $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); + +// New style +$barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); +$renderer = new Picqer\Barcode\Renderers\SvgRenderer(); +echo $renderer->render($barcode); +``` + +The width in the renderer is now the width of the end result, instead of the widthFactor. If you want to keep dynamic widths, you can get the width of the encoded Barcode and multiply it by the widthFactor to get the same result as before. See here an example for a widthFactor of 2: +```php +// Old style +$generator = new Picqer\Barcode\BarcodeGeneratorSVG(); +echo $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 2. 30); + +// New style +$barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); +$renderer = new Picqer\Barcode\Renderers\SvgRenderer(); +echo $renderer->render($barcode, $barcode->getWidth() * 2, 30); +``` + +--- + +## Previous style generators +In version 3 the barcode type encoders and image renderers are completely separate. This makes building your own renderer way easier. The old way was using "generators". Below are the old examples of these generators, which still works in v3 as well. + +### Usage +Initiate the barcode generator for the output you want, then call the ->getBarcode() routine as many times as you want. + +```php +getBarcode('081231723897', $generator::TYPE_CODE_128); +``` + +Will result in this beauty:
+![Barcode 081231723897 as Code 128](tests/verified-files/081231723897-ean13.svg) + +The `getBarcode()` method accepts the following parameters: +- `$barcode` String needed to encode in the barcode +- `$type` Type of barcode, use the constants defined in the class +- `$widthFactor` Width is based on the length of the data, with this factor you can make the barcode bars wider than default +- `$height` The total height of the barcode in pixels +- `$foregroundColor` Hex code as string, or array of RGB, of the colors of the bars (the foreground color) + +Example of usage of all parameters: + +```php +getBarcode('081231723897', $generator::TYPE_CODE_128, 3, 50, $redColor)); +``` + +### Image types +```php +$generatorSVG = new Picqer\Barcode\BarcodeGeneratorSVG(); // Vector based SVG +$generatorPNG = new Picqer\Barcode\BarcodeGeneratorPNG(); // Pixel based PNG +$generatorJPG = new Picqer\Barcode\BarcodeGeneratorJPG(); // Pixel based JPG +$generatorHTML = new Picqer\Barcode\BarcodeGeneratorHTML(); // Pixel based HTML +$generatorHTML = new Picqer\Barcode\BarcodeGeneratorDynamicHTML(); // Vector based HTML +``` + +#### Embedded PNG image in HTML +```php +$generator = new Picqer\Barcode\BarcodeGeneratorPNG(); echo ''; ``` + +#### Save JPG barcode to disk +```php +$generator = new Picqer\Barcode\BarcodeGeneratorJPG(); +file_put_contents('barcode.jpg', $generator->getBarcode('081231723897', $generator::TYPE_CODABAR)); +``` + +#### Oneliner SVG output to disk +```php +file_put_contents('barcode.svg', (new Picqer\Barcode\BarcodeGeneratorSVG())->getBarcode('6825ME601', Picqer\Barcode\BarcodeGeneratorSVG::TYPE_KIX)); +``` + +--- +*The codebase is based on the [TCPDF barcode generator](https://github.com/tecnickcom/TCPDF) by Nicola Asuni. This code is therefor licensed under LGPLv3.* diff --git a/composer.json b/composer.json index 22f4c42d..a67a4e88 100644 --- a/composer.json +++ b/composer.json @@ -3,28 +3,39 @@ "type": "library", "description": "An easy to use, non-bloated, barcode generator in PHP. Creates SVG, PNG, JPG and HTML images from the most used 1D barcode standards.", "keywords": [ "php", "barcode", "barcode generator", "EAN", "EAN13", "UPC", "Code39", "Code128", "Code93", "Standard 2 of 5", "MSI", "POSTNET", "KIX", "KIXCODE", "CODABAR", "PHARMA", "Code11", "SVG", "PNG", "HTML", "JPG", "JPEG" ], - "homepage": "http://github.com/picqer/php-barcode-generator", - "license": "LGPLv3", + "homepage": "https://github.com/picqer/php-barcode-generator", + "license": "LGPL-3.0-or-later", "authors": [ + { + "name": "Casper Bakker", + "email": "info@picqer.com", + "homepage": "https://picqer.com/" + }, { "name": "Nicola Asuni", "email": "info@tecnick.com", "homepage": "http://nicolaasuni.tecnick.com/" - }, - { - "name": "Casper Bakker", - "email": "info@picqer.com" } ], "require": { - "php": ">=5.4.0" + "php": "^8.2", + "ext-mbstring": "*" }, "require-dev": { - "phpunit/phpunit": "^5.3" + "phpunit/phpunit": "^9.5", + "phpstan/phpstan": "^1.10" + }, + "suggest": { + "ext-bcmath": "Barcode IMB (Intelligent Mail Barcode) needs bcmath extension", + "ext-gd": "For JPG and PNG generators, GD or Imagick is required", + "ext-imagick": "For JPG and PNG generators, GD or Imagick is required" }, "autoload": { "psr-4": { "Picqer\\Barcode\\": "src" } + }, + "scripts": { + "test": "vendor/bin/phpunit" } } diff --git a/examples.md b/examples.md new file mode 100644 index 00000000..689c1928 --- /dev/null +++ b/examples.md @@ -0,0 +1,149 @@ +# Examples of supported barcodes + +These are examples of supported barcodes with this library. + +All types can be found in the src/Types directory. + + + +### TypeCode39 + +![Barcode 1234567890ABC as Picqer\Barcode\Types\TypeCode39](tests/verified-files/TypeCode39-1234567890ABC.svg) + +### TypeCode39Checksum + +![Barcode 1234567890ABC as Picqer\Barcode\Types\TypeCode39Checksum](tests/verified-files/TypeCode39Checksum-1234567890ABC.svg) + +### TypeCode39Extended + +![Barcode 1234567890abcABC as Picqer\Barcode\Types\TypeCode39Extended](tests/verified-files/TypeCode39Extended-1234567890abcABC.svg) + +### TypeCode39ExtendedChecksum + +![Barcode 1234567890abcABC as Picqer\Barcode\Types\TypeCode39ExtendedChecksum](tests/verified-files/TypeCode39ExtendedChecksum-1234567890abcABC.svg) + +### TypeCode93 + +![Barcode 1234567890abcABC as Picqer\Barcode\Types\TypeCode93](tests/verified-files/TypeCode93-1234567890abcABC.svg) + +### TypeStandard2of5 + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeStandard2of5](tests/verified-files/TypeStandard2of5-1234567890.svg) + +### TypeStandard2of5Checksum + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeStandard2of5Checksum](tests/verified-files/TypeStandard2of5Checksum-1234567890.svg) + +### TypeInterleaved25 + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeInterleaved25](tests/verified-files/TypeInterleaved25-1234567890.svg) + +### TypeInterleaved25Checksum + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeInterleaved25Checksum](tests/verified-files/TypeInterleaved25Checksum-1234567890.svg) + +### TypeEan13 + +![Barcode 081231723897 as Picqer\Barcode\Types\TypeEan13](tests/verified-files/TypeEan13-081231723897.svg) + +![Barcode 0049000004632 as Picqer\Barcode\Types\TypeEan13](tests/verified-files/TypeEan13-0049000004632.svg) + +![Barcode 004900000463 as Picqer\Barcode\Types\TypeEan13](tests/verified-files/TypeEan13-004900000463.svg) + +### TypeITF14 + +![Barcode 00012345600012 as Picqer\Barcode\Types\TypeITF14](tests/verified-files/TypeITF14-00012345600012.svg) + +![Barcode 05400141288766 as Picqer\Barcode\Types\TypeITF14](tests/verified-files/TypeITF14-05400141288766.svg) + +### TypeCode128 + +![Barcode 081231723897 as Picqer\Barcode\Types\TypeCode128](tests/verified-files/TypeCode128-081231723897.svg) + +![Barcode 1234567890abcABC-283*33 as Picqer\Barcode\Types\TypeCode128](tests/verified-files/TypeCode128-1234567890abcABC-283-33.svg) + +### TypeCode128A + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeCode128A](tests/verified-files/TypeCode128A-1234567890.svg) + +### TypeCode128B + +![Barcode 081231723897 as Picqer\Barcode\Types\TypeCode128B](tests/verified-files/TypeCode128B-081231723897.svg) + +![Barcode 1234567890abcABC-283*33 as Picqer\Barcode\Types\TypeCode128B](tests/verified-files/TypeCode128B-1234567890abcABC-283-33.svg) + +### TypeUpcExtension2 + +![Barcode 22 as Picqer\Barcode\Types\TypeUpcExtension2](tests/verified-files/TypeUpcExtension2-22.svg) + +### TypeUpcExtension5 + +![Barcode 1234567890abcABC-283*33 as Picqer\Barcode\Types\TypeUpcExtension5](tests/verified-files/TypeUpcExtension5-1234567890abcABC-283-33.svg) + +### TypeEan8 + +![Barcode 1234568 as Picqer\Barcode\Types\TypeEan8](tests/verified-files/TypeEan8-1234568.svg) + +### TypeUpcA + +![Barcode 123456789 as Picqer\Barcode\Types\TypeUpcA](tests/verified-files/TypeUpcA-123456789.svg) + +### TypeUpcE + +![Barcode 123456789 as Picqer\Barcode\Types\TypeUpcE](tests/verified-files/TypeUpcE-123456789.svg) + +### TypeMsi + +![Barcode 123456789 as Picqer\Barcode\Types\TypeMsi](tests/verified-files/TypeMsi-123456789.svg) + +### TypeMsiChecksum + +![Barcode 123456789 as Picqer\Barcode\Types\TypeMsiChecksum](tests/verified-files/TypeMsiChecksum-123456789.svg) + +### TypePostnet + +![Barcode 123456789 as Picqer\Barcode\Types\TypePostnet](tests/verified-files/TypePostnet-123456789.svg) + +### TypePlanet + +![Barcode 123456789 as Picqer\Barcode\Types\TypePlanet](tests/verified-files/TypePlanet-123456789.svg) + +### TypeRms4cc + +![Barcode 123456789 as Picqer\Barcode\Types\TypeRms4cc](tests/verified-files/TypeRms4cc-123456789.svg) + +### TypeKix + +![Barcode 123456789 as Picqer\Barcode\Types\TypeKix](tests/verified-files/TypeKix-123456789.svg) + +### TypeIntelligentMailBarcode + +![Barcode 123456789 as Picqer\Barcode\Types\TypeIntelligentMailBarcode](tests/verified-files/TypeIntelligentMailBarcode-123456789.svg) + +### TypeCodabar + +![Barcode 123456789 as Picqer\Barcode\Types\TypeCodabar](tests/verified-files/TypeCodabar-123456789.svg) + +### TypeCode11 + +![Barcode 123456789 as Picqer\Barcode\Types\TypeCode11](tests/verified-files/TypeCode11-123456789.svg) + +### TypePharmacode + +![Barcode 123456789 as Picqer\Barcode\Types\TypePharmacode](tests/verified-files/TypePharmacode-123456789.svg) + +### TypePharmacodeTwoCode + +![Barcode 123456789 as Picqer\Barcode\Types\TypePharmacodeTwoCode](tests/verified-files/TypePharmacodeTwoCode-123456789.svg) + +### TypeTelepen + +![Barcode 1234567890ASCD as Picqer\Barcode\Types\TypeTelepen](tests/verified-files/TypeTelepen-1234567890ASCD.svg) + +### TypeTelepenNumeric + +![Barcode 1234567890 as Picqer\Barcode\Types\TypeTelepenNumeric](tests/verified-files/TypeTelepenNumeric-1234567890.svg) + + + +*This file is generated by generate-examples.php* \ No newline at end of file diff --git a/generate-examples.php b/generate-examples.php new file mode 100644 index 00000000..7be0f009 --- /dev/null +++ b/generate-examples.php @@ -0,0 +1,28 @@ +getBarcode('081231723897', $generatorSVG::TYPE_EAN_13)); +$svgRenderer = new Picqer\Barcode\Renderers\SvgRenderer(); +$htmlRenderer = new Picqer\Barcode\Renderers\HtmlRenderer(); +$dynamicHtmlRenderer = new Picqer\Barcode\Renderers\DynamicHtmlRenderer(); -$generatorHTML = new Picqer\Barcode\BarcodeGeneratorHTML(); -file_put_contents('tests/verified-files/081231723897-code128.html', $generatorHTML->getBarcode('081231723897', $generatorHTML::TYPE_CODE_128)); +$typeEncoderEan13 = new \Picqer\Barcode\Types\TypeEan13(); +$typeEncoderCode128 = new \Picqer\Barcode\Types\TypeCode128(); +$typeEncoderIMB = new \Picqer\Barcode\Types\TypeIntelligentMailBarcode(); -$generatorSVG = new Picqer\Barcode\BarcodeGeneratorSVG(); -file_put_contents('tests/verified-files/0049000004632-ean13.svg', $generatorSVG->getBarcode('0049000004632', $generatorSVG::TYPE_EAN_13)); +$barcode = $typeEncoderEan13->getBarcode('081231723897'); +file_put_contents('tests/verified-files/081231723897-ean13.svg', $svgRenderer->render($barcode, $barcode->getWidth() * 2)); +file_put_contents('tests/verified-files/081231723897-ean13-fractional-width.svg', $svgRenderer->render($barcode, $barcode->getWidth() * 0.25, 25.75)); + +$svgRendererRed = new Picqer\Barcode\Renderers\SvgRenderer(); +$svgRendererRed->setBackgroundColor([255, 0, 0]); +file_put_contents('tests/verified-files/081231723897-ean13-red-background.svg', $svgRendererRed->render($barcode, $barcode->getWidth() * 2)); + +$barcode = $typeEncoderCode128->getBarcode('081231723897'); +file_put_contents('tests/verified-files/081231723897-code128.html', $htmlRenderer->render($barcode, $barcode->getWidth() * 2)); +$htmlRendererRed = new Picqer\Barcode\Renderers\HtmlRenderer(); +$htmlRendererRed->setBackgroundColor([255, 0, 0]); +file_put_contents('tests/verified-files/081231723897-code128-red-background.html', $htmlRendererRed->render($barcode, $barcode->getWidth() * 2)); + +$barcode = $typeEncoderIMB->getBarcode('12345678903'); +file_put_contents('tests/verified-files/12345678903-imb.html', $htmlRenderer->render($barcode, $barcode->getWidth() * 2)); + +$barcode = $typeEncoderCode128->getBarcode('081231723897'); +file_put_contents('tests/verified-files/081231723897-dynamic-code128.html', $dynamicHtmlRenderer->render($barcode)); + +$barcode = $typeEncoderIMB->getBarcode('12345678903'); +file_put_contents('tests/verified-files/12345678903-dynamic-imb.html', $dynamicHtmlRenderer->render($barcode)); + +$barcode = $typeEncoderEan13->getBarcode('0049000004632'); +file_put_contents('tests/verified-files/0049000004632-ean13.svg', $svgRenderer->render($barcode, $barcode->getWidth() * 2)); + + +// New style of verified files, defined in VerifiedBarcodeTest.php +require(__DIR__ . '/tests/VerifiedBarcodeTest.php'); +$verifiedFiles = VerifiedBarcodeTest::$supportedBarcodes; + +foreach ($verifiedFiles as $verifiedFile) { + foreach ($verifiedFile['barcodes'] as $barcodeText) { + $barcode = (new $verifiedFile['type']())->getBarcode($barcodeText); + $result = $svgRenderer->render($barcode, $barcode->getWidth() * 2); + + file_put_contents('tests/verified-files/' . Picqer\Barcode\Helpers\StringHelpers::getSafeFilenameFrom($verifiedFile['type'] . '-' . $barcodeText) . '.svg', $result); + } +} diff --git a/phpstan.neon.dist b/phpstan.neon.dist new file mode 100644 index 00000000..0a16ff99 --- /dev/null +++ b/phpstan.neon.dist @@ -0,0 +1,4 @@ +parameters: + paths: + - src + level: 5 diff --git a/phpunit.xml b/phpunit.xml index 0d549dc3..9f49e2cf 100644 --- a/phpunit.xml +++ b/phpunit.xml @@ -1,24 +1,13 @@ - - - - ./tests/ - - - - - ./src - - - \ No newline at end of file + + + + ./src + + + + + ./tests/ + + + diff --git a/src/Barcode.php b/src/Barcode.php new file mode 100644 index 00000000..5b210b33 --- /dev/null +++ b/src/Barcode.php @@ -0,0 +1,44 @@ +barcode = $barcode; + } + + // Add a bar to the barcode, either a bar or a space, at the right side of the barcode + public function addBar(BarcodeBar $bar): void + { + $this->bars[] = $bar; + $this->width += $bar->getWidth(); + $this->height = max($this->height, $bar->getHeight()); + } + + public function getBarcode(): string + { + return $this->barcode; + } + + public function getWidth(): int + { + return $this->width; + } + + public function getHeight(): int + { + return $this->height; + } + + public function getBars(): array + { + return $this->bars; + } +} diff --git a/src/BarcodeBar.php b/src/BarcodeBar.php new file mode 100644 index 00000000..28bd595e --- /dev/null +++ b/src/BarcodeBar.php @@ -0,0 +1,43 @@ +width = $width; + $this->height = $height; + $this->positionVertical = $positionVertical; + $this->type = $drawBar ? self::TYPE_BAR : self::TYPE_SPACING; + } + + public function getWidth(): int + { + return $this->width; + } + + public function getHeight(): int + { + return $this->height; + } + + public function getPositionVertical(): int + { + return $this->positionVertical; + } + + public function isBar(): bool + { + return $this->type === self::TYPE_BAR; + } +} diff --git a/src/BarcodeGenerator.php b/src/BarcodeGenerator.php index 33e6fa02..8555b061 100644 --- a/src/BarcodeGenerator.php +++ b/src/BarcodeGenerator.php @@ -9,7 +9,7 @@ // Copyright (C) 2002-2015 Nicola Asuni - Tecnick.com LTD // -// This file is part of TCPDF software library. +// This file was part of TCPDF software library. // // TCPDF is free software: you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License as @@ -29,2833 +29,196 @@ namespace Picqer\Barcode; -use Picqer\Barcode\Exceptions\BarcodeException; -use Picqer\Barcode\Exceptions\InvalidCharacterException; -use Picqer\Barcode\Exceptions\InvalidCheckDigitException; -use Picqer\Barcode\Exceptions\InvalidFormatException; -use Picqer\Barcode\Exceptions\InvalidLengthException; use Picqer\Barcode\Exceptions\UnknownTypeException; +use Picqer\Barcode\Types\TypeCodabar; +use Picqer\Barcode\Types\TypeCode11; +use Picqer\Barcode\Types\TypeCode128; +use Picqer\Barcode\Types\TypeCode128A; +use Picqer\Barcode\Types\TypeCode128B; +use Picqer\Barcode\Types\TypeCode128C; +use Picqer\Barcode\Types\TypeCode32; +use Picqer\Barcode\Types\TypeCode39; +use Picqer\Barcode\Types\TypeCode39Checksum; +use Picqer\Barcode\Types\TypeCode39Extended; +use Picqer\Barcode\Types\TypeCode39ExtendedChecksum; +use Picqer\Barcode\Types\TypeCode93; +use Picqer\Barcode\Types\TypeEan13; +use Picqer\Barcode\Types\TypeEan8; +use Picqer\Barcode\Types\TypeIntelligentMailBarcode; +use Picqer\Barcode\Types\TypeInterface; +use Picqer\Barcode\Types\TypeInterleaved25; +use Picqer\Barcode\Types\TypeInterleaved25Checksum; +use Picqer\Barcode\Types\TypeITF14; +use Picqer\Barcode\Types\TypeKix; +use Picqer\Barcode\Types\TypeMsi; +use Picqer\Barcode\Types\TypeMsiChecksum; +use Picqer\Barcode\Types\TypePharmacode; +use Picqer\Barcode\Types\TypePharmacodeTwoCode; +use Picqer\Barcode\Types\TypePlanet; +use Picqer\Barcode\Types\TypePostnet; +use Picqer\Barcode\Types\TypeRms4cc; +use Picqer\Barcode\Types\TypeStandard2of5; +use Picqer\Barcode\Types\TypeStandard2of5Checksum; +use Picqer\Barcode\Types\TypeTelepen; +use Picqer\Barcode\Types\TypeUpcA; +use Picqer\Barcode\Types\TypeUpcE; +use Picqer\Barcode\Types\TypeUpcExtension2; +use Picqer\Barcode\Types\TypeUpcExtension5; abstract class BarcodeGenerator { + const TYPE_CODE_32 = 'C32'; const TYPE_CODE_39 = 'C39'; const TYPE_CODE_39_CHECKSUM = 'C39+'; - const TYPE_CODE_39E = 'C39E'; - const TYPE_CODE_39E_CHECKSUM = 'C39E+'; + const TYPE_CODE_39E = 'C39E'; // CODE 39 EXTENDED + const TYPE_CODE_39E_CHECKSUM = 'C39E+'; // CODE 39 EXTENDED + CHECKSUM const TYPE_CODE_93 = 'C93'; const TYPE_STANDARD_2_5 = 'S25'; const TYPE_STANDARD_2_5_CHECKSUM = 'S25+'; const TYPE_INTERLEAVED_2_5 = 'I25'; const TYPE_INTERLEAVED_2_5_CHECKSUM = 'I25+'; + const TYPE_ITF_14 = 'ITF14'; const TYPE_CODE_128 = 'C128'; const TYPE_CODE_128_A = 'C128A'; const TYPE_CODE_128_B = 'C128B'; const TYPE_CODE_128_C = 'C128C'; - const TYPE_EAN_2 = 'EAN2'; - const TYPE_EAN_5 = 'EAN5'; + const TYPE_EAN_2 = 'EAN2'; // 2-Digits UPC-Based Extention + const TYPE_EAN_5 = 'EAN5'; // 5-Digits UPC-Based Extention const TYPE_EAN_8 = 'EAN8'; const TYPE_EAN_13 = 'EAN13'; const TYPE_UPC_A = 'UPCA'; const TYPE_UPC_E = 'UPCE'; - const TYPE_MSI = 'MSI'; - const TYPE_MSI_CHECKSUM = 'MSI+'; + const TYPE_MSI = 'MSI'; // MSI (Variation of Plessey code) + const TYPE_MSI_CHECKSUM = 'MSI+'; // MSI + CHECKSUM (modulo 11) const TYPE_POSTNET = 'POSTNET'; const TYPE_PLANET = 'PLANET'; - const TYPE_RMS4CC = 'RMS4CC'; - const TYPE_KIX = 'KIX'; - const TYPE_IMB = 'IMB'; + const TYPE_TELEPEN_ALPHA = 'TELEPENALPHA'; + const TYPE_TELEPEN_NUMERIC = 'TELEPENNUMERIC'; + const TYPE_RMS4CC = 'RMS4CC'; // RMS4CC (Royal Mail 4-state Customer Code) - CBC (Customer Bar Code) + const TYPE_KIX = 'KIX'; // KIX (Klant index - Customer index) + const TYPE_IMB = 'IMB'; // IMB - Intelligent Mail Barcode - Onecode - USPS-B-3200 const TYPE_CODABAR = 'CODABAR'; const TYPE_CODE_11 = 'CODE11'; const TYPE_PHARMA_CODE = 'PHARMA'; const TYPE_PHARMA_CODE_TWO_TRACKS = 'PHARMA2T'; /** - * Get the barcode data - * - * @param string $code code to print - * @param string $type type of barcode - * @return array barcode array - * @public + * @throws UnknownTypeException */ - protected function getBarcodeData($code, $type) + protected function getBarcodeData(string $code, string $type): Barcode { - switch (strtoupper($type)) { - case self::TYPE_CODE_39: { // CODE 39 - ANSI MH10.8M-1983 - USD-3 - 3 of 9. - $arrcode = $this->barcode_code39($code, false, false); - break; - } - case self::TYPE_CODE_39_CHECKSUM: { // CODE 39 with checksum - $arrcode = $this->barcode_code39($code, false, true); - break; - } - case self::TYPE_CODE_39E: { // CODE 39 EXTENDED - $arrcode = $this->barcode_code39($code, true, false); - break; - } - case self::TYPE_CODE_39E_CHECKSUM: { // CODE 39 EXTENDED + CHECKSUM - $arrcode = $this->barcode_code39($code, true, true); - break; - } - case self::TYPE_CODE_93: { // CODE 93 - USS-93 - $arrcode = $this->barcode_code93($code); - break; - } - case self::TYPE_STANDARD_2_5: { // Standard 2 of 5 - $arrcode = $this->barcode_s25($code, false); - break; - } - case self::TYPE_STANDARD_2_5_CHECKSUM: { // Standard 2 of 5 + CHECKSUM - $arrcode = $this->barcode_s25($code, true); - break; - } - case self::TYPE_INTERLEAVED_2_5: { // Interleaved 2 of 5 - $arrcode = $this->barcode_i25($code, false); - break; - } - case self::TYPE_INTERLEAVED_2_5_CHECKSUM: { // Interleaved 2 of 5 + CHECKSUM - $arrcode = $this->barcode_i25($code, true); - break; - } - case self::TYPE_CODE_128: { // CODE 128 - $arrcode = $this->barcode_c128($code, ''); - break; - } - case self::TYPE_CODE_128_A: { // CODE 128 A - $arrcode = $this->barcode_c128($code, 'A'); - break; - } - case self::TYPE_CODE_128_B: { // CODE 128 B - $arrcode = $this->barcode_c128($code, 'B'); - break; - } - case self::TYPE_CODE_128_C: { // CODE 128 C - $arrcode = $this->barcode_c128($code, 'C'); - break; - } - case self::TYPE_EAN_2: { // 2-Digits UPC-Based Extention - $arrcode = $this->barcode_eanext($code, 2); - break; - } - case self::TYPE_EAN_5: { // 5-Digits UPC-Based Extention - $arrcode = $this->barcode_eanext($code, 5); - break; - } - case self::TYPE_EAN_8: { // EAN 8 - $arrcode = $this->barcode_eanupc($code, 8); - break; - } - case self::TYPE_EAN_13: { // EAN 13 - $arrcode = $this->barcode_eanupc($code, 13); - break; - } - case self::TYPE_UPC_A: { // UPC-A - $arrcode = $this->barcode_eanupc($code, 12); - break; - } - case self::TYPE_UPC_E: { // UPC-E - $arrcode = $this->barcode_eanupc($code, 6); - break; - } - case self::TYPE_MSI: { // MSI (Variation of Plessey code) - $arrcode = $this->barcode_msi($code, false); - break; - } - case self::TYPE_MSI_CHECKSUM: { // MSI + CHECKSUM (modulo 11) - $arrcode = $this->barcode_msi($code, true); - break; - } - case self::TYPE_POSTNET: { // POSTNET - $arrcode = $this->barcode_postnet($code, false); - break; - } - case self::TYPE_PLANET: { // PLANET - $arrcode = $this->barcode_postnet($code, true); - break; - } - case self::TYPE_RMS4CC: { // RMS4CC (Royal Mail 4-state Customer Code) - CBC (Customer Bar Code) - $arrcode = $this->barcode_rms4cc($code, false); - break; - } - case self::TYPE_KIX: { // KIX (Klant index - Customer index) - $arrcode = $this->barcode_rms4cc($code, true); - break; - } - case self::TYPE_IMB: { // IMB - Intelligent Mail Barcode - Onecode - USPS-B-3200 - $arrcode = $this->barcode_imb($code); - break; - } - case self::TYPE_CODABAR: { // CODABAR - $arrcode = $this->barcode_codabar($code); - break; - } - case self::TYPE_CODE_11: { // CODE 11 - $arrcode = $this->barcode_code11($code); - break; - } - case self::TYPE_PHARMA_CODE: { // PHARMACODE - $arrcode = $this->barcode_pharmacode($code); - break; - } - case self::TYPE_PHARMA_CODE_TWO_TRACKS: { // PHARMACODE TWO-TRACKS - $arrcode = $this->barcode_pharmacode2t($code); - break; - } - default: { - throw new UnknownTypeException(); - break; - } - } - - if ( ! isset($arrcode['maxWidth'])) { - $arrcode = $this->convertBarcodeArrayToNewStyle($arrcode); - } - - return $arrcode; - } - - /** - * CODE 39 - ANSI MH10.8M-1983 - USD-3 - 3 of 9. - * General-purpose code in very wide use world-wide - * - * @param $code (string) code to represent. - * @param $extended (boolean) if true uses the extended mode. - * @param $checksum (boolean) if true add a checksum to the code. - * @return array barcode representation. - * @protected - */ - protected function barcode_code39($code, $extended = false, $checksum = false) - { - $chr = []; - $chr['0'] = '111331311'; - $chr['1'] = '311311113'; - $chr['2'] = '113311113'; - $chr['3'] = '313311111'; - $chr['4'] = '111331113'; - $chr['5'] = '311331111'; - $chr['6'] = '113331111'; - $chr['7'] = '111311313'; - $chr['8'] = '311311311'; - $chr['9'] = '113311311'; - $chr['A'] = '311113113'; - $chr['B'] = '113113113'; - $chr['C'] = '313113111'; - $chr['D'] = '111133113'; - $chr['E'] = '311133111'; - $chr['F'] = '113133111'; - $chr['G'] = '111113313'; - $chr['H'] = '311113311'; - $chr['I'] = '113113311'; - $chr['J'] = '111133311'; - $chr['K'] = '311111133'; - $chr['L'] = '113111133'; - $chr['M'] = '313111131'; - $chr['N'] = '111131133'; - $chr['O'] = '311131131'; - $chr['P'] = '113131131'; - $chr['Q'] = '111111333'; - $chr['R'] = '311111331'; - $chr['S'] = '113111331'; - $chr['T'] = '111131331'; - $chr['U'] = '331111113'; - $chr['V'] = '133111113'; - $chr['W'] = '333111111'; - $chr['X'] = '131131113'; - $chr['Y'] = '331131111'; - $chr['Z'] = '133131111'; - $chr['-'] = '131111313'; - $chr['.'] = '331111311'; - $chr[' '] = '133111311'; - $chr['$'] = '131313111'; - $chr['/'] = '131311131'; - $chr['+'] = '131113131'; - $chr['%'] = '111313131'; - $chr['*'] = '131131311'; - - $code = strtoupper($code); - - if ($extended) { - // extended mode - $code = $this->encode_code39_ext($code); - } - - if ($checksum) { - // checksum - $code .= $this->checksum_code39($code); - } - - // add start and stop codes - $code = '*' . $code . '*'; - - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $k = 0; - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - $char = $code{$i}; - if ( ! isset($chr[$char])) { - throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); - } - for ($j = 0; $j < 9; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $chr[$char]{$j}; - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - } - // intercharacter gap - $bararray['bcode'][$k] = array('t' => false, 'w' => 1, 'h' => 1, 'p' => 0); - $bararray['maxw'] += 1; - ++$k; - } - - return $bararray; - } - - /** - * Encode a string to be used for CODE 39 Extended mode. - * - * @param string $code code to represent. - * @return bool|string encoded string. - * @protected - */ - protected function encode_code39_ext($code) - { - $encode = array( - chr(0) => '%U', - chr(1) => '$A', - chr(2) => '$B', - chr(3) => '$C', - chr(4) => '$D', - chr(5) => '$E', - chr(6) => '$F', - chr(7) => '$G', - chr(8) => '$H', - chr(9) => '$I', - chr(10) => '$J', - chr(11) => '£K', - chr(12) => '$L', - chr(13) => '$M', - chr(14) => '$N', - chr(15) => '$O', - chr(16) => '$P', - chr(17) => '$Q', - chr(18) => '$R', - chr(19) => '$S', - chr(20) => '$T', - chr(21) => '$U', - chr(22) => '$V', - chr(23) => '$W', - chr(24) => '$X', - chr(25) => '$Y', - chr(26) => '$Z', - chr(27) => '%A', - chr(28) => '%B', - chr(29) => '%C', - chr(30) => '%D', - chr(31) => '%E', - chr(32) => ' ', - chr(33) => '/A', - chr(34) => '/B', - chr(35) => '/C', - chr(36) => '/D', - chr(37) => '/E', - chr(38) => '/F', - chr(39) => '/G', - chr(40) => '/H', - chr(41) => '/I', - chr(42) => '/J', - chr(43) => '/K', - chr(44) => '/L', - chr(45) => '-', - chr(46) => '.', - chr(47) => '/O', - chr(48) => '0', - chr(49) => '1', - chr(50) => '2', - chr(51) => '3', - chr(52) => '4', - chr(53) => '5', - chr(54) => '6', - chr(55) => '7', - chr(56) => '8', - chr(57) => '9', - chr(58) => '/Z', - chr(59) => '%F', - chr(60) => '%G', - chr(61) => '%H', - chr(62) => '%I', - chr(63) => '%J', - chr(64) => '%V', - chr(65) => 'A', - chr(66) => 'B', - chr(67) => 'C', - chr(68) => 'D', - chr(69) => 'E', - chr(70) => 'F', - chr(71) => 'G', - chr(72) => 'H', - chr(73) => 'I', - chr(74) => 'J', - chr(75) => 'K', - chr(76) => 'L', - chr(77) => 'M', - chr(78) => 'N', - chr(79) => 'O', - chr(80) => 'P', - chr(81) => 'Q', - chr(82) => 'R', - chr(83) => 'S', - chr(84) => 'T', - chr(85) => 'U', - chr(86) => 'V', - chr(87) => 'W', - chr(88) => 'X', - chr(89) => 'Y', - chr(90) => 'Z', - chr(91) => '%K', - chr(92) => '%L', - chr(93) => '%M', - chr(94) => '%N', - chr(95) => '%O', - chr(96) => '%W', - chr(97) => '+A', - chr(98) => '+B', - chr(99) => '+C', - chr(100) => '+D', - chr(101) => '+E', - chr(102) => '+F', - chr(103) => '+G', - chr(104) => '+H', - chr(105) => '+I', - chr(106) => '+J', - chr(107) => '+K', - chr(108) => '+L', - chr(109) => '+M', - chr(110) => '+N', - chr(111) => '+O', - chr(112) => '+P', - chr(113) => '+Q', - chr(114) => '+R', - chr(115) => '+S', - chr(116) => '+T', - chr(117) => '+U', - chr(118) => '+V', - chr(119) => '+W', - chr(120) => '+X', - chr(121) => '+Y', - chr(122) => '+Z', - chr(123) => '%P', - chr(124) => '%Q', - chr(125) => '%R', - chr(126) => '%S', - chr(127) => '%T' - ); - $code_ext = ''; - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - if (ord($code{$i}) > 127) { - throw new InvalidCharacterException('Only supports till char 127'); - } - $code_ext .= $encode[$code{$i}]; - } - - return $code_ext; - } + $barcodeDataBuilder = $this->createDataBuilderForType($type); - /** - * Calculate CODE 39 checksum (modulo 43). - * - * @param string $code code to represent. - * @return string char checksum. - * @protected - */ - protected function checksum_code39($code) - { - $chars = array( - '0', - '1', - '2', - '3', - '4', - '5', - '6', - '7', - '8', - '9', - 'A', - 'B', - 'C', - 'D', - 'E', - 'F', - 'G', - 'H', - 'I', - 'J', - 'K', - 'L', - 'M', - 'N', - 'O', - 'P', - 'Q', - 'R', - 'S', - 'T', - 'U', - 'V', - 'W', - 'X', - 'Y', - 'Z', - '-', - '.', - ' ', - '$', - '/', - '+', - '%' - ); - $sum = 0; - $codelength = strlen($code); - for ($i = 0; $i < $codelength; ++$i) { - $k = array_keys($chars, $code{$i}); - $sum += $k[0]; - } - $j = ($sum % 43); - - return $chars[$j]; + return $barcodeDataBuilder->getBarcode($code); } - /** - * CODE 93 - USS-93 - * Compact code similar to Code 39 - * - * @param $code (string) code to represent. - * @return array barcode representation. - * @protected - */ - protected function barcode_code93($code) + protected function createDataBuilderForType(string $type): TypeInterface { - $chr = []; - $chr[48] = '131112'; // 0 - $chr[49] = '111213'; // 1 - $chr[50] = '111312'; // 2 - $chr[51] = '111411'; // 3 - $chr[52] = '121113'; // 4 - $chr[53] = '121212'; // 5 - $chr[54] = '121311'; // 6 - $chr[55] = '111114'; // 7 - $chr[56] = '131211'; // 8 - $chr[57] = '141111'; // 9 - $chr[65] = '211113'; // A - $chr[66] = '211212'; // B - $chr[67] = '211311'; // C - $chr[68] = '221112'; // D - $chr[69] = '221211'; // E - $chr[70] = '231111'; // F - $chr[71] = '112113'; // G - $chr[72] = '112212'; // H - $chr[73] = '112311'; // I - $chr[74] = '122112'; // J - $chr[75] = '132111'; // K - $chr[76] = '111123'; // L - $chr[77] = '111222'; // M - $chr[78] = '111321'; // N - $chr[79] = '121122'; // O - $chr[80] = '131121'; // P - $chr[81] = '212112'; // Q - $chr[82] = '212211'; // R - $chr[83] = '211122'; // S - $chr[84] = '211221'; // T - $chr[85] = '221121'; // U - $chr[86] = '222111'; // V - $chr[87] = '112122'; // W - $chr[88] = '112221'; // X - $chr[89] = '122121'; // Y - $chr[90] = '123111'; // Z - $chr[45] = '121131'; // - - $chr[46] = '311112'; // . - $chr[32] = '311211'; // - $chr[36] = '321111'; // $ - $chr[47] = '112131'; // / - $chr[43] = '113121'; // + - $chr[37] = '211131'; // % - $chr[128] = '121221'; // ($) - $chr[129] = '311121'; // (/) - $chr[130] = '122211'; // (+) - $chr[131] = '312111'; // (%) - $chr[42] = '111141'; // start-stop - $code = strtoupper($code); - $encode = array( - chr(0) => chr(131) . 'U', - chr(1) => chr(128) . 'A', - chr(2) => chr(128) . 'B', - chr(3) => chr(128) . 'C', - chr(4) => chr(128) . 'D', - chr(5) => chr(128) . 'E', - chr(6) => chr(128) . 'F', - chr(7) => chr(128) . 'G', - chr(8) => chr(128) . 'H', - chr(9) => chr(128) . 'I', - chr(10) => chr(128) . 'J', - chr(11) => '£K', - chr(12) => chr(128) . 'L', - chr(13) => chr(128) . 'M', - chr(14) => chr(128) . 'N', - chr(15) => chr(128) . 'O', - chr(16) => chr(128) . 'P', - chr(17) => chr(128) . 'Q', - chr(18) => chr(128) . 'R', - chr(19) => chr(128) . 'S', - chr(20) => chr(128) . 'T', - chr(21) => chr(128) . 'U', - chr(22) => chr(128) . 'V', - chr(23) => chr(128) . 'W', - chr(24) => chr(128) . 'X', - chr(25) => chr(128) . 'Y', - chr(26) => chr(128) . 'Z', - chr(27) => chr(131) . 'A', - chr(28) => chr(131) . 'B', - chr(29) => chr(131) . 'C', - chr(30) => chr(131) . 'D', - chr(31) => chr(131) . 'E', - chr(32) => ' ', - chr(33) => chr(129) . 'A', - chr(34) => chr(129) . 'B', - chr(35) => chr(129) . 'C', - chr(36) => chr(129) . 'D', - chr(37) => chr(129) . 'E', - chr(38) => chr(129) . 'F', - chr(39) => chr(129) . 'G', - chr(40) => chr(129) . 'H', - chr(41) => chr(129) . 'I', - chr(42) => chr(129) . 'J', - chr(43) => chr(129) . 'K', - chr(44) => chr(129) . 'L', - chr(45) => '-', - chr(46) => '.', - chr(47) => chr(129) . 'O', - chr(48) => '0', - chr(49) => '1', - chr(50) => '2', - chr(51) => '3', - chr(52) => '4', - chr(53) => '5', - chr(54) => '6', - chr(55) => '7', - chr(56) => '8', - chr(57) => '9', - chr(58) => chr(129) . 'Z', - chr(59) => chr(131) . 'F', - chr(60) => chr(131) . 'G', - chr(61) => chr(131) . 'H', - chr(62) => chr(131) . 'I', - chr(63) => chr(131) . 'J', - chr(64) => chr(131) . 'V', - chr(65) => 'A', - chr(66) => 'B', - chr(67) => 'C', - chr(68) => 'D', - chr(69) => 'E', - chr(70) => 'F', - chr(71) => 'G', - chr(72) => 'H', - chr(73) => 'I', - chr(74) => 'J', - chr(75) => 'K', - chr(76) => 'L', - chr(77) => 'M', - chr(78) => 'N', - chr(79) => 'O', - chr(80) => 'P', - chr(81) => 'Q', - chr(82) => 'R', - chr(83) => 'S', - chr(84) => 'T', - chr(85) => 'U', - chr(86) => 'V', - chr(87) => 'W', - chr(88) => 'X', - chr(89) => 'Y', - chr(90) => 'Z', - chr(91) => chr(131) . 'K', - chr(92) => chr(131) . 'L', - chr(93) => chr(131) . 'M', - chr(94) => chr(131) . 'N', - chr(95) => chr(131) . 'O', - chr(96) => chr(131) . 'W', - chr(97) => chr(130) . 'A', - chr(98) => chr(130) . 'B', - chr(99) => chr(130) . 'C', - chr(100) => chr(130) . 'D', - chr(101) => chr(130) . 'E', - chr(102) => chr(130) . 'F', - chr(103) => chr(130) . 'G', - chr(104) => chr(130) . 'H', - chr(105) => chr(130) . 'I', - chr(106) => chr(130) . 'J', - chr(107) => chr(130) . 'K', - chr(108) => chr(130) . 'L', - chr(109) => chr(130) . 'M', - chr(110) => chr(130) . 'N', - chr(111) => chr(130) . 'O', - chr(112) => chr(130) . 'P', - chr(113) => chr(130) . 'Q', - chr(114) => chr(130) . 'R', - chr(115) => chr(130) . 'S', - chr(116) => chr(130) . 'T', - chr(117) => chr(130) . 'U', - chr(118) => chr(130) . 'V', - chr(119) => chr(130) . 'W', - chr(120) => chr(130) . 'X', - chr(121) => chr(130) . 'Y', - chr(122) => chr(130) . 'Z', - chr(123) => chr(131) . 'P', - chr(124) => chr(131) . 'Q', - chr(125) => chr(131) . 'R', - chr(126) => chr(131) . 'S', - chr(127) => chr(131) . 'T' - ); - $code_ext = ''; - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - if (ord($code{$i}) > 127) { - throw new InvalidCharacterException('Only supports till char 127'); - } - $code_ext .= $encode[$code{$i}]; - } - // checksum - $code_ext .= $this->checksum_code93($code_ext); - // add start and stop codes - $code = '*' . $code_ext . '*'; - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $k = 0; - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - $char = ord($code{$i}); - if ( ! isset($chr[$char])) { - throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); - } - for ($j = 0; $j < 6; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $chr[$char]{$j}; - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - } - } - $bararray['bcode'][$k] = array('t' => true, 'w' => 1, 'h' => 1, 'p' => 0); - $bararray['maxw'] += 1; - - return $bararray; - } - - /** - * Calculate CODE 93 checksum (modulo 47). - * - * @param $code (string) code to represent. - * @return string checksum code. - * @protected - */ - protected function checksum_code93($code) - { - $chars = array( - '0', - '1', - '2', - '3', - '4', - '5', - '6', - '7', - '8', - '9', - 'A', - 'B', - 'C', - 'D', - 'E', - 'F', - 'G', - 'H', - 'I', - 'J', - 'K', - 'L', - 'M', - 'N', - 'O', - 'P', - 'Q', - 'R', - 'S', - 'T', - 'U', - 'V', - 'W', - 'X', - 'Y', - 'Z', - '-', - '.', - ' ', - '$', - '/', - '+', - '%', - '<', - '=', - '>', - '?' - ); - // translate special characters - $code = strtr($code, chr(128) . chr(131) . chr(129) . chr(130), '<=>?'); - $len = strlen($code); - // calculate check digit C - $p = 1; - $check = 0; - for ($i = ($len - 1); $i >= 0; --$i) { - $k = array_keys($chars, $code{$i}); - $check += ($k[0] * $p); - ++$p; - if ($p > 20) { - $p = 1; - } - } - $check %= 47; - $c = $chars[$check]; - $code .= $c; - // calculate check digit K - $p = 1; - $check = 0; - for ($i = $len; $i >= 0; --$i) { - $k = array_keys($chars, $code{$i}); - $check += ($k[0] * $p); - ++$p; - if ($p > 15) { - $p = 1; - } - } - $check %= 47; - $k = $chars[$check]; - $checksum = $c . $k; - // resto respecial characters - $checksum = strtr($checksum, '<=>?', chr(128) . chr(131) . chr(129) . chr(130)); - - return $checksum; - } - - /** - * Checksum for standard 2 of 5 barcodes. - * - * @param $code (string) code to process. - * @return int checksum. - * @protected - */ - protected function checksum_s25($code) - { - $len = strlen($code); - $sum = 0; - for ($i = 0; $i < $len; $i += 2) { - $sum += $code{$i}; - } - $sum *= 3; - for ($i = 1; $i < $len; $i += 2) { - $sum += ($code{$i}); - } - $r = $sum % 10; - if ($r > 0) { - $r = (10 - $r); - } - - return $r; - } - - /** - * MSI. - * Variation of Plessey code, with similar applications - * Contains digits (0 to 9) and encodes the data only in the width of bars. - * - * @param $code (string) code to represent. - * @param $checksum (boolean) if true add a checksum to the code (modulo 11) - * @return array barcode representation. - * @protected - */ - protected function barcode_msi($code, $checksum = false) - { - $chr['0'] = '100100100100'; - $chr['1'] = '100100100110'; - $chr['2'] = '100100110100'; - $chr['3'] = '100100110110'; - $chr['4'] = '100110100100'; - $chr['5'] = '100110100110'; - $chr['6'] = '100110110100'; - $chr['7'] = '100110110110'; - $chr['8'] = '110100100100'; - $chr['9'] = '110100100110'; - $chr['A'] = '110100110100'; - $chr['B'] = '110100110110'; - $chr['C'] = '110110100100'; - $chr['D'] = '110110100110'; - $chr['E'] = '110110110100'; - $chr['F'] = '110110110110'; - if ($checksum) { - // add checksum - $clen = strlen($code); - $p = 2; - $check = 0; - for ($i = ($clen - 1); $i >= 0; --$i) { - $check += (hexdec($code{$i}) * $p); - ++$p; - if ($p > 7) { - $p = 2; - } - } - $check %= 11; - if ($check > 0) { - $check = 11 - $check; - } - $code .= $check; - } - $seq = '110'; // left guard - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - $digit = $code{$i}; - if ( ! isset($chr[$digit])) { - throw new InvalidCharacterException('Char ' . $digit . ' is unsupported'); - } - $seq .= $chr[$digit]; - } - $seq .= '1001'; // right guard - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - - return $this->binseq_to_array($seq, $bararray); - } - - /** - * Standard 2 of 5 barcodes. - * Used in airline ticket marking, photofinishing - * Contains digits (0 to 9) and encodes the data only in the width of bars. - * - * @param $code (string) code to represent. - * @param $checksum (boolean) if true add a checksum to the code - * @return array barcode representation. - * @protected - */ - protected function barcode_s25($code, $checksum = false) - { - $chr['0'] = '10101110111010'; - $chr['1'] = '11101010101110'; - $chr['2'] = '10111010101110'; - $chr['3'] = '11101110101010'; - $chr['4'] = '10101110101110'; - $chr['5'] = '11101011101010'; - $chr['6'] = '10111011101010'; - $chr['7'] = '10101011101110'; - $chr['8'] = '10101110111010'; - $chr['9'] = '10111010111010'; - if ($checksum) { - // add checksum - $code .= $this->checksum_s25($code); - } - if ((strlen($code) % 2) != 0) { - // add leading zero if code-length is odd - $code = '0' . $code; - } - $seq = '11011010'; - $clen = strlen($code); - for ($i = 0; $i < $clen; ++$i) { - $digit = $code{$i}; - if ( ! isset($chr[$digit])) { - throw new InvalidCharacterException('Char ' . $digit . ' is unsupported'); - } - $seq .= $chr[$digit]; - } - $seq .= '1101011'; - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - - return $this->binseq_to_array($seq, $bararray); - } - - /** - * Convert binary barcode sequence to TCPDF barcode array. - * - * @param $seq (string) barcode as binary sequence. - * @param $bararray (array) barcode array. - * òparam array $bararray TCPDF barcode array to fill up - * @return array barcode representation. - * @protected - */ - protected function binseq_to_array($seq, $bararray) - { - $len = strlen($seq); - $w = 0; - $k = 0; - for ($i = 0; $i < $len; ++$i) { - $w += 1; - if (($i == ($len - 1)) OR (($i < ($len - 1)) AND ($seq{$i} != $seq{($i + 1)}))) { - if ($seq{$i} == '1') { - $t = true; // bar - } else { - $t = false; // space - } - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - $w = 0; - } - } - - return $bararray; - } - - /** - * Interleaved 2 of 5 barcodes. - * Compact numeric code, widely used in industry, air cargo - * Contains digits (0 to 9) and encodes the data in the width of both bars and spaces. - * - * @param $code (string) code to represent. - * @param $checksum (boolean) if true add a checksum to the code - * @return array barcode representation. - * @protected - */ - protected function barcode_i25($code, $checksum = false) - { - $chr['0'] = '11221'; - $chr['1'] = '21112'; - $chr['2'] = '12112'; - $chr['3'] = '22111'; - $chr['4'] = '11212'; - $chr['5'] = '21211'; - $chr['6'] = '12211'; - $chr['7'] = '11122'; - $chr['8'] = '21121'; - $chr['9'] = '12121'; - $chr['A'] = '11'; - $chr['Z'] = '21'; - if ($checksum) { - // add checksum - $code .= $this->checksum_s25($code); - } - if ((strlen($code) % 2) != 0) { - // add leading zero if code-length is odd - $code = '0' . $code; - } - // add start and stop codes - $code = 'AA' . strtolower($code) . 'ZA'; - - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $k = 0; - $clen = strlen($code); - for ($i = 0; $i < $clen; $i = ($i + 2)) { - $char_bar = $code{$i}; - $char_space = $code{$i + 1}; - if ( ! isset($chr[$char_bar]) || ! isset($chr[$char_space])) { - throw new InvalidCharacterException(); - } - // create a bar-space sequence - $seq = ''; - $chrlen = strlen($chr[$char_bar]); - for ($s = 0; $s < $chrlen; $s++) { - $seq .= $chr[$char_bar]{$s} . $chr[$char_space]{$s}; - } - $seqlen = strlen($seq); - for ($j = 0; $j < $seqlen; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $seq{$j}; - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - } - } - - return $bararray; - } - - /** - * C128 barcodes. - * Very capable code, excellent density, high reliability; in very wide use world-wide - * - * @param $code (string) code to represent. - * @param $type (string) barcode type: A, B, C or empty for automatic switch (AUTO mode) - * @return array barcode representation. - * @protected - */ - protected function barcode_c128($code, $type = '') - { - $chr = array( - '212222', /* 00 */ - '222122', /* 01 */ - '222221', /* 02 */ - '121223', /* 03 */ - '121322', /* 04 */ - '131222', /* 05 */ - '122213', /* 06 */ - '122312', /* 07 */ - '132212', /* 08 */ - '221213', /* 09 */ - '221312', /* 10 */ - '231212', /* 11 */ - '112232', /* 12 */ - '122132', /* 13 */ - '122231', /* 14 */ - '113222', /* 15 */ - '123122', /* 16 */ - '123221', /* 17 */ - '223211', /* 18 */ - '221132', /* 19 */ - '221231', /* 20 */ - '213212', /* 21 */ - '223112', /* 22 */ - '312131', /* 23 */ - '311222', /* 24 */ - '321122', /* 25 */ - '321221', /* 26 */ - '312212', /* 27 */ - '322112', /* 28 */ - '322211', /* 29 */ - '212123', /* 30 */ - '212321', /* 31 */ - '232121', /* 32 */ - '111323', /* 33 */ - '131123', /* 34 */ - '131321', /* 35 */ - '112313', /* 36 */ - '132113', /* 37 */ - '132311', /* 38 */ - '211313', /* 39 */ - '231113', /* 40 */ - '231311', /* 41 */ - '112133', /* 42 */ - '112331', /* 43 */ - '132131', /* 44 */ - '113123', /* 45 */ - '113321', /* 46 */ - '133121', /* 47 */ - '313121', /* 48 */ - '211331', /* 49 */ - '231131', /* 50 */ - '213113', /* 51 */ - '213311', /* 52 */ - '213131', /* 53 */ - '311123', /* 54 */ - '311321', /* 55 */ - '331121', /* 56 */ - '312113', /* 57 */ - '312311', /* 58 */ - '332111', /* 59 */ - '314111', /* 60 */ - '221411', /* 61 */ - '431111', /* 62 */ - '111224', /* 63 */ - '111422', /* 64 */ - '121124', /* 65 */ - '121421', /* 66 */ - '141122', /* 67 */ - '141221', /* 68 */ - '112214', /* 69 */ - '112412', /* 70 */ - '122114', /* 71 */ - '122411', /* 72 */ - '142112', /* 73 */ - '142211', /* 74 */ - '241211', /* 75 */ - '221114', /* 76 */ - '413111', /* 77 */ - '241112', /* 78 */ - '134111', /* 79 */ - '111242', /* 80 */ - '121142', /* 81 */ - '121241', /* 82 */ - '114212', /* 83 */ - '124112', /* 84 */ - '124211', /* 85 */ - '411212', /* 86 */ - '421112', /* 87 */ - '421211', /* 88 */ - '212141', /* 89 */ - '214121', /* 90 */ - '412121', /* 91 */ - '111143', /* 92 */ - '111341', /* 93 */ - '131141', /* 94 */ - '114113', /* 95 */ - '114311', /* 96 */ - '411113', /* 97 */ - '411311', /* 98 */ - '113141', /* 99 */ - '114131', /* 100 */ - '311141', /* 101 */ - '411131', /* 102 */ - '211412', /* 103 START A */ - '211214', /* 104 START B */ - '211232', /* 105 START C */ - '233111', /* STOP */ - '200000' /* END */ - ); - // ASCII characters for code A (ASCII 00 - 95) - $keys_a = ' !"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_'; - $keys_a .= chr(0) . chr(1) . chr(2) . chr(3) . chr(4) . chr(5) . chr(6) . chr(7) . chr(8) . chr(9); - $keys_a .= chr(10) . chr(11) . chr(12) . chr(13) . chr(14) . chr(15) . chr(16) . chr(17) . chr(18) . chr(19); - $keys_a .= chr(20) . chr(21) . chr(22) . chr(23) . chr(24) . chr(25) . chr(26) . chr(27) . chr(28) . chr(29); - $keys_a .= chr(30) . chr(31); - // ASCII characters for code B (ASCII 32 - 127) - $keys_b = ' !"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~' . chr(127); - // special codes - $fnc_a = array(241 => 102, 242 => 97, 243 => 96, 244 => 101); - $fnc_b = array(241 => 102, 242 => 97, 243 => 96, 244 => 100); - // array of symbols - $code_data = array(); - // length of the code - $len = strlen($code); switch (strtoupper($type)) { - case 'A': { // MODE A - $startid = 103; - for ($i = 0; $i < $len; ++$i) { - $char = $code{$i}; - $char_id = ord($char); - if (($char_id >= 241) AND ($char_id <= 244)) { - $code_data[] = $fnc_a[$char_id]; - } elseif (($char_id >= 0) AND ($char_id <= 95)) { - $code_data[] = strpos($keys_a, $char); - } else { - throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); - } - } - break; - } - case 'B': { // MODE B - $startid = 104; - for ($i = 0; $i < $len; ++$i) { - $char = $code{$i}; - $char_id = ord($char); - if (($char_id >= 241) AND ($char_id <= 244)) { - $code_data[] = $fnc_b[$char_id]; - } elseif (($char_id >= 32) AND ($char_id <= 127)) { - $code_data[] = strpos($keys_b, $char); - } else { - throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); - } - } - break; - } - case 'C': { // MODE C - $startid = 105; - if (ord($code[0]) == 241) { - $code_data[] = 102; - $code = substr($code, 1); - --$len; - } - if (($len % 2) != 0) { - throw new InvalidLengthException('Length must be even'); - } - for ($i = 0; $i < $len; $i += 2) { - $chrnum = $code{$i} . $code{$i + 1}; - if (preg_match('/([0-9]{2})/', $chrnum) > 0) { - $code_data[] = intval($chrnum); - } else { - throw new InvalidCharacterException(); - } - } - break; - } - default: { // MODE AUTO - // split code into sequences - $sequence = array(); - // get numeric sequences (if any) - $numseq = array(); - preg_match_all('/([0-9]{4,})/', $code, $numseq, PREG_OFFSET_CAPTURE); - if (isset($numseq[1]) AND ! empty($numseq[1])) { - $end_offset = 0; - foreach ($numseq[1] as $val) { - $offset = $val[1]; - - // numeric sequence - $slen = strlen($val[0]); - if (($slen % 2) != 0) { - // the length must be even - ++$offset; - $val[0] = substr($val[0],1); - } - - if ($offset > $end_offset) { - // non numeric sequence - $sequence = array_merge($sequence, - $this->get128ABsequence(substr($code, $end_offset, ($offset - $end_offset)))); - } - // numeric sequence fallback - $slen = strlen($val[0]); - if (($slen % 2) != 0) { - // the length must be even - --$slen; - } - $sequence[] = array('C', substr($code, $offset, $slen), $slen); - $end_offset = $offset + $slen; - } - if ($end_offset < $len) { - $sequence = array_merge($sequence, $this->get128ABsequence(substr($code, $end_offset))); - } - } else { - // text code (non C mode) - $sequence = array_merge($sequence, $this->get128ABsequence($code)); - } - // process the sequence - foreach ($sequence as $key => $seq) { - switch ($seq[0]) { - case 'A': { - if ($key == 0) { - $startid = 103; - } elseif ($sequence[($key - 1)][0] != 'A') { - if (($seq[2] == 1) AND ($key > 0) AND ($sequence[($key - 1)][0] == 'B') AND ( ! isset($sequence[($key - 1)][3]))) { - // single character shift - $code_data[] = 98; - // mark shift - $sequence[$key][3] = true; - } elseif ( ! isset($sequence[($key - 1)][3])) { - $code_data[] = 101; - } - } - for ($i = 0; $i < $seq[2]; ++$i) { - $char = $seq[1]{$i}; - $char_id = ord($char); - if (($char_id >= 241) AND ($char_id <= 244)) { - $code_data[] = $fnc_a[$char_id]; - } else { - $code_data[] = strpos($keys_a, $char); - } - } - break; - } - case 'B': { - if ($key == 0) { - $tmpchr = ord($seq[1][0]); - if (($seq[2] == 1) AND ($tmpchr >= 241) AND ($tmpchr <= 244) AND isset($sequence[($key + 1)]) AND ($sequence[($key + 1)][0] != 'B')) { - switch ($sequence[($key + 1)][0]) { - case 'A': { - $startid = 103; - $sequence[$key][0] = 'A'; - $code_data[] = $fnc_a[$tmpchr]; - break; - } - case 'C': { - $startid = 105; - $sequence[$key][0] = 'C'; - $code_data[] = $fnc_a[$tmpchr]; - break; - } - } - break; - } else { - $startid = 104; - } - } elseif ($sequence[($key - 1)][0] != 'B') { - if (($seq[2] == 1) AND ($key > 0) AND ($sequence[($key - 1)][0] == 'A') AND ( ! isset($sequence[($key - 1)][3]))) { - // single character shift - $code_data[] = 98; - // mark shift - $sequence[$key][3] = true; - } elseif ( ! isset($sequence[($key - 1)][3])) { - $code_data[] = 100; - } - } - for ($i = 0; $i < $seq[2]; ++$i) { - $char = $seq[1]{$i}; - $char_id = ord($char); - if (($char_id >= 241) AND ($char_id <= 244)) { - $code_data[] = $fnc_b[$char_id]; - } else { - $code_data[] = strpos($keys_b, $char); - } - } - break; - } - case 'C': { - if ($key == 0) { - $startid = 105; - } elseif ($sequence[($key - 1)][0] != 'C') { - $code_data[] = 99; - } - for ($i = 0; $i < $seq[2]; $i += 2) { - $chrnum = $seq[1]{$i} . $seq[1]{$i + 1}; - $code_data[] = intval($chrnum); - } - break; - } - } - } - } - } - // calculate check character - $sum = $startid; - foreach ($code_data as $key => $val) { - $sum += ($val * ($key + 1)); - } - // add check character - $code_data[] = ($sum % 103); - // add stop sequence - $code_data[] = 106; - $code_data[] = 107; - // add start code at the beginning - array_unshift($code_data, $startid); - // build barcode array - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - foreach ($code_data as $val) { - $seq = $chr[$val]; - for ($j = 0; $j < 6; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $seq{$j}; - $bararray['bcode'][] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - } - } + case self::TYPE_CODE_32: + return new TypeCode32(); + + case self::TYPE_CODE_39: + return new TypeCode39(); - return $bararray; - } + case self::TYPE_CODE_39_CHECKSUM: + return new TypeCode39Checksum(); - /** - * Split text code in A/B sequence for 128 code - * - * @param $code (string) code to split. - * @return array sequence - * @protected - */ - protected function get128ABsequence($code) - { - $len = strlen($code); - $sequence = array(); - // get A sequences (if any) - $numseq = array(); - preg_match_all('/([\x00-\x1f])/', $code, $numseq, PREG_OFFSET_CAPTURE); - if (isset($numseq[1]) AND ! empty($numseq[1])) { - $end_offset = 0; - foreach ($numseq[1] as $val) { - $offset = $val[1]; - if ($offset > $end_offset) { - // B sequence - $sequence[] = array( - 'B', - substr($code, $end_offset, ($offset - $end_offset)), - ($offset - $end_offset) - ); - } - // A sequence - $slen = strlen($val[0]); - $sequence[] = array('A', substr($code, $offset, $slen), $slen); - $end_offset = $offset + $slen; - } - if ($end_offset < $len) { - $sequence[] = array('B', substr($code, $end_offset), ($len - $end_offset)); - } - } else { - // only B sequence - $sequence[] = array('B', $code, $len); - } + case self::TYPE_CODE_39E: + return new TypeCode39Extended(); - return $sequence; - } + case self::TYPE_CODE_39E_CHECKSUM: + return new TypeCode39ExtendedChecksum(); - /** - * EAN13 and UPC-A barcodes. - * EAN13: European Article Numbering international retail product code - * UPC-A: Universal product code seen on almost all retail products in the USA and Canada - * UPC-E: Short version of UPC symbol - * - * @param $code (string) code to represent. - * @param $len (string) barcode type: 6 = UPC-E, 8 = EAN8, 13 = EAN13, 12 = UPC-A - * @return array barcode representation. - * @protected - */ - protected function barcode_eanupc($code, $len = 13) - { - $upce = false; - if ($len == 6) { - $len = 12; // UPC-A - $upce = true; // UPC-E mode - } - $data_len = $len - 1; - //Padding - $code = str_pad($code, $data_len, '0', STR_PAD_LEFT); - $code_len = strlen($code); - // calculate check digit - $sum_a = 0; - for ($i = 1; $i < $data_len; $i += 2) { - $sum_a += $code{$i}; - } - if ($len > 12) { - $sum_a *= 3; - } - $sum_b = 0; - for ($i = 0; $i < $data_len; $i += 2) { - $sum_b += ($code{$i}); - } - if ($len < 13) { - $sum_b *= 3; - } - $r = ($sum_a + $sum_b) % 10; - if ($r > 0) { - $r = (10 - $r); - } - if ($code_len == $data_len) { - // add check digit - $code .= $r; - } elseif ($r !== intval($code{$data_len})) { - throw new InvalidCheckDigitException(); - } - if ($len == 12) { - // UPC-A - $code = '0' . $code; - ++$len; - } - if ($upce) { - // convert UPC-A to UPC-E - $tmp = substr($code, 4, 3); - if (($tmp == '000') OR ($tmp == '100') OR ($tmp == '200')) { - // manufacturer code ends in 000, 100, or 200 - $upce_code = substr($code, 2, 2) . substr($code, 9, 3) . substr($code, 4, 1); - } else { - $tmp = substr($code, 5, 2); - if ($tmp == '00') { - // manufacturer code ends in 00 - $upce_code = substr($code, 2, 3) . substr($code, 10, 2) . '3'; - } else { - $tmp = substr($code, 6, 1); - if ($tmp == '0') { - // manufacturer code ends in 0 - $upce_code = substr($code, 2, 4) . substr($code, 11, 1) . '4'; - } else { - // manufacturer code does not end in zero - $upce_code = substr($code, 2, 5) . substr($code, 11, 1); - } - } - } - } - //Convert digits to bars - $codes = array( - 'A' => array( // left odd parity - '0' => '0001101', - '1' => '0011001', - '2' => '0010011', - '3' => '0111101', - '4' => '0100011', - '5' => '0110001', - '6' => '0101111', - '7' => '0111011', - '8' => '0110111', - '9' => '0001011' - ), - 'B' => array( // left even parity - '0' => '0100111', - '1' => '0110011', - '2' => '0011011', - '3' => '0100001', - '4' => '0011101', - '5' => '0111001', - '6' => '0000101', - '7' => '0010001', - '8' => '0001001', - '9' => '0010111' - ), - 'C' => array( // right - '0' => '1110010', - '1' => '1100110', - '2' => '1101100', - '3' => '1000010', - '4' => '1011100', - '5' => '1001110', - '6' => '1010000', - '7' => '1000100', - '8' => '1001000', - '9' => '1110100' - ) - ); - $parities = array( - '0' => array('A', 'A', 'A', 'A', 'A', 'A'), - '1' => array('A', 'A', 'B', 'A', 'B', 'B'), - '2' => array('A', 'A', 'B', 'B', 'A', 'B'), - '3' => array('A', 'A', 'B', 'B', 'B', 'A'), - '4' => array('A', 'B', 'A', 'A', 'B', 'B'), - '5' => array('A', 'B', 'B', 'A', 'A', 'B'), - '6' => array('A', 'B', 'B', 'B', 'A', 'A'), - '7' => array('A', 'B', 'A', 'B', 'A', 'B'), - '8' => array('A', 'B', 'A', 'B', 'B', 'A'), - '9' => array('A', 'B', 'B', 'A', 'B', 'A') - ); - $upce_parities = array(); - $upce_parities[0] = array( - '0' => array('B', 'B', 'B', 'A', 'A', 'A'), - '1' => array('B', 'B', 'A', 'B', 'A', 'A'), - '2' => array('B', 'B', 'A', 'A', 'B', 'A'), - '3' => array('B', 'B', 'A', 'A', 'A', 'B'), - '4' => array('B', 'A', 'B', 'B', 'A', 'A'), - '5' => array('B', 'A', 'A', 'B', 'B', 'A'), - '6' => array('B', 'A', 'A', 'A', 'B', 'B'), - '7' => array('B', 'A', 'B', 'A', 'B', 'A'), - '8' => array('B', 'A', 'B', 'A', 'A', 'B'), - '9' => array('B', 'A', 'A', 'B', 'A', 'B') - ); - $upce_parities[1] = array( - '0' => array('A', 'A', 'A', 'B', 'B', 'B'), - '1' => array('A', 'A', 'B', 'A', 'B', 'B'), - '2' => array('A', 'A', 'B', 'B', 'A', 'B'), - '3' => array('A', 'A', 'B', 'B', 'B', 'A'), - '4' => array('A', 'B', 'A', 'A', 'B', 'B'), - '5' => array('A', 'B', 'B', 'A', 'A', 'B'), - '6' => array('A', 'B', 'B', 'B', 'A', 'A'), - '7' => array('A', 'B', 'A', 'B', 'A', 'B'), - '8' => array('A', 'B', 'A', 'B', 'B', 'A'), - '9' => array('A', 'B', 'B', 'A', 'B', 'A') - ); - $k = 0; - $seq = '101'; // left guard bar - if ($upce) { - $bararray = array('code' => $upce_code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $p = $upce_parities[$code[1]][$r]; - for ($i = 0; $i < 6; ++$i) { - $seq .= $codes[$p[$i]][$upce_code{$i}]; - } - $seq .= '010101'; // right guard bar - } else { - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $half_len = intval(ceil($len / 2)); - if ($len == 8) { - for ($i = 0; $i < $half_len; ++$i) { - $seq .= $codes['A'][$code{$i}]; - } - } else { - $p = $parities[$code[0]]; - for ($i = 1; $i < $half_len; ++$i) { - $seq .= $codes[$p[$i - 1]][$code{$i}]; - } - } - $seq .= '01010'; // center guard bar - for ($i = $half_len; $i < $len; ++$i) { - if ( ! isset($codes['C'][$code{$i}])) { - throw new InvalidCharacterException('Char ' . $code{$i} . ' not allowed'); - } - $seq .= $codes['C'][$code{$i}]; - } - $seq .= '101'; // right guard bar - } - $clen = strlen($seq); - $w = 0; - for ($i = 0; $i < $clen; ++$i) { - $w += 1; - if (($i == ($clen - 1)) OR (($i < ($clen - 1)) AND ($seq{$i} != $seq{($i + 1)}))) { - if ($seq{$i} == '1') { - $t = true; // bar - } else { - $t = false; // space - } - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - $w = 0; - } - } + case self::TYPE_CODE_93: + return new TypeCode93(); - return $bararray; - } + case self::TYPE_STANDARD_2_5: + return new TypeStandard2of5(); - /** - * UPC-Based Extensions - * 2-Digit Ext.: Used to indicate magazines and newspaper issue numbers - * 5-Digit Ext.: Used to mark suggested retail price of books - * - * @param $code (string) code to represent. - * @param $len (string) barcode type: 2 = 2-Digit, 5 = 5-Digit - * @return array barcode representation. - * @protected - */ - protected function barcode_eanext($code, $len = 5) - { - //Padding - $code = str_pad($code, $len, '0', STR_PAD_LEFT); - // calculate check digit - if ($len == 2) { - $r = $code % 4; - } elseif ($len == 5) { - $r = (3 * ($code[0] + $code[2] + $code[4])) + (9 * ($code[1] + $code[3])); - $r %= 10; - } else { - throw new InvalidCheckDigitException(); - } - //Convert digits to bars - $codes = array( - 'A' => array( // left odd parity - '0' => '0001101', - '1' => '0011001', - '2' => '0010011', - '3' => '0111101', - '4' => '0100011', - '5' => '0110001', - '6' => '0101111', - '7' => '0111011', - '8' => '0110111', - '9' => '0001011' - ), - 'B' => array( // left even parity - '0' => '0100111', - '1' => '0110011', - '2' => '0011011', - '3' => '0100001', - '4' => '0011101', - '5' => '0111001', - '6' => '0000101', - '7' => '0010001', - '8' => '0001001', - '9' => '0010111' - ) - ); - $parities = array(); - $parities[2] = array( - '0' => array('A', 'A'), - '1' => array('A', 'B'), - '2' => array('B', 'A'), - '3' => array('B', 'B') - ); - $parities[5] = array( - '0' => array('B', 'B', 'A', 'A', 'A'), - '1' => array('B', 'A', 'B', 'A', 'A'), - '2' => array('B', 'A', 'A', 'B', 'A'), - '3' => array('B', 'A', 'A', 'A', 'B'), - '4' => array('A', 'B', 'B', 'A', 'A'), - '5' => array('A', 'A', 'B', 'B', 'A'), - '6' => array('A', 'A', 'A', 'B', 'B'), - '7' => array('A', 'B', 'A', 'B', 'A'), - '8' => array('A', 'B', 'A', 'A', 'B'), - '9' => array('A', 'A', 'B', 'A', 'B') - ); - $p = $parities[$len][$r]; - $seq = '1011'; // left guard bar - $seq .= $codes[$p[0]][$code[0]]; - for ($i = 1; $i < $len; ++$i) { - $seq .= '01'; // separator - $seq .= $codes[$p[$i]][$code{$i}]; - } - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); + case self::TYPE_STANDARD_2_5_CHECKSUM: + return new TypeStandard2of5Checksum(); - return $this->binseq_to_array($seq, $bararray); - } + case self::TYPE_INTERLEAVED_2_5: + return new TypeInterleaved25(); - /** - * POSTNET and PLANET barcodes. - * Used by U.S. Postal Service for automated mail sorting - * - * @param $code (string) zip code to represent. Must be a string containing a zip code of the form DDDDD or - * DDDDD-DDDD. - * @param $planet (boolean) if true print the PLANET barcode, otherwise print POSTNET - * @return array barcode representation. - * @protected - */ - protected function barcode_postnet($code, $planet = false) - { - // bar length - if ($planet) { - $barlen = Array( - 0 => Array(1, 1, 2, 2, 2), - 1 => Array(2, 2, 2, 1, 1), - 2 => Array(2, 2, 1, 2, 1), - 3 => Array(2, 2, 1, 1, 2), - 4 => Array(2, 1, 2, 2, 1), - 5 => Array(2, 1, 2, 1, 2), - 6 => Array(2, 1, 1, 2, 2), - 7 => Array(1, 2, 2, 2, 1), - 8 => Array(1, 2, 2, 1, 2), - 9 => Array(1, 2, 1, 2, 2) - ); - } else { - $barlen = Array( - 0 => Array(2, 2, 1, 1, 1), - 1 => Array(1, 1, 1, 2, 2), - 2 => Array(1, 1, 2, 1, 2), - 3 => Array(1, 1, 2, 2, 1), - 4 => Array(1, 2, 1, 1, 2), - 5 => Array(1, 2, 1, 2, 1), - 6 => Array(1, 2, 2, 1, 1), - 7 => Array(2, 1, 1, 1, 2), - 8 => Array(2, 1, 1, 2, 1), - 9 => Array(2, 1, 2, 1, 1) - ); - } - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 2, 'bcode' => array()); - $k = 0; - $code = str_replace('-', '', $code); - $code = str_replace(' ', '', $code); - $len = strlen($code); - // calculate checksum - $sum = 0; - for ($i = 0; $i < $len; ++$i) { - $sum += intval($code{$i}); - } - $chkd = ($sum % 10); - if ($chkd > 0) { - $chkd = (10 - $chkd); - } - $code .= $chkd; - $len = strlen($code); - // start bar - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - for ($i = 0; $i < $len; ++$i) { - for ($j = 0; $j < 5; ++$j) { - $h = $barlen[$code{$i}][$j]; - $p = floor(1 / $h); - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => $h, 'p' => $p); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - } - // end bar - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 1; - - return $bararray; - } + case self::TYPE_INTERLEAVED_2_5_CHECKSUM: + return new TypeInterleaved25Checksum(); - /** - * RMS4CC - CBC - KIX - * RMS4CC (Royal Mail 4-state Customer Code) - CBC (Customer Bar Code) - KIX (Klant index - Customer index) - * RM4SCC is the name of the barcode symbology used by the Royal Mail for its Cleanmail service. - * - * @param $code (string) code to print - * @param $kix (boolean) if true prints the KIX variation (doesn't use the start and end symbols, and the checksum) - * - in this case the house number must be sufficed with an X and placed at the end of the code. - * @return array barcode representation. - * @protected - */ - protected function barcode_rms4cc($code, $kix = false) - { - $notkix = ! $kix; - // bar mode - // 1 = pos 1, length 2 - // 2 = pos 1, length 3 - // 3 = pos 2, length 1 - // 4 = pos 2, length 2 - $barmode = array( - '0' => array(3, 3, 2, 2), - '1' => array(3, 4, 1, 2), - '2' => array(3, 4, 2, 1), - '3' => array(4, 3, 1, 2), - '4' => array(4, 3, 2, 1), - '5' => array(4, 4, 1, 1), - '6' => array(3, 1, 4, 2), - '7' => array(3, 2, 3, 2), - '8' => array(3, 2, 4, 1), - '9' => array(4, 1, 3, 2), - 'A' => array(4, 1, 4, 1), - 'B' => array(4, 2, 3, 1), - 'C' => array(3, 1, 2, 4), - 'D' => array(3, 2, 1, 4), - 'E' => array(3, 2, 2, 3), - 'F' => array(4, 1, 1, 4), - 'G' => array(4, 1, 2, 3), - 'H' => array(4, 2, 1, 3), - 'I' => array(1, 3, 4, 2), - 'J' => array(1, 4, 3, 2), - 'K' => array(1, 4, 4, 1), - 'L' => array(2, 3, 3, 2), - 'M' => array(2, 3, 4, 1), - 'N' => array(2, 4, 3, 1), - 'O' => array(1, 3, 2, 4), - 'P' => array(1, 4, 1, 4), - 'Q' => array(1, 4, 2, 3), - 'R' => array(2, 3, 1, 4), - 'S' => array(2, 3, 2, 3), - 'T' => array(2, 4, 1, 3), - 'U' => array(1, 1, 4, 4), - 'V' => array(1, 2, 3, 4), - 'W' => array(1, 2, 4, 3), - 'X' => array(2, 1, 3, 4), - 'Y' => array(2, 1, 4, 3), - 'Z' => array(2, 2, 3, 3) - ); - $code = strtoupper($code); - $len = strlen($code); - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 3, 'bcode' => array()); - if ($notkix) { - // table for checksum calculation (row,col) - $checktable = array( - '0' => array(1, 1), - '1' => array(1, 2), - '2' => array(1, 3), - '3' => array(1, 4), - '4' => array(1, 5), - '5' => array(1, 0), - '6' => array(2, 1), - '7' => array(2, 2), - '8' => array(2, 3), - '9' => array(2, 4), - 'A' => array(2, 5), - 'B' => array(2, 0), - 'C' => array(3, 1), - 'D' => array(3, 2), - 'E' => array(3, 3), - 'F' => array(3, 4), - 'G' => array(3, 5), - 'H' => array(3, 0), - 'I' => array(4, 1), - 'J' => array(4, 2), - 'K' => array(4, 3), - 'L' => array(4, 4), - 'M' => array(4, 5), - 'N' => array(4, 0), - 'O' => array(5, 1), - 'P' => array(5, 2), - 'Q' => array(5, 3), - 'R' => array(5, 4), - 'S' => array(5, 5), - 'T' => array(5, 0), - 'U' => array(0, 1), - 'V' => array(0, 2), - 'W' => array(0, 3), - 'X' => array(0, 4), - 'Y' => array(0, 5), - 'Z' => array(0, 0) - ); - $row = 0; - $col = 0; - for ($i = 0; $i < $len; ++$i) { - $row += $checktable[$code{$i}][0]; - $col += $checktable[$code{$i}][1]; - } - $row %= 6; - $col %= 6; - $chk = array_keys($checktable, array($row, $col)); - $code .= $chk[0]; - ++$len; - } - $k = 0; - if ($notkix) { - // start bar - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - for ($i = 0; $i < $len; ++$i) { - for ($j = 0; $j < 4; ++$j) { - switch ($barmode[$code{$i}][$j]) { - case 1: { - $p = 0; - $h = 2; - break; - } - case 2: { - $p = 0; - $h = 3; - break; - } - case 3: { - $p = 1; - $h = 1; - break; - } - case 4: { - $p = 1; - $h = 2; - break; - } - } - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => $h, 'p' => $p); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - } - if ($notkix) { - // stop bar - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => 3, 'p' => 0); - $bararray['maxw'] += 1; - } - - return $bararray; - } - - /** - * CODABAR barcodes. - * Older code often used in library systems, sometimes in blood banks - * - * @param $code (string) code to represent. - * @return array barcode representation. - * @protected - */ - protected function barcode_codabar($code) - { - $chr = array( - '0' => '11111221', - '1' => '11112211', - '2' => '11121121', - '3' => '22111111', - '4' => '11211211', - '5' => '21111211', - '6' => '12111121', - '7' => '12112111', - '8' => '12211111', - '9' => '21121111', - '-' => '11122111', - '$' => '11221111', - ':' => '21112121', - '/' => '21211121', - '.' => '21212111', - '+' => '11222221', - 'A' => '11221211', - 'B' => '12121121', - 'C' => '11121221', - 'D' => '11122211' - ); - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $k = 0; - $w = 0; - $seq = ''; - $code = 'A' . strtoupper($code) . 'A'; - $len = strlen($code); - for ($i = 0; $i < $len; ++$i) { - if ( ! isset($chr[$code{$i}])) { - throw new InvalidCharacterException('Char ' . $code{$i} . ' is unsupported'); - } - $seq = $chr[$code{$i}]; - for ($j = 0; $j < 8; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $seq{$j}; - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - } - } - - return $bararray; - } - - /** - * CODE11 barcodes. - * Used primarily for labeling telecommunications equipment - * - * @param $code (string) code to represent. - * @return array barcode representation. - * @protected - */ - protected function barcode_code11($code) - { - $chr = array( - '0' => '111121', - '1' => '211121', - '2' => '121121', - '3' => '221111', - '4' => '112121', - '5' => '212111', - '6' => '122111', - '7' => '111221', - '8' => '211211', - '9' => '211111', - '-' => '112111', - 'S' => '112211' - ); - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); - $k = 0; - $w = 0; - $seq = ''; - $len = strlen($code); - // calculate check digit C - $p = 1; - $check = 0; - for ($i = ($len - 1); $i >= 0; --$i) { - $digit = $code{$i}; - if ($digit == '-') { - $dval = 10; - } else { - $dval = intval($digit); - } - $check += ($dval * $p); - ++$p; - if ($p > 10) { - $p = 1; - } - } - $check %= 11; - if ($check == 10) { - $check = '-'; - } - $code .= $check; - if ($len > 10) { - // calculate check digit K - $p = 1; - $check = 0; - for ($i = $len; $i >= 0; --$i) { - $digit = $code{$i}; - if ($digit == '-') { - $dval = 10; - } else { - $dval = intval($digit); - } - $check += ($dval * $p); - ++$p; - if ($p > 9) { - $p = 1; - } - } - $check %= 11; - $code .= $check; - ++$len; - } - $code = 'S' . $code . 'S'; - $len += 3; - for ($i = 0; $i < $len; ++$i) { - if ( ! isset($chr[$code{$i}])) { - throw new InvalidCharacterException('Char ' . $code{$i} . ' is unsupported'); - } - $seq = $chr[$code{$i}]; - for ($j = 0; $j < 6; ++$j) { - if (($j % 2) == 0) { - $t = true; // bar - } else { - $t = false; // space - } - $w = $seq{$j}; - $bararray['bcode'][$k] = array('t' => $t, 'w' => $w, 'h' => 1, 'p' => 0); - $bararray['maxw'] += $w; - ++$k; - } - } - - return $bararray; - } - - /** - * Pharmacode - * Contains digits (0 to 9) - * - * @param $code (string) code to represent. - * @return array barcode representation. - * @protected - */ - protected function barcode_pharmacode($code) - { - $seq = ''; - $code = intval($code); - while ($code > 0) { - if (($code % 2) == 0) { - $seq .= '11100'; - $code -= 2; - } else { - $seq .= '100'; - $code -= 1; - } - $code /= 2; - } - $seq = substr($seq, 0, -2); - $seq = strrev($seq); - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 1, 'bcode' => array()); + case self::TYPE_ITF_14: + return new TypeITF14(); - return $this->binseq_to_array($seq, $bararray); - } + case self::TYPE_CODE_128: + return new TypeCode128(); - /** - * Pharmacode two-track - * Contains digits (0 to 9) - * - * @param $code (string) code to represent. - * @return array barcode representation. - * @protected - */ - protected function barcode_pharmacode2t($code) - { - $seq = ''; - $code = intval($code); - do { - switch ($code % 3) { - case 0: { - $seq .= '3'; - $code = ($code - 3) / 3; - break; - } - case 1: { - $seq .= '1'; - $code = ($code - 1) / 3; - break; - } - case 2: { - $seq .= '2'; - $code = ($code - 2) / 3; - break; - } - } - } while ($code != 0); - $seq = strrev($seq); - $k = 0; - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 2, 'bcode' => array()); - $len = strlen($seq); - for ($i = 0; $i < $len; ++$i) { - switch ($seq{$i}) { - case '1': { - $p = 1; - $h = 1; - break; - } - case '2': { - $p = 0; - $h = 1; - break; - } - case '3': { - $p = 0; - $h = 2; - break; - } - } - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => $h, 'p' => $p); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - unset($bararray['bcode'][($k - 1)]); - --$bararray['maxw']; + case self::TYPE_CODE_128_A: + return new TypeCode128A(); - return $bararray; - } + case self::TYPE_CODE_128_B: + return new TypeCode128B(); - /** - * IMB - Intelligent Mail Barcode - Onecode - USPS-B-3200 - * (requires PHP bcmath extension) - * Intelligent Mail barcode is a 65-bar code for use on mail in the United States. - * The fields are described as follows: - * - * @param $code (string) code to print, separate the ZIP (routing code) from the rest using a minus char '-' - * (BarcodeID_ServiceTypeID_MailerID_SerialNumber-RoutingCode) - * @return array barcode representation. - * @protected - */ - protected function barcode_imb($code) - { - $asc_chr = array( - 4, - 0, - 2, - 6, - 3, - 5, - 1, - 9, - 8, - 7, - 1, - 2, - 0, - 6, - 4, - 8, - 2, - 9, - 5, - 3, - 0, - 1, - 3, - 7, - 4, - 6, - 8, - 9, - 2, - 0, - 5, - 1, - 9, - 4, - 3, - 8, - 6, - 7, - 1, - 2, - 4, - 3, - 9, - 5, - 7, - 8, - 3, - 0, - 2, - 1, - 4, - 0, - 9, - 1, - 7, - 0, - 2, - 4, - 6, - 3, - 7, - 1, - 9, - 5, - 8 - ); - $dsc_chr = array( - 7, - 1, - 9, - 5, - 8, - 0, - 2, - 4, - 6, - 3, - 5, - 8, - 9, - 7, - 3, - 0, - 6, - 1, - 7, - 4, - 6, - 8, - 9, - 2, - 5, - 1, - 7, - 5, - 4, - 3, - 8, - 7, - 6, - 0, - 2, - 5, - 4, - 9, - 3, - 0, - 1, - 6, - 8, - 2, - 0, - 4, - 5, - 9, - 6, - 7, - 5, - 2, - 6, - 3, - 8, - 5, - 1, - 9, - 8, - 7, - 4, - 0, - 2, - 6, - 3 - ); - $asc_pos = array( - 3, - 0, - 8, - 11, - 1, - 12, - 8, - 11, - 10, - 6, - 4, - 12, - 2, - 7, - 9, - 6, - 7, - 9, - 2, - 8, - 4, - 0, - 12, - 7, - 10, - 9, - 0, - 7, - 10, - 5, - 7, - 9, - 6, - 8, - 2, - 12, - 1, - 4, - 2, - 0, - 1, - 5, - 4, - 6, - 12, - 1, - 0, - 9, - 4, - 7, - 5, - 10, - 2, - 6, - 9, - 11, - 2, - 12, - 6, - 7, - 5, - 11, - 0, - 3, - 2 - ); - $dsc_pos = array( - 2, - 10, - 12, - 5, - 9, - 1, - 5, - 4, - 3, - 9, - 11, - 5, - 10, - 1, - 6, - 3, - 4, - 1, - 10, - 0, - 2, - 11, - 8, - 6, - 1, - 12, - 3, - 8, - 6, - 4, - 4, - 11, - 0, - 6, - 1, - 9, - 11, - 5, - 3, - 7, - 3, - 10, - 7, - 11, - 8, - 2, - 10, - 3, - 5, - 8, - 0, - 3, - 12, - 11, - 8, - 4, - 5, - 1, - 3, - 0, - 7, - 12, - 9, - 8, - 10 - ); - $code_arr = explode('-', $code); - $tracking_number = $code_arr[0]; - if (isset($code_arr[1])) { - $routing_code = $code_arr[1]; - } else { - $routing_code = ''; - } - // Conversion of Routing Code - switch (strlen($routing_code)) { - case 0: { - $binary_code = 0; - break; - } - case 5: { - $binary_code = bcadd($routing_code, '1'); - break; - } - case 9: { - $binary_code = bcadd($routing_code, '100001'); - break; - } - case 11: { - $binary_code = bcadd($routing_code, '1000100001'); - break; - } - default: { - throw new BarcodeException('Routing code unknown'); - break; - } - } - $binary_code = bcmul($binary_code, 10); - $binary_code = bcadd($binary_code, $tracking_number[0]); - $binary_code = bcmul($binary_code, 5); - $binary_code = bcadd($binary_code, $tracking_number[1]); - $binary_code .= substr($tracking_number, 2, 18); - // convert to hexadecimal - $binary_code = $this->dec_to_hex($binary_code); - // pad to get 13 bytes - $binary_code = str_pad($binary_code, 26, '0', STR_PAD_LEFT); - // convert string to array of bytes - $binary_code_arr = chunk_split($binary_code, 2, "\r"); - $binary_code_arr = substr($binary_code_arr, 0, -1); - $binary_code_arr = explode("\r", $binary_code_arr); - // calculate frame check sequence - $fcs = $this->imb_crc11fcs($binary_code_arr); - // exclude first 2 bits from first byte - $first_byte = sprintf('%2s', dechex((hexdec($binary_code_arr[0]) << 2) >> 2)); - $binary_code_102bit = $first_byte . substr($binary_code, 2); - // convert binary data to codewords - $codewords = array(); - $data = $this->hex_to_dec($binary_code_102bit); - $codewords[0] = bcmod($data, 636) * 2; - $data = bcdiv($data, 636); - for ($i = 1; $i < 9; ++$i) { - $codewords[$i] = bcmod($data, 1365); - $data = bcdiv($data, 1365); - } - $codewords[9] = $data; - if (($fcs >> 10) == 1) { - $codewords[9] += 659; - } - // generate lookup tables - $table2of13 = $this->imb_tables(2, 78); - $table5of13 = $this->imb_tables(5, 1287); - // convert codewords to characters - $characters = array(); - $bitmask = 512; - foreach ($codewords as $k => $val) { - if ($val <= 1286) { - $chrcode = $table5of13[$val]; - } else { - $chrcode = $table2of13[($val - 1287)]; - } - if (($fcs & $bitmask) > 0) { - // bitwise invert - $chrcode = ((~$chrcode) & 8191); - } - $characters[] = $chrcode; - $bitmask /= 2; - } - $characters = array_reverse($characters); - // build bars - $k = 0; - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 3, 'bcode' => array()); - for ($i = 0; $i < 65; ++$i) { - $asc = (($characters[$asc_chr[$i]] & pow(2, $asc_pos[$i])) > 0); - $dsc = (($characters[$dsc_chr[$i]] & pow(2, $dsc_pos[$i])) > 0); - if ($asc AND $dsc) { - // full bar (F) - $p = 0; - $h = 3; - } elseif ($asc) { - // ascender (A) - $p = 0; - $h = 2; - } elseif ($dsc) { - // descender (D) - $p = 1; - $h = 2; - } else { - // tracker (T) - $p = 1; - $h = 1; - } - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => $h, 'p' => $p); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - unset($bararray['bcode'][($k - 1)]); - --$bararray['maxw']; + case self::TYPE_CODE_128_C: + return new TypeCode128C(); - return $bararray; - } + case self::TYPE_EAN_2: + return new TypeUpcExtension2(); - /** - * IMB - Intelligent Mail Barcode - Onecode - USPS-B-3200 - * - * @param $code (string) pre-formatted IMB barcode (65 chars "FADT") - * @return array barcode representation. - * @protected - */ - protected function barcode_imb_pre($code) - { - if ( ! preg_match('/^[fadtFADT]{65}$/', $code) == 1) { - throw new InvalidFormatException(); - } - $characters = str_split(strtolower($code), 1); - // build bars - $k = 0; - $bararray = array('code' => $code, 'maxw' => 0, 'maxh' => 3, 'bcode' => array()); - for ($i = 0; $i < 65; ++$i) { - switch ($characters[$i]) { - case 'f': { - // full bar - $p = 0; - $h = 3; - break; - } - case 'a': { - // ascender - $p = 0; - $h = 2; - break; - } - case 'd': { - // descender - $p = 1; - $h = 2; - break; - } - case 't': { - // tracker (short) - $p = 1; - $h = 1; - break; - } - } - $bararray['bcode'][$k++] = array('t' => 1, 'w' => 1, 'h' => $h, 'p' => $p); - $bararray['bcode'][$k++] = array('t' => 0, 'w' => 1, 'h' => 2, 'p' => 0); - $bararray['maxw'] += 2; - } - unset($bararray['bcode'][($k - 1)]); - --$bararray['maxw']; + case self::TYPE_EAN_5: + return new TypeUpcExtension5(); - return $bararray; - } + case self::TYPE_EAN_8: + return new TypeEan8(); - /** - * Convert large integer number to hexadecimal representation. - * (requires PHP bcmath extension) - * - * @param $number (string) number to convert specified as a string - * @return string hexadecimal representation - */ - protected function dec_to_hex($number) - { - if ($number == 0) { - return '00'; - } + case self::TYPE_EAN_13: + return new TypeEan13(); - $hex = []; + case self::TYPE_UPC_A: + return new TypeUpcA(); - while ($number > 0) { - array_push($hex, strtoupper(dechex(bcmod($number, '16')))); - $number = bcdiv($number, '16', 0); - } - $hex = array_reverse($hex); + case self::TYPE_UPC_E: + return new TypeUpcE(); - return implode($hex); - } + case self::TYPE_MSI: + return new TypeMsi(); - /** - * Convert large hexadecimal number to decimal representation (string). - * (requires PHP bcmath extension) - * - * @param $hex (string) hexadecimal number to convert specified as a string - * @return string hexadecimal representation - */ - protected function hex_to_dec($hex) - { - $dec = 0; - $bitval = 1; - $len = strlen($hex); - for ($pos = ($len - 1); $pos >= 0; --$pos) { - $dec = bcadd($dec, bcmul(hexdec($hex{$pos}), $bitval)); - $bitval = bcmul($bitval, 16); - } + case self::TYPE_MSI_CHECKSUM: + return new TypeMsiChecksum(); - return $dec; - } + case self::TYPE_POSTNET: + return new TypePostnet(); - /** - * Intelligent Mail Barcode calculation of Frame Check Sequence - * - * @param $code_arr (string) array of hexadecimal values (13 bytes holding 102 bits right justified). - * @return int 11 bit Frame Check Sequence as integer (decimal base) - * @protected - */ - protected function imb_crc11fcs($code_arr) - { - $genpoly = 0x0F35; // generator polynomial - $fcs = 0x07FF; // Frame Check Sequence - // do most significant byte skipping the 2 most significant bits - $data = hexdec($code_arr[0]) << 5; - for ($bit = 2; $bit < 8; ++$bit) { - if (($fcs ^ $data) & 0x400) { - $fcs = ($fcs << 1) ^ $genpoly; - } else { - $fcs = ($fcs << 1); - } - $fcs &= 0x7FF; - $data <<= 1; - } - // do rest of bytes - for ($byte = 1; $byte < 13; ++$byte) { - $data = hexdec($code_arr[$byte]) << 3; - for ($bit = 0; $bit < 8; ++$bit) { - if (($fcs ^ $data) & 0x400) { - $fcs = ($fcs << 1) ^ $genpoly; - } else { - $fcs = ($fcs << 1); - } - $fcs &= 0x7FF; - $data <<= 1; - } - } + case self::TYPE_PLANET: + return new TypePlanet(); - return $fcs; - } + case self::TYPE_RMS4CC: + return new TypeRms4cc(); - /** - * Reverse unsigned short value - * - * @param $num (int) value to reversr - * @return int reversed value - * @protected - */ - protected function imb_reverse_us($num) - { - $rev = 0; - for ($i = 0; $i < 16; ++$i) { - $rev <<= 1; - $rev |= ($num & 1); - $num >>= 1; - } + case self::TYPE_KIX: + return new TypeKix(); - return $rev; - } + case self::TYPE_IMB: + return new TypeIntelligentMailBarcode(); - /** - * generate Nof13 tables used for Intelligent Mail Barcode - * - * @param $n (int) is the type of table: 2 for 2of13 table, 5 for 5of13table - * @param $size (int) size of table (78 for n=2 and 1287 for n=5) - * @return array requested table - * @protected - */ - protected function imb_tables($n, $size) - { - $table = array(); - $lli = 0; // LUT lower index - $lui = $size - 1; // LUT upper index - for ($count = 0; $count < 8192; ++$count) { - $bit_count = 0; - for ($bit_index = 0; $bit_index < 13; ++$bit_index) { - $bit_count += intval(($count & (1 << $bit_index)) != 0); - } - // if we don't have the right number of bits on, go on to the next value - if ($bit_count == $n) { - $reverse = ($this->imb_reverse_us($count) >> 3); - // if the reverse is less than count, we have already visited this pair before - if ($reverse >= $count) { - // If count is symmetric, place it at the first free slot from the end of the list. - // Otherwise, place it at the first free slot from the beginning of the list AND place $reverse ath the next free slot from the beginning of the list - if ($reverse == $count) { - $table[$lui] = $count; - --$lui; - } else { - $table[$lli] = $count; - ++$lli; - $table[$lli] = $reverse; - ++$lli; - } - } - } - } + case self::TYPE_CODABAR: + return new TypeCodabar(); - return $table; - } + case self::TYPE_CODE_11: + return new TypeCode11(); - protected function convertBarcodeArrayToNewStyle($oldBarcodeArray) - { - $newBarcodeArray = []; - $newBarcodeArray['code'] = $oldBarcodeArray['code']; - $newBarcodeArray['maxWidth'] = $oldBarcodeArray['maxw']; - $newBarcodeArray['maxHeight'] = $oldBarcodeArray['maxh']; - $newBarcodeArray['bars'] = []; - foreach ($oldBarcodeArray['bcode'] as $oldbar) { - $newBar = []; - $newBar['width'] = $oldbar['w']; - $newBar['height'] = $oldbar['h']; - $newBar['positionVertical'] = $oldbar['p']; - $newBar['drawBar'] = $oldbar['t']; - $newBar['drawSpacing'] = ! $oldbar['t']; + case self::TYPE_PHARMA_CODE: + return new TypePharmacode(); - $newBarcodeArray['bars'][] = $newBar; + case self::TYPE_PHARMA_CODE_TWO_TRACKS: + return new TypePharmacodeTwoCode(); + + case self::TYPE_TELEPEN_ALPHA: + return new TypeTelepen(); + + case self::TYPE_TELEPEN_NUMERIC: + return new TypeTelepen('numeric'); + } - return $newBarcodeArray; + throw new UnknownTypeException(); } } diff --git a/src/BarcodeGeneratorDynamicHTML.php b/src/BarcodeGeneratorDynamicHTML.php new file mode 100644 index 00000000..5cebc2b5 --- /dev/null +++ b/src/BarcodeGeneratorDynamicHTML.php @@ -0,0 +1,29 @@ +getBarcodeData($barcode, $type); + + $renderer = new \Picqer\Barcode\Renderers\DynamicHtmlRenderer(); + $renderer->setForegroundColor(ColorHelper::getArrayFromColorString($foregroundColor)); + + return $renderer->render($barcodeData); + } +} diff --git a/src/BarcodeGeneratorHTML.php b/src/BarcodeGeneratorHTML.php index eeb95078..01b8b95e 100644 --- a/src/BarcodeGeneratorHTML.php +++ b/src/BarcodeGeneratorHTML.php @@ -2,42 +2,32 @@ namespace Picqer\Barcode; +use Picqer\Barcode\Exceptions\UnknownTypeException; +use Picqer\Barcode\Helpers\ColorHelper; + class BarcodeGeneratorHTML extends BarcodeGenerator { - /** * Return an HTML representation of barcode. + * This original version uses pixel based widths and heights. Use Dynamic HTML version for better quality representation. * - * @param string $code code to print - * @param string $type type of barcode + * @param string $barcode code to print + * @param BarcodeGenerator::TYPE_* $type (string) type of barcode * @param int $widthFactor Width of a single bar element in pixels. - * @param int $totalHeight Height of a single bar element in pixels. - * @param int|string $color Foreground color for bar elements (background is transparent). + * @param int $height Height of a single bar element in pixels. + * @param string $foregroundColor Foreground color for bar elements as '#333' or 'orange' for example (background is transparent). * @return string HTML code. - * @public + * @throws UnknownTypeException */ - public function getBarcode($code, $type, $widthFactor = 2, $totalHeight = 30, $color = 'black') + public function getBarcode(string $barcode, $type, int $widthFactor = 2, int $height = 30, string $foregroundColor = 'black'): string { - $barcodeData = $this->getBarcodeData($code, $type); - - $html = '
' . "\n"; - - $positionHorizontal = 0; - foreach ($barcodeData['bars'] as $bar) { - $barWidth = round(($bar['width'] * $widthFactor), 3); - $barHeight = round(($bar['height'] * $totalHeight / $barcodeData['maxHeight']), 3); - - if ($bar['drawBar']) { - $positionVertical = round(($bar['positionVertical'] * $totalHeight / $barcodeData['maxHeight']), 3); - // draw a vertical bar - $html .= '
 
' . "\n"; - } + $barcodeData = $this->getBarcodeData($barcode, $type); - $positionHorizontal += $barWidth; - } + $width = round(($barcodeData->getWidth() * $widthFactor), 3); - $html .= '
' . "\n"; + $renderer = new \Picqer\Barcode\Renderers\HtmlRenderer(); + $renderer->setForegroundColor(ColorHelper::getArrayFromColorString($foregroundColor)); - return $html; + return $renderer->render($barcodeData, $width, $height); } -} \ No newline at end of file +} diff --git a/src/BarcodeGeneratorJPG.php b/src/BarcodeGeneratorJPG.php index 9772226e..51eaf48e 100644 --- a/src/BarcodeGeneratorJPG.php +++ b/src/BarcodeGeneratorJPG.php @@ -2,76 +2,54 @@ namespace Picqer\Barcode; -use Picqer\Barcode\Exceptions\BarcodeException; +use Picqer\Barcode\Exceptions\UnknownTypeException; class BarcodeGeneratorJPG extends BarcodeGenerator { + protected ?bool $useImagick = null; /** - * Return a JPG image representation of barcode (requires GD or Imagick library). + * Return a PNG image representation of barcode (requires GD or Imagick library). * - * @param string $code code to print - * @param string $type type of barcode: + * @param string $barcode code to print + * @param BarcodeGenerator::TYPE_* $type (string) type of barcode * @param int $widthFactor Width of a single bar element in pixels. - * @param int $totalHeight Height of a single bar element in pixels. - * @param array $color RGB (0-255) foreground color for bar elements (background is transparent). + * @param int $height Height of a single bar element in pixels. + * @param array $foregroundColor RGB (0-255) foreground color for bar elements (background is transparent). * @return string image data or false in case of error. - * @public - * @throws BarcodeException + * @throws UnknownTypeException */ - public function getBarcode($code, $type, $widthFactor = 2, $totalHeight = 30, $color = array(0, 0, 0)) + public function getBarcode(string $barcode, $type, int $widthFactor = 2, int $height = 30, array $foregroundColor = [0, 0, 0]): string { - $barcodeData = $this->getBarcodeData($code, $type); + $barcodeData = $this->getBarcodeData($barcode, $type); - // calculate image size - $width = ($barcodeData['maxWidth'] * $widthFactor); - $height = $totalHeight; + $renderer = new \Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->setForegroundColor($foregroundColor); - if (function_exists('imagecreate')) { - // GD library - $imagick = false; - $jpg = imagecreate($width, $height); - $colorBackground = imagecolorallocate($jpg, 255, 255, 255); - imagecolortransparent($jpg, $colorBackground); - $colorForeground = imagecolorallocate($jpg, $color[0], $color[1], $color[2]); - } elseif (extension_loaded('imagick')) { - $imagick = true; - $colorForeground = new \imagickpixel('rgb(' . $color[0] . ',' . $color[1] . ',' . $color[2] . ')'); - $jpg = new \Imagick(); - $jpg->newImage($width, $height, 'white', 'jpg'); - $imageMagickObject = new \imagickdraw(); - $imageMagickObject->setFillColor($colorForeground); - } else { - throw new BarcodeException('Neither gd-lib or imagick are installed!'); - } - - // print bars - $positionHorizontal = 0; - foreach ($barcodeData['bars'] as $bar) { - $bw = round(($bar['width'] * $widthFactor), 3); - $bh = round(($bar['height'] * $totalHeight / $barcodeData['maxHeight']), 3); - if ($bar['drawBar']) { - $y = round(($bar['positionVertical'] * $totalHeight / $barcodeData['maxHeight']), 3); - // draw a vertical bar - if ($imagick && isset($imageMagickObject)) { - $imageMagickObject->rectangle($positionHorizontal, $y, ($positionHorizontal + $bw), ($y + $bh)); - } else { - imagefilledrectangle($jpg, $positionHorizontal, $y, ($positionHorizontal + $bw) - 1, ($y + $bh), - $colorForeground); - } + if (! is_null($this->useImagick)) { + if ($this->useImagick) { + $renderer->useImagick(); + } else { + $renderer->useGd(); } - $positionHorizontal += $bw; } - ob_start(); - if ($imagick && isset($imageMagickObject)) { - $jpg->drawImage($imageMagickObject); - echo $jpg; - } else { - imagejpeg($jpg); - imagedestroy($jpg); - } - $image = ob_get_clean(); - return $image; + return $renderer->render($barcodeData, $barcodeData->getWidth() * $widthFactor, $height); + } + + /** + * Force the use of Imagick image extension + */ + public function useImagick(): void + { + $this->useImagick = true; + } + + /** + * Force the use of the GD image library + */ + public function useGd(): void + { + $this->useImagick = false; } } diff --git a/src/BarcodeGeneratorPNG.php b/src/BarcodeGeneratorPNG.php index f7883922..f8f3d0c2 100644 --- a/src/BarcodeGeneratorPNG.php +++ b/src/BarcodeGeneratorPNG.php @@ -2,75 +2,54 @@ namespace Picqer\Barcode; -use Picqer\Barcode\Exceptions\BarcodeException; +use Picqer\Barcode\Exceptions\UnknownTypeException; class BarcodeGeneratorPNG extends BarcodeGenerator { + protected ?bool $useImagick = null; /** * Return a PNG image representation of barcode (requires GD or Imagick library). * - * @param string $code code to print - * @param string $type type of barcode: + * @param string $barcode code to print + * @param BarcodeGenerator::TYPE_* $type (string) type of barcode * @param int $widthFactor Width of a single bar element in pixels. - * @param int $totalHeight Height of a single bar element in pixels. - * @param array $color RGB (0-255) foreground color for bar elements (background is transparent). + * @param int $height Height of a single bar element in pixels. + * @param array $foregroundColor RGB (0-255) foreground color for bar elements (background is transparent). * @return string image data or false in case of error. - * @public + * @throws UnknownTypeException */ - public function getBarcode($code, $type, $widthFactor = 2, $totalHeight = 30, $color = array(0, 0, 0)) + public function getBarcode(string $barcode, $type, int $widthFactor = 2, int $height = 30, array $foregroundColor = [0, 0, 0]): string { - $barcodeData = $this->getBarcodeData($code, $type); + $barcodeData = $this->getBarcodeData($barcode, $type); - // calculate image size - $width = ($barcodeData['maxWidth'] * $widthFactor); - $height = $totalHeight; + $renderer = new \Picqer\Barcode\Renderers\PngRenderer(); + $renderer->setForegroundColor($foregroundColor); - if (function_exists('imagecreate')) { - // GD library - $imagick = false; - $png = imagecreate($width, $height); - $colorBackground = imagecolorallocate($png, 255, 255, 255); - imagecolortransparent($png, $colorBackground); - $colorForeground = imagecolorallocate($png, $color[0], $color[1], $color[2]); - } elseif (extension_loaded('imagick')) { - $imagick = true; - $colorForeground = new \imagickpixel('rgb(' . $color[0] . ',' . $color[1] . ',' . $color[2] . ')'); - $png = new \Imagick(); - $png->newImage($width, $height, 'none', 'png'); - $imageMagickObject = new \imagickdraw(); - $imageMagickObject->setFillColor($colorForeground); - } else { - throw new BarcodeException('Neither gd-lib or imagick are installed!'); - } - - // print bars - $positionHorizontal = 0; - foreach ($barcodeData['bars'] as $bar) { - $bw = round(($bar['width'] * $widthFactor), 3); - $bh = round(($bar['height'] * $totalHeight / $barcodeData['maxHeight']), 3); - if ($bar['drawBar']) { - $y = round(($bar['positionVertical'] * $totalHeight / $barcodeData['maxHeight']), 3); - // draw a vertical bar - if ($imagick && isset($imageMagickObject)) { - $imageMagickObject->rectangle($positionHorizontal, $y, ($positionHorizontal + $bw), ($y + $bh)); - } else { - imagefilledrectangle($png, $positionHorizontal, $y, ($positionHorizontal + $bw) - 1, ($y + $bh), - $colorForeground); - } + if (! is_null($this->useImagick)) { + if ($this->useImagick) { + $renderer->useImagick(); + } else { + $renderer->useGd(); } - $positionHorizontal += $bw; } - ob_start(); - if ($imagick && isset($imageMagickObject)) { - $png->drawImage($imageMagickObject); - echo $png; - } else { - imagepng($png); - imagedestroy($png); - } - $image = ob_get_clean(); - return $image; + return $renderer->render($barcodeData, $barcodeData->getWidth() * $widthFactor, $height); + } + + /** + * Force the use of Imagick image extension + */ + public function useImagick(): void + { + $this->useImagick = true; + } + + /** + * Force the use of the GD image library + */ + public function useGd(): void + { + $this->useImagick = false; } -} \ No newline at end of file +} diff --git a/src/BarcodeGeneratorSVG.php b/src/BarcodeGeneratorSVG.php index 01af454d..64b725ea 100644 --- a/src/BarcodeGeneratorSVG.php +++ b/src/BarcodeGeneratorSVG.php @@ -2,49 +2,32 @@ namespace Picqer\Barcode; +use Picqer\Barcode\Exceptions\UnknownTypeException; +use Picqer\Barcode\Helpers\ColorHelper; + class BarcodeGeneratorSVG extends BarcodeGenerator { - /** * Return a SVG string representation of barcode. * - * @param $code (string) code to print - * @param $type (const) type of barcode - * @param $widthFactor (int) Minimum width of a single bar in user units. - * @param $totalHeight (int) Height of barcode in user units. - * @param $color (string) Foreground color (in SVG format) for bar elements (background is transparent). + * @param $barcode (string) code to print + * @param BarcodeGenerator::TYPE_* $type (string) type of barcode + * @param $widthFactor (float) Minimum width of a single bar in user units. + * @param $height (float) Height of barcode in user units. + * @param $foregroundColor (string) Foreground color (in SVG format) for bar elements (background is transparent). * @return string SVG code. * @public + * @throws UnknownTypeException */ - public function getBarcode($code, $type, $widthFactor = 2, $totalHeight = 30, $color = 'black') + public function getBarcode(string $barcode, $type, float $widthFactor = 2, float $height = 30, string $foregroundColor = 'black'): string { - $barcodeData = $this->getBarcodeData($code, $type); - - // replace table for special characters - $repstr = array("\0" => '', '&' => '&', '<' => '<', '>' => '>'); + $barcodeData = $this->getBarcodeData($barcode, $type); - $width = round(($barcodeData['maxWidth'] * $widthFactor), 3); + $width = round(($barcodeData->getWidth() * $widthFactor), 3); - $svg = '' . "\n"; - $svg .= '' . "\n"; - $svg .= '' . "\n"; - $svg .= "\t" . '' . strtr($barcodeData['code'], $repstr) . '' . "\n"; - $svg .= "\t" . '' . "\n"; - // print bars - $positionHorizontal = 0; - foreach ($barcodeData['bars'] as $bar) { - $barWidth = round(($bar['width'] * $widthFactor), 3); - $barHeight = round(($bar['height'] * $totalHeight / $barcodeData['maxHeight']), 3); - if ($bar['drawBar']) { - $positionVertical = round(($bar['positionVertical'] * $totalHeight / $barcodeData['maxHeight']), 3); - // draw a vertical bar - $svg .= "\t\t" . '' . "\n"; - } - $positionHorizontal += $barWidth; - } - $svg .= "\t" . '' . "\n"; - $svg .= '' . "\n"; + $renderer = new \Picqer\Barcode\Renderers\SvgRenderer(); + $renderer->setForegroundColor(ColorHelper::getArrayFromColorString($foregroundColor)); - return $svg; + return $renderer->render($barcodeData, $width, $height); } } diff --git a/src/Exceptions/InvalidOptionException.php b/src/Exceptions/InvalidOptionException.php new file mode 100644 index 00000000..6a9b5ad8 --- /dev/null +++ b/src/Exceptions/InvalidOptionException.php @@ -0,0 +1,5 @@ +addBar(new BarcodeBar($barWidth, 1, $drawBar)); + $barWidth = 0; + } + } + + return $barcode; + } +} diff --git a/src/Helpers/ColorHelper.php b/src/Helpers/ColorHelper.php new file mode 100644 index 00000000..92121b3a --- /dev/null +++ b/src/Helpers/ColorHelper.php @@ -0,0 +1,46 @@ +backgroundColor ? ';background-color:rgb(' . implode(',', $this->backgroundColor) . ')' : '') . '">' . PHP_EOL; + + $positionHorizontal = 0; + /** @var BarcodeBar $bar */ + foreach ($barcode->getBars() as $bar) { + $barWidth = $bar->getWidth() / $barcode->getWidth() * 100; + $barHeight = round(($bar->getHeight() / $barcode->getHeight() * 100), 3); + + if ($bar->isBar() && $barWidth > 0) { + $positionVertical = round(($bar->getPositionVertical() / $barcode->getHeight() * 100), 3); + + // draw a vertical bar + $html .= '
 
' . PHP_EOL; + } + + $positionHorizontal += $barWidth; + } + + $html .= '' . PHP_EOL; + + return $html; + } + + public function setForegroundColor(array $color): self + { + $this->foregroundColor = $color; + return $this; + } + + public function setBackgroundColor(?array $color): self + { + $this->backgroundColor = $color; + return $this; + } +} diff --git a/src/Renderers/HtmlRenderer.php b/src/Renderers/HtmlRenderer.php new file mode 100644 index 00000000..9b93651d --- /dev/null +++ b/src/Renderers/HtmlRenderer.php @@ -0,0 +1,51 @@ +getWidth(); + + $html = '
' . PHP_EOL; + + $positionHorizontal = 0; + /** @var BarcodeBar $bar */ + foreach ($barcode->getBars() as $bar) { + $barWidth = round(($bar->getWidth() * $widthFactor), 3); + $barHeight = round(($bar->getHeight() * $height / $barcode->getHeight()), 3); + + if ($bar->isBar() && $barWidth > 0) { + $positionVertical = round(($bar->getPositionVertical() * $height / $barcode->getHeight()), 3); + + // draw a vertical bar + $html .= '
 
' . PHP_EOL; + } + + $positionHorizontal += $barWidth; + } + + $html .= '
' . PHP_EOL; + + return $html; + } + + public function setForegroundColor(array $color): self + { + $this->foregroundColor = $color; + return $this; + } + + public function setBackgroundColor(?array $color): self + { + $this->backgroundColor = $color; + return $this; + } +} diff --git a/src/Renderers/JpgRenderer.php b/src/Renderers/JpgRenderer.php new file mode 100644 index 00000000..ae2aca51 --- /dev/null +++ b/src/Renderers/JpgRenderer.php @@ -0,0 +1,30 @@ +backgroundColor !== null) { + // Colored background + $backgroundColor = new ImagickPixel('rgb(' . implode(',', $this->backgroundColor) . ')'); + } else { + // Use white background + $backgroundColor = new ImagickPixel('rgb(255,255,255)'); + } + $image->newImage($width, $height, $backgroundColor, 'JPG'); + + return $image; + } + + protected function generateGdImage($image): void + { + \imagejpeg($image); + \imagedestroy($image); + } +} diff --git a/src/Renderers/PngRenderer.php b/src/Renderers/PngRenderer.php new file mode 100644 index 00000000..72c29094 --- /dev/null +++ b/src/Renderers/PngRenderer.php @@ -0,0 +1,153 @@ +useImagick = true; + } elseif (function_exists('imagecreate')) { + $this->useImagick = false; + } else { + throw new BarcodeException('Neither gd-lib or imagick are installed!'); + } + } + + /** + * Force the use of Imagick image extension + */ + public function useImagick(): self + { + $this->useImagick = true; + return $this; + } + + /** + * Force the use of the GD image library + */ + public function useGd(): self + { + $this->useImagick = false; + return $this; + } + + // Floats in width and height will be rounded to integers + // For best (and valid) result, use a width as a factor of the width of the Barcode object + // Example: $width = $barcode->getWidth() * 3 + public function render(Barcode $barcode, float $width = 200, float $height = 30): string + { + $width = (int)round($width); + $height = (int)round($height); + + $widthFactor = $width / $barcode->getWidth(); + + if ($this->useImagick) { + $image = $this->createImagickImageObject($width, $height); + $imagickBarsShape = new ImagickDraw(); + $imagickBarsShape->setFillColor(new ImagickPixel('rgb(' . implode(',', $this->foregroundColor) .')')); + } else { + $image = $this->createGdImageObject($width, $height); + $gdForegroundColor = \imagecolorallocate($image, $this->foregroundColor[0], $this->foregroundColor[1], $this->foregroundColor[2]); + } + + // print bars + $positionHorizontal = 0; + /** @var BarcodeBar $bar */ + foreach ($barcode->getBars() as $bar) { + $barWidth = $bar->getWidth() * $widthFactor; + + if ($bar->isBar() && $barWidth > 0) { + $y = (int)round(($bar->getPositionVertical() * $height / $barcode->getHeight())); + $barHeight = (int)round(($bar->getHeight() * $height / $barcode->getHeight())); + + // draw a vertical bar + if ($this->useImagick) { + $imagickBarsShape->rectangle((int)round($positionHorizontal), $y, (int)round($positionHorizontal + $barWidth - 1), ($y + $barHeight)); + } else { + \imagefilledrectangle($image, (int)round($positionHorizontal), $y, (int)round($positionHorizontal + $barWidth - 1), ($y + $barHeight), $gdForegroundColor); + } + } + $positionHorizontal += $barWidth; + } + + if ($this->useImagick) { + $image->drawImage($imagickBarsShape); + return $image->getImageBlob(); + } else { + ob_start(); + $this->generateGdImage($image); + return ob_get_clean(); + } + } + + // Use RGB color definitions, like [0, 0, 0] or [255, 255, 255] + public function setForegroundColor(array $color): self + { + $this->foregroundColor = $color; + return $this; + } + + // Use RGB color definitions, like [0, 0, 0] or [255, 255, 255] + // If no color is set, the background will be transparent + public function setBackgroundColor(?array $color): self + { + $this->backgroundColor = $color; + return $this; + } + + protected function createGdImageObject(int $width, int $height) + { + $image = \imagecreate($width, $height); + + if ($this->backgroundColor !== null) { + // Colored background + $backgroundColor = \imagecolorallocate($image, $this->backgroundColor[0], $this->backgroundColor[1], $this->backgroundColor[2]); + \imagefill($image, 0, 0, $backgroundColor); + } else { + // Use transparent background + $backgroundColor = \imagecolorallocate($image, 255, 255, 255); + \imagecolortransparent($image, $backgroundColor); + } + + return $image; + } + + protected function createImagickImageObject(int $width, int $height): Imagick + { + $image = new Imagick(); + if ($this->backgroundColor !== null) { + // Colored background + $backgroundColor = new ImagickPixel('rgb(' . implode(',', $this->backgroundColor) . ')'); + } else { + // Use transparent background + $backgroundColor = new ImagickPixel('none'); + } + $image->newImage($width, $height, $backgroundColor, 'PNG'); + + return $image; + } + + protected function generateGdImage($image): void + { + \imagepng($image); + \imagedestroy($image); + } +} diff --git a/src/Renderers/RendererInterface.php b/src/Renderers/RendererInterface.php new file mode 100644 index 00000000..eaba33d6 --- /dev/null +++ b/src/Renderers/RendererInterface.php @@ -0,0 +1,14 @@ +getWidth(); + + $svg = ''; + if ($this->svgType === self::TYPE_SVG_STANDALONE) { + $svg .= '' . PHP_EOL; + $svg .= '' . PHP_EOL; + } + $svg .= '' . PHP_EOL; + $svg .= "\t" . '' . htmlspecialchars($barcode->getBarcode()) . '' . PHP_EOL; + + // Add background rectangle if backgroundColor is set + if ($this->backgroundColor !== null) { + $svg .= "\t" . '' . PHP_EOL; + } + + $svg .= "\t" . '' . PHP_EOL; + + // print bars + $positionHorizontal = 0; + /** @var BarcodeBar $bar */ + foreach ($barcode->getBars() as $bar) { + $barWidth = round(($bar->getWidth() * $widthFactor), 3); + $barHeight = round(($bar->getHeight() * $height / $barcode->getHeight()), 3); + + if ($bar->isBar() && $barWidth > 0) { + $positionVertical = round(($bar->getPositionVertical() * $height / $barcode->getHeight()), 3); + // draw a vertical bar + $svg .= "\t\t" . '' . PHP_EOL; + } + + $positionHorizontal += $barWidth; + } + + $svg .= "\t" . PHP_EOL; + $svg .= '' . PHP_EOL; + + return $svg; + } + + public function setForegroundColor(array $color): self + { + $this->foregroundColor = $color; + return $this; + } + + public function setBackgroundColor(?array $color): self + { + $this->backgroundColor = $color; + return $this; + } + + public function setSvgType(string $svgType): self + { + if (! in_array($svgType, [self::TYPE_SVG_INLINE, self::TYPE_SVG_STANDALONE])) { + throw new InvalidOptionException(); + } + + $this->svgType = $svgType; + return $this; + } +} diff --git a/src/Types/TypeCodabar.php b/src/Types/TypeCodabar.php new file mode 100644 index 00000000..e782dc4e --- /dev/null +++ b/src/Types/TypeCodabar.php @@ -0,0 +1,64 @@ + '11111221', + '1' => '11112211', + '2' => '11121121', + '3' => '22111111', + '4' => '11211211', + '5' => '21111211', + '6' => '12111121', + '7' => '12112111', + '8' => '12211111', + '9' => '21121111', + '-' => '11122111', + '$' => '11221111', + ':' => '21112121', + '/' => '21211121', + '.' => '21212111', + '+' => '11222221', + 'A' => '11221211', + 'B' => '12121121', + 'C' => '11121221', + 'D' => '11122211' + ]; + + public function getBarcode(string $code): Barcode + { + $barcode = new Barcode($code); + + $code = 'A' . strtoupper($code) . 'A'; + + for ($i = 0; $i < strlen($code); ++$i) { + if (! isset($this->conversionTable[(string)$code[$i]])) { + throw new InvalidCharacterException('Char ' . $code[$i] . ' is unsupported'); + } + + $seq = $this->conversionTable[(string)$code[$i]]; + for ($j = 0; $j < 8; ++$j) { + if (($j % 2) == 0) { + $drawBar = true; + } else { + $drawBar = false; + } + $barWidth = $seq[$j]; + $barcode->addBar(new BarcodeBar($barWidth, 1, $drawBar)); + } + } + + return $barcode; + } +} diff --git a/src/Types/TypeCode11.php b/src/Types/TypeCode11.php new file mode 100644 index 00000000..d70cf5f9 --- /dev/null +++ b/src/Types/TypeCode11.php @@ -0,0 +1,111 @@ + '111121', + '1' => '211121', + '2' => '121121', + '3' => '221111', + '4' => '112121', + '5' => '212111', + '6' => '122111', + '7' => '111221', + '8' => '211211', + '9' => '211111', + '-' => '112111', + 'S' => '112211', + ]; + + public function getBarcode(string $code): Barcode + { + $barcode = new Barcode($code); + + $code .= $this->getCheckDigitC($code); + $code .= $this->getCheckDigitK($code); + + $code = 'S' . $code . 'S'; + + for ($i = 0; $i < strlen($code); ++$i) { + if (! isset($this->conversionTable[$code[$i]])) { + throw new InvalidCharacterException('Char ' . $code[$i] . ' is unsupported'); + } + + $seq = $this->conversionTable[$code[$i]]; + for ($j = 0; $j < strlen($seq); ++$j) { + if (($j % 2) == 0) { + $drawBar = true; + } else { + $drawBar = false; + } + $barWidth = $seq[$j]; + + $barcode->addBar(new BarcodeBar($barWidth, 1, $drawBar)); + } + } + + return $barcode; + } + + private function getCheckDigitC(string $code): string + { + $weight = 1; + $checksum = 0; + for ($i = (strlen($code) - 1); $i >= 0; --$i) { + $digit = $code[$i]; + if ($digit == '-') { + $digitValue = 10; + } else { + $digitValue = intval($digit); + } + $checksum += ($digitValue * $weight); + ++$weight; + if ($weight > 10) { + $weight = 1; + } + } + $checksum %= 11; + if ($checksum == 10) { + $checksum = '-'; + } + + return $checksum; + } + + private function getCheckDigitK(string $code): string + { + if (strlen($code) <= 10) { + return ''; + } + + $weight = 1; + $checksum = 0; + for ($i = (strlen($code) - 1); $i >= 0; --$i) { + $digit = $code[$i]; + if ($digit == '-') { + $digitValue = 10; + } else { + $digitValue = intval($digit); + } + $checksum += ($digitValue * $weight); + ++$weight; + if ($weight > 9) { + $weight = 1; + } + } + $checksum %= 11; + + return (string)$checksum; + } +} diff --git a/src/Types/TypeCode128.php b/src/Types/TypeCode128.php new file mode 100644 index 00000000..5e330954 --- /dev/null +++ b/src/Types/TypeCode128.php @@ -0,0 +1,420 @@ +?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_'; + $keys_a .= chr(0) . chr(1) . chr(2) . chr(3) . chr(4) . chr(5) . chr(6) . chr(7) . chr(8) . chr(9); + $keys_a .= chr(10) . chr(11) . chr(12) . chr(13) . chr(14) . chr(15) . chr(16) . chr(17) . chr(18) . chr(19); + $keys_a .= chr(20) . chr(21) . chr(22) . chr(23) . chr(24) . chr(25) . chr(26) . chr(27) . chr(28) . chr(29); + $keys_a .= chr(30) . chr(31); + + // ASCII characters for code B (ASCII 32 - 127) + $keys_b = ' !"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~' . chr(127); + + // special codes + $fnc_a = [241 => 102, 242 => 97, 243 => 96, 244 => 101]; + $fnc_b = [241 => 102, 242 => 97, 243 => 96, 244 => 100]; + + // array of symbols + $code_data = []; + + // length of the code + $len = strlen($code); + + switch (strtoupper($this->type ?? "")) { + case 'A': + $startid = 103; + for ($i = 0; $i < $len; ++$i) { + $char = $code[$i]; + $char_id = ord($char); + if (($char_id >= 241) AND ($char_id <= 244)) { + $code_data[] = $fnc_a[$char_id]; + } elseif ($char_id <= 95) { + $code_data[] = strpos($keys_a, $char); + } else { + throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); + } + } + break; + + case 'B': + $startid = 104; + for ($i = 0; $i < $len; ++$i) { + $char = $code[$i]; + $char_id = ord($char); + if (($char_id >= 241) AND ($char_id <= 244)) { + $code_data[] = $fnc_b[$char_id]; + } elseif (($char_id >= 32) AND ($char_id <= 127)) { + $code_data[] = strpos($keys_b, $char); + } else { + throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); + } + } + break; + + case 'C': + $startid = 105; + if (ord($code[0]) == 241) { + $code_data[] = 102; + $code = substr($code, 1); + --$len; + } + if (($len % 2) != 0) { + throw new InvalidLengthException('Length must be even'); + } + for ($i = 0; $i < $len; $i += 2) { + $chrnum = $code[$i] . $code[$i + 1]; + if (preg_match('/([0-9]{2})/', $chrnum) > 0) { + $code_data[] = intval($chrnum); + } else { + throw new InvalidCharacterException(); + } + } + break; + + default: + // split code into sequences + $sequence = []; + // get numeric sequences (if any) + $numseq = []; + preg_match_all('/([0-9]{4,})/', $code, $numseq, PREG_OFFSET_CAPTURE); + if (! empty($numseq[1])) { + $end_offset = 0; + foreach ($numseq[1] as $val) { + $offset = $val[1]; + + // numeric sequence + $slen = strlen($val[0]); + if (($slen % 2) != 0) { + // the length must be even + ++$offset; + $val[0] = substr($val[0], 1); + } + + if ($offset > $end_offset) { + // non numeric sequence + $sequence = array_merge($sequence, + $this->get128ABsequence(substr($code, $end_offset, ($offset - $end_offset)))); + } + // numeric sequence fallback + $slen = strlen($val[0]); + if (($slen % 2) != 0) { + // the length must be even + --$slen; + } + $sequence[] = ['C', substr($code, $offset, $slen), $slen]; + $end_offset = $offset + $slen; + } + if ($end_offset < $len) { + $sequence = array_merge($sequence, $this->get128ABsequence(substr($code, $end_offset))); + } + } else { + // text code (non C mode) + $sequence = array_merge($sequence, $this->get128ABsequence($code)); + } + + // process the sequence + foreach ($sequence as $key => $seq) { + switch ($seq[0]) { + case 'A': + if ($key == 0) { + $startid = 103; + } elseif ($sequence[($key - 1)][0] != 'A') { + if (($seq[2] == 1) AND ($key > 0) AND ($sequence[($key - 1)][0] == 'B') AND (! isset($sequence[($key - 1)][3]))) { + // single character shift + $code_data[] = 98; + // mark shift + $sequence[$key][3] = true; + } elseif (! isset($sequence[($key - 1)][3])) { + $code_data[] = 101; + } + } + for ($i = 0; $i < $seq[2]; ++$i) { + $char = $seq[1][$i]; + $char_id = ord($char); + if (($char_id >= 241) AND ($char_id <= 244)) { + $code_data[] = $fnc_a[$char_id]; + } else { + $code_data[] = strpos($keys_a, $char); + } + } + break; + + case 'B': + if ($key == 0) { + $tmpchr = ord($seq[1][0]); + if (($seq[2] == 1) AND ($tmpchr >= 241) AND ($tmpchr <= 244) AND isset($sequence[($key + 1)]) AND ($sequence[($key + 1)][0] != 'B')) { + switch ($sequence[($key + 1)][0]) { + case 'A': + { + $startid = 103; + $sequence[$key][0] = 'A'; + $code_data[] = $fnc_a[$tmpchr]; + break; + } + case 'C': + { + $startid = 105; + $sequence[$key][0] = 'C'; + $code_data[] = $fnc_a[$tmpchr]; + break; + } + } + break; + } else { + $startid = 104; + } + } elseif ($sequence[($key - 1)][0] != 'B') { + if (($seq[2] == 1) AND ($key > 0) AND ($sequence[($key - 1)][0] == 'A') AND (! isset($sequence[($key - 1)][3]))) { + // single character shift + $code_data[] = 98; + // mark shift + $sequence[$key][3] = true; + } elseif (! isset($sequence[($key - 1)][3])) { + $code_data[] = 100; + } + } + for ($i = 0; $i < $seq[2]; ++$i) { + $char = $seq[1][$i]; + $char_id = ord($char); + if (($char_id >= 241) AND ($char_id <= 244)) { + $code_data[] = $fnc_b[$char_id]; + } else { + $code_data[] = strpos($keys_b, $char); + } + } + break; + + case 'C': + if ($key == 0) { + $startid = 105; + } elseif ($sequence[($key - 1)][0] != 'C') { + $code_data[] = 99; + } + for ($i = 0; $i < $seq[2]; $i += 2) { + $chrnum = $seq[1][$i] . $seq[1][$i + 1]; + $code_data[] = intval($chrnum); + } + break; + + default: + throw new InvalidCharacterException('Do not support different mode then A, B or C.'); + } + } + } + + // calculate check character + if (! isset($startid)) { + throw new BarcodeException('Could not determine start char for barcode.'); + } + + $sum = $startid; + foreach ($code_data as $key => $val) { + $sum += ($val * ($key + 1)); + } + // add check character + $code_data[] = ($sum % 103); + // add stop sequence + $code_data[] = 106; + $code_data[] = 107; + // add start code at the beginning + array_unshift($code_data, $startid); + + // build barcode array + $barcode = new Barcode($code); + foreach ($code_data as $val) { + $seq = $this->conversionTable[$val]; + for ($j = 0; $j < 6; ++$j) { + if (($j % 2) == 0) { + $t = true; // bar + } else { + $t = false; // space + } + $w = $seq[$j]; + + $barcode->addBar(new BarcodeBar($w, 1, $t)); + } + } + + return $barcode; + } + + + /** + * Split text code in A/B sequence for 128 code + * + * @param $code (string) code to split. + * @return array sequence + * @protected + */ + protected function get128ABsequence($code): array + { + $len = strlen($code); + $sequence = []; + // get A sequences (if any) + $numseq = []; + preg_match_all('/([\x00-\x1f])/', $code, $numseq, PREG_OFFSET_CAPTURE); + if (! empty($numseq[1])) { + $end_offset = 0; + foreach ($numseq[1] as $val) { + $offset = $val[1]; + if ($offset > $end_offset) { + // B sequence + $sequence[] = [ + 'B', + substr($code, $end_offset, ($offset - $end_offset)), + ($offset - $end_offset) + ]; + } + // A sequence + $slen = strlen($val[0]); + $sequence[] = ['A', substr($code, $offset, $slen), $slen]; + $end_offset = $offset + $slen; + } + if ($end_offset < $len) { + $sequence[] = ['B', substr($code, $end_offset), ($len - $end_offset)]; + } + } else { + // only B sequence + $sequence[] = ['B', $code, $len]; + } + + return $sequence; + } +} diff --git a/src/Types/TypeCode128A.php b/src/Types/TypeCode128A.php new file mode 100644 index 00000000..78bb46e5 --- /dev/null +++ b/src/Types/TypeCode128A.php @@ -0,0 +1,16 @@ + '0', + '1' => '1', + '2' => '2', + '3' => '3', + '4' => '4', + '5' => '5', + '6' => '6', + '7' => '7', + '8' => '8', + '9' => '9', + '10' => 'B', + '11' => 'C', + '12' => 'D', + '13' => 'F', + '14' => 'G', + '15' => 'H', + '16' => 'J', + '17' => 'K', + '18' => 'L', + '19' => 'M', + '20' => 'N', + '21' => 'P', + '22' => 'Q', + '23' => 'R', + '24' => 'S', + '25' => 'T', + '26' => 'U', + '27' => 'V', + '28' => 'W', + '29' => 'X', + '30' => 'Y', + '31' => 'Z' + ]; + + public function getBarcode(string $code): Barcode + { + // Validate code 32. + $stringLength = strlen($code); + + for ($i = 0; $i < $stringLength; ++$i) { + if (!is_numeric($code[$i])) { + throw new InvalidCharacterException('Character "' . $code[$i] . '" is not supported.'); + } + } + + // Prepare code 32. + $code = str_pad($code, 8, '0', STR_PAD_LEFT); + $checksumDigit = $this->checksum_code32(substr($code, 0, 8)); + $stringLength = max($stringLength, 8); + + if ($stringLength === 8) { + $code .= $checksumDigit; + ++$stringLength; + } + if ($stringLength !== 9) { + throw new InvalidLengthException('Only a code consisting of no more than 9 numbers is supported.'); + } + if ($code[8] !== $checksumDigit) { + throw new InvalidCheckDigitException('Provided checksum digit is wrong for provided code.'); + } + + // Convert code 32 into code 39. + $code39 = ''; + $codeElab = $code; + + for ($e = 5; $e >= 0; --$e) { + $code39 .= $this->conversionTable32[intval($codeElab / pow(32, $e))]; + $codeElab = $codeElab % pow(32, $e); + } + + // Return barcode data for code 39. + return parent::getBarcode($code39); + } + + + /** + * Calculate CODE 32 checksum (modulo 10). + * + * @param string $code code to represent. + * @return string char checksum. + * @protected + */ + protected function checksum_code32(string $code): string + { + $checksum = 0; + + foreach (str_split($code) as $i => $c) { + if (0 === $i % 2) { + $checksum += (int)$c; + } else { + $c = 2 * (int)$c; + $checksum += (int)floor($c / 10) + ($c % 10); + } + } + + return (string)($checksum % 10); + } +} diff --git a/src/Types/TypeCode39.php b/src/Types/TypeCode39.php new file mode 100644 index 00000000..bf169944 --- /dev/null +++ b/src/Types/TypeCode39.php @@ -0,0 +1,330 @@ + '111331311', + '1' => '311311113', + '2' => '113311113', + '3' => '313311111', + '4' => '111331113', + '5' => '311331111', + '6' => '113331111', + '7' => '111311313', + '8' => '311311311', + '9' => '113311311', + 'A' => '311113113', + 'B' => '113113113', + 'C' => '313113111', + 'D' => '111133113', + 'E' => '311133111', + 'F' => '113133111', + 'G' => '111113313', + 'H' => '311113311', + 'I' => '113113311', + 'J' => '111133311', + 'K' => '311111133', + 'L' => '113111133', + 'M' => '313111131', + 'N' => '111131133', + 'O' => '311131131', + 'P' => '113131131', + 'Q' => '111111333', + 'R' => '311111331', + 'S' => '113111331', + 'T' => '111131331', + 'U' => '331111113', + 'V' => '133111113', + 'W' => '333111111', + 'X' => '131131113', + 'Y' => '331131111', + 'Z' => '133131111', + '-' => '131111313', + '.' => '331111311', + ' ' => '133111311', + '$' => '131313111', + '/' => '131311131', + '+' => '131113131', + '%' => '111313131', + '*' => '131131311', + ]; + + public function getBarcode(string $code): Barcode + { + if (strlen(trim($code)) === 0) { + throw new InvalidLengthException('You should provide a barcode string.'); + } + + if ($this->extended) { + // extended mode + $code = $this->encode_code39_ext($code); + } + + if ($this->checksum) { + // checksum + $code .= $this->checksum_code39($code); + } + + // add start and stop codes + $code = '*' . $code . '*'; + + $barcode = new Barcode($code); + + for ($i = 0; $i < strlen($code); ++$i) { + $char = $code[$i]; + if (! isset($this->conversionTable[$char])) { + throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); + } + + for ($j = 0; $j < 9; ++$j) { + if (($j % 2) == 0) { + $drawBar = true; + } else { + $drawBar = false; + } + $barWidth = $this->conversionTable[$char][$j]; + $barcode->addBar(new BarcodeBar($barWidth, 1, $drawBar)); + } + + // inter character gap + $barcode->addBar(new BarcodeBar(1, 1, false)); + } + + return $barcode; + } + + + /** + * Encode a string to be used for CODE 39 Extended mode. + * + * @param string $code code to represent. + * @return string encoded string. + * @protected + * @throws InvalidCharacterException + */ + protected function encode_code39_ext(string $code): string + { + $encode = [ + chr(0) => '%U', + chr(1) => '$A', + chr(2) => '$B', + chr(3) => '$C', + chr(4) => '$D', + chr(5) => '$E', + chr(6) => '$F', + chr(7) => '$G', + chr(8) => '$H', + chr(9) => '$I', + chr(10) => '$J', + chr(11) => '$K', + chr(12) => '$L', + chr(13) => '$M', + chr(14) => '$N', + chr(15) => '$O', + chr(16) => '$P', + chr(17) => '$Q', + chr(18) => '$R', + chr(19) => '$S', + chr(20) => '$T', + chr(21) => '$U', + chr(22) => '$V', + chr(23) => '$W', + chr(24) => '$X', + chr(25) => '$Y', + chr(26) => '$Z', + chr(27) => '%A', + chr(28) => '%B', + chr(29) => '%C', + chr(30) => '%D', + chr(31) => '%E', + chr(32) => ' ', + chr(33) => '/A', + chr(34) => '/B', + chr(35) => '/C', + chr(36) => '/D', + chr(37) => '/E', + chr(38) => '/F', + chr(39) => '/G', + chr(40) => '/H', + chr(41) => '/I', + chr(42) => '/J', + chr(43) => '/K', + chr(44) => '/L', + chr(45) => '-', + chr(46) => '.', + chr(47) => '/O', + chr(48) => '0', + chr(49) => '1', + chr(50) => '2', + chr(51) => '3', + chr(52) => '4', + chr(53) => '5', + chr(54) => '6', + chr(55) => '7', + chr(56) => '8', + chr(57) => '9', + chr(58) => '/Z', + chr(59) => '%F', + chr(60) => '%G', + chr(61) => '%H', + chr(62) => '%I', + chr(63) => '%J', + chr(64) => '%V', + chr(65) => 'A', + chr(66) => 'B', + chr(67) => 'C', + chr(68) => 'D', + chr(69) => 'E', + chr(70) => 'F', + chr(71) => 'G', + chr(72) => 'H', + chr(73) => 'I', + chr(74) => 'J', + chr(75) => 'K', + chr(76) => 'L', + chr(77) => 'M', + chr(78) => 'N', + chr(79) => 'O', + chr(80) => 'P', + chr(81) => 'Q', + chr(82) => 'R', + chr(83) => 'S', + chr(84) => 'T', + chr(85) => 'U', + chr(86) => 'V', + chr(87) => 'W', + chr(88) => 'X', + chr(89) => 'Y', + chr(90) => 'Z', + chr(91) => '%K', + chr(92) => '%L', + chr(93) => '%M', + chr(94) => '%N', + chr(95) => '%O', + chr(96) => '%W', + chr(97) => '+A', + chr(98) => '+B', + chr(99) => '+C', + chr(100) => '+D', + chr(101) => '+E', + chr(102) => '+F', + chr(103) => '+G', + chr(104) => '+H', + chr(105) => '+I', + chr(106) => '+J', + chr(107) => '+K', + chr(108) => '+L', + chr(109) => '+M', + chr(110) => '+N', + chr(111) => '+O', + chr(112) => '+P', + chr(113) => '+Q', + chr(114) => '+R', + chr(115) => '+S', + chr(116) => '+T', + chr(117) => '+U', + chr(118) => '+V', + chr(119) => '+W', + chr(120) => '+X', + chr(121) => '+Y', + chr(122) => '+Z', + chr(123) => '%P', + chr(124) => '%Q', + chr(125) => '%R', + chr(126) => '%S', + chr(127) => '%T' + ]; + + $code_ext = ''; + for ($i = 0; $i < strlen($code); ++$i) { + if (ord($code[$i]) > 127) { + throw new InvalidCharacterException('Only supports till char 127'); + } + + $code_ext .= $encode[$code[$i]]; + } + + return $code_ext; + } + + + /** + * Calculate CODE 39 checksum (modulo 43). + * + * @param string $code code to represent. + * @return string char checksum. + * @protected + */ + protected function checksum_code39($code) + { + $chars = [ + '0', + '1', + '2', + '3', + '4', + '5', + '6', + '7', + '8', + '9', + 'A', + 'B', + 'C', + 'D', + 'E', + 'F', + 'G', + 'H', + 'I', + 'J', + 'K', + 'L', + 'M', + 'N', + 'O', + 'P', + 'Q', + 'R', + 'S', + 'T', + 'U', + 'V', + 'W', + 'X', + 'Y', + 'Z', + '-', + '.', + ' ', + '$', + '/', + '+', + '%' + ]; + + $checksum = 0; + for ($index = 0; $index < strlen($code); ++$index) { + $charPosition = array_keys($chars, $code[$index]); + $checksum += $charPosition[0]; + } + $checksumIndex = ($checksum % 43); + + return $chars[$checksumIndex]; + } +} diff --git a/src/Types/TypeCode39Checksum.php b/src/Types/TypeCode39Checksum.php new file mode 100644 index 00000000..7a47bb67 --- /dev/null +++ b/src/Types/TypeCode39Checksum.php @@ -0,0 +1,14 @@ + '131112', // 0 + 49 => '111213', // 1 + 50 => '111312', // 2 + 51 => '111411', // 3 + 52 => '121113', // 4 + 53 => '121212', // 5 + 54 => '121311', // 6 + 55 => '111114', // 7 + 56 => '131211', // 8 + 57 => '141111', // 9 + 65 => '211113', // A + 66 => '211212', // B + 67 => '211311', // C + 68 => '221112', // D + 69 => '221211', // E + 70 => '231111', // F + 71 => '112113', // G + 72 => '112212', // H + 73 => '112311', // I + 74 => '122112', // J + 75 => '132111', // K + 76 => '111123', // L + 77 => '111222', // M + 78 => '111321', // N + 79 => '121122', // O + 80 => '131121', // P + 81 => '212112', // Q + 82 => '212211', // R + 83 => '211122', // S + 84 => '211221', // T + 85 => '221121', // U + 86 => '222111', // V + 87 => '112122', // W + 88 => '112221', // X + 89 => '122121', // Y + 90 => '123111', // Z + 45 => '121131', // - + 46 => '311112', // . + 32 => '311211', // + 36 => '321111', // $ + 47 => '112131', // / + 43 => '113121', // + + 37 => '211131', // % + 97 => '121221', // ($) + 98 => '312111', // (%) + 99 => '311121', // (/) + 100 => '122211', // (+) + 42 => '111141', // start-stop + ]; + + public function getBarcode(string $code): Barcode + { + $encode = [ + chr(0) => 'bU', + chr(1) => 'aA', + chr(2) => 'aB', + chr(3) => 'aC', + chr(4) => 'aD', + chr(5) => 'aE', + chr(6) => 'aF', + chr(7) => 'aG', + chr(8) => 'aH', + chr(9) => 'aI', + chr(10) => 'aJ', + chr(11) => 'aK', + chr(12) => 'aL', + chr(13) => 'aM', + chr(14) => 'aN', + chr(15) => 'aO', + chr(16) => 'aP', + chr(17) => 'aQ', + chr(18) => 'aR', + chr(19) => 'aS', + chr(20) => 'aT', + chr(21) => 'aU', + chr(22) => 'aV', + chr(23) => 'aW', + chr(24) => 'aX', + chr(25) => 'aY', + chr(26) => 'aZ', + chr(27) => 'bA', + chr(28) => 'bB', + chr(29) => 'bC', + chr(30) => 'bD', + chr(31) => 'bE', + chr(32) => ' ', + chr(33) => 'cA', + chr(34) => 'cB', + chr(35) => 'cC', + chr(36) => '$', + chr(37) => '%', + chr(38) => 'cF', + chr(39) => 'cG', + chr(40) => 'cH', + chr(41) => 'cI', + chr(42) => 'cJ', + chr(43) => '+', + chr(44) => 'cL', + chr(45) => '-', + chr(46) => '.', + chr(47) => '/', + chr(48) => '0', + chr(49) => '1', + chr(50) => '2', + chr(51) => '3', + chr(52) => '4', + chr(53) => '5', + chr(54) => '6', + chr(55) => '7', + chr(56) => '8', + chr(57) => '9', + chr(58) => 'cZ', + chr(59) => 'bF', + chr(60) => 'bG', + chr(61) => 'bH', + chr(62) => 'bI', + chr(63) => 'bJ', + chr(64) => 'bV', + chr(65) => 'A', + chr(66) => 'B', + chr(67) => 'C', + chr(68) => 'D', + chr(69) => 'E', + chr(70) => 'F', + chr(71) => 'G', + chr(72) => 'H', + chr(73) => 'I', + chr(74) => 'J', + chr(75) => 'K', + chr(76) => 'L', + chr(77) => 'M', + chr(78) => 'N', + chr(79) => 'O', + chr(80) => 'P', + chr(81) => 'Q', + chr(82) => 'R', + chr(83) => 'S', + chr(84) => 'T', + chr(85) => 'U', + chr(86) => 'V', + chr(87) => 'W', + chr(88) => 'X', + chr(89) => 'Y', + chr(90) => 'Z', + chr(91) => 'bK', + chr(92) => 'bL', + chr(93) => 'bM', + chr(94) => 'bN', + chr(95) => 'bO', + chr(96) => 'bW', + chr(97) => 'dA', + chr(98) => 'dB', + chr(99) => 'dC', + chr(100) => 'dD', + chr(101) => 'dE', + chr(102) => 'dF', + chr(103) => 'dG', + chr(104) => 'dH', + chr(105) => 'dI', + chr(106) => 'dJ', + chr(107) => 'dK', + chr(108) => 'dL', + chr(109) => 'dM', + chr(110) => 'dN', + chr(111) => 'dO', + chr(112) => 'dP', + chr(113) => 'dQ', + chr(114) => 'dR', + chr(115) => 'dS', + chr(116) => 'dT', + chr(117) => 'dU', + chr(118) => 'dV', + chr(119) => 'dW', + chr(120) => 'dX', + chr(121) => 'dY', + chr(122) => 'dZ', + chr(123) => 'bP', + chr(124) => 'bQ', + chr(125) => 'bR', + chr(126) => 'bS', + chr(127) => 'bT', + ]; + + $code_ext = ''; + $clen = strlen($code); + for ($i = 0; $i < $clen; ++$i) { + if (ord($code[$i]) > 127) { + throw new InvalidCharacterException('Only supports till char 127'); + } + $code_ext .= $encode[$code[$i]]; + } + + // checksum + $code_ext .= $this->checksum_code93($code_ext); + + // add start and stop codes + $code = '*' . $code_ext . '*'; + + $barcode = new Barcode($code); + + for ($i = 0; $i < strlen($code); ++$i) { + $char = ord($code[$i]); + if (! isset($this->conversionTable[$char])) { + throw new InvalidCharacterException('Char ' . $char . ' is unsupported'); + } + + for ($j = 0; $j < 6; ++$j) { + if (($j % 2) == 0) { + $drawBar = true; + } else { + $drawBar = false; + } + $barWidth = $this->conversionTable[$char][$j]; + + $barcode->addBar(new BarcodeBar($barWidth, 1, $drawBar)); + } + } + + $barcode->addBar(new BarcodeBar(1, 1, true)); + + return $barcode; + } + + /** + * Calculate CODE 93 checksum (modulo 47). + * + * @param $code (string) code to represent. + * @return string checksum code. + * @protected + */ + protected function checksum_code93(string $code): string + { + $chars = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '-', '.', ' ', '$', '/', '+', '%', 'a', 'b', 'c', 'd']; + + // calculate check digit C + $codeLength = strlen($code); + $weight = 1; + $checksum = 0; + for ($i = ($codeLength - 1); $i >= 0; --$i) { + $charIndex = array_keys($chars, $code[$i]); + $checksum += ($charIndex[0] * $weight); + ++$weight; + if ($weight > 20) { + $weight = 1; + } + } + $checksumC = $checksum % 47; + $checkDigitC = $chars[$checksumC]; + $codeWithC = $code . $checkDigitC; + + // calculate check digit K + $weight = 1; + $checksum = 0; + for ($i = $codeLength; $i >= 0; --$i) { + $charIndex = array_keys($chars, $codeWithC[$i]); + $checksum += ($charIndex[0] * $weight); + ++$weight; + if ($weight > 15) { + $weight = 1; + } + } + $checksumK = $checksum % 47; + $checkDigitK = $chars[$checksumK]; + + return $checkDigitC . $checkDigitK; + } +} diff --git a/src/Types/TypeEan13.php b/src/Types/TypeEan13.php new file mode 100644 index 00000000..3f2fe75d --- /dev/null +++ b/src/Types/TypeEan13.php @@ -0,0 +1,20 @@ +length; + + $dataLength = $length - 1; + + // Add zero padding in front + $code = str_pad($code, $dataLength, '0', STR_PAD_LEFT); + + $checksumDigit = $this->calculateChecksumDigit($code); + + if (strlen($code) == $dataLength) { + $code .= $checksumDigit; + } elseif ($checksumDigit !== intval($code[$dataLength])) { + // If length of given barcode is same as final length, barcode is including checksum + // Make sure that checksum is the same as we calculated + throw new InvalidCheckDigitException(); + } + + if ($this->upca || $this->upce) { + $code = '0' . $code; + ++$length; + } + + if ($this->upce) { + // convert UPC-A to UPC-E + $tmp = substr($code, 4, 3); + if (($tmp == '000') OR ($tmp == '100') OR ($tmp == '200')) { + // manufacturer code ends in 000, 100, or 200 + $upce_code = substr($code, 2, 2) . substr($code, 9, 3) . substr($code, 4, 1); + } else { + $tmp = substr($code, 5, 2); + if ($tmp == '00') { + // manufacturer code ends in 00 + $upce_code = substr($code, 2, 3) . substr($code, 10, 2) . '3'; + } else { + $tmp = substr($code, 6, 1); + if ($tmp == '0') { + // manufacturer code ends in 0 + $upce_code = substr($code, 2, 4) . substr($code, 11, 1) . '4'; + } else { + // manufacturer code does not end in zero + $upce_code = substr($code, 2, 5) . substr($code, 11, 1); + } + } + } + } + + // Convert digits to bars + $codes = [ + 'A' => [ // left odd parity + '0' => '0001101', + '1' => '0011001', + '2' => '0010011', + '3' => '0111101', + '4' => '0100011', + '5' => '0110001', + '6' => '0101111', + '7' => '0111011', + '8' => '0110111', + '9' => '0001011' + ], + 'B' => [ // left even parity + '0' => '0100111', + '1' => '0110011', + '2' => '0011011', + '3' => '0100001', + '4' => '0011101', + '5' => '0111001', + '6' => '0000101', + '7' => '0010001', + '8' => '0001001', + '9' => '0010111' + ], + 'C' => [ // right + '0' => '1110010', + '1' => '1100110', + '2' => '1101100', + '3' => '1000010', + '4' => '1011100', + '5' => '1001110', + '6' => '1010000', + '7' => '1000100', + '8' => '1001000', + '9' => '1110100' + ] + ]; + + $parities = [ + '0' => ['A', 'A', 'A', 'A', 'A', 'A'], + '1' => ['A', 'A', 'B', 'A', 'B', 'B'], + '2' => ['A', 'A', 'B', 'B', 'A', 'B'], + '3' => ['A', 'A', 'B', 'B', 'B', 'A'], + '4' => ['A', 'B', 'A', 'A', 'B', 'B'], + '5' => ['A', 'B', 'B', 'A', 'A', 'B'], + '6' => ['A', 'B', 'B', 'B', 'A', 'A'], + '7' => ['A', 'B', 'A', 'B', 'A', 'B'], + '8' => ['A', 'B', 'A', 'B', 'B', 'A'], + '9' => ['A', 'B', 'B', 'A', 'B', 'A'], + ]; + + $upce_parities = [ + [ + '0' => ['B', 'B', 'B', 'A', 'A', 'A'], + '1' => ['B', 'B', 'A', 'B', 'A', 'A'], + '2' => ['B', 'B', 'A', 'A', 'B', 'A'], + '3' => ['B', 'B', 'A', 'A', 'A', 'B'], + '4' => ['B', 'A', 'B', 'B', 'A', 'A'], + '5' => ['B', 'A', 'A', 'B', 'B', 'A'], + '6' => ['B', 'A', 'A', 'A', 'B', 'B'], + '7' => ['B', 'A', 'B', 'A', 'B', 'A'], + '8' => ['B', 'A', 'B', 'A', 'A', 'B'], + '9' => ['B', 'A', 'A', 'B', 'A', 'B'], + ], + [ + '0' => ['A', 'A', 'A', 'B', 'B', 'B'], + '1' => ['A', 'A', 'B', 'A', 'B', 'B'], + '2' => ['A', 'A', 'B', 'B', 'A', 'B'], + '3' => ['A', 'A', 'B', 'B', 'B', 'A'], + '4' => ['A', 'B', 'A', 'A', 'B', 'B'], + '5' => ['A', 'B', 'B', 'A', 'A', 'B'], + '6' => ['A', 'B', 'B', 'B', 'A', 'A'], + '7' => ['A', 'B', 'A', 'B', 'A', 'B'], + '8' => ['A', 'B', 'A', 'B', 'B', 'A'], + '9' => ['A', 'B', 'B', 'A', 'B', 'A'], + ], + ]; + + $seq = '101'; // left guard bar + if ($this->upce) { + $barcode = new Barcode($upce_code); + $p = $upce_parities[$code[1]][$checksumDigit]; + for ($i = 0; $i < 6; ++$i) { + $seq .= $codes[$p[$i]][$upce_code[$i]]; + } + $seq .= '010101'; // right guard bar + } else { + $barcode = new Barcode($code); + $half_len = intval(ceil($length / 2)); + if ($length == 8) { + for ($i = 0; $i < $half_len; ++$i) { + $seq .= $codes['A'][$code[$i]]; + } + } else { + $p = $parities[$code[0]]; + for ($i = 1; $i < $half_len; ++$i) { + $seq .= $codes[$p[$i - 1]][$code[$i]]; + } + } + $seq .= '01010'; // center guard bar + for ($i = $half_len; $i < $length; ++$i) { + if (! isset($codes['C'][$code[$i]])) { + throw new InvalidCharacterException('Char ' . $code[$i] . ' not allowed'); + } + $seq .= $codes['C'][$code[$i]]; + } + $seq .= '101'; // right guard bar + } + + $clen = strlen($seq); + $w = 0; + for ($i = 0; $i < $clen; ++$i) { + $w += 1; + if (($i == ($clen - 1)) OR (($i < ($clen - 1)) AND ($seq[$i] != $seq[($i + 1)]))) { + if ($seq[$i] == '1') { + $t = true; // bar + } else { + $t = false; // space + } + + $barcode->addBar(new BarcodeBar($w, 1, $t)); + $w = 0; + } + } + + return $barcode; + } + + protected function calculateChecksumDigit(string $code): int + { + // calculate check digit + $sum_a = 0; + for ($i = 1; $i < $this->length - 1; $i += 2) { + $sum_a += intval($code[$i]); + } + if ($this->length > 12) { + $sum_a *= 3; + } + $sum_b = 0; + for ($i = 0; $i < $this->length - 1; $i += 2) { + $sum_b += intval(($code[$i])); + } + if ($this->length < 13) { + $sum_b *= 3; + } + $checksumDigit = ($sum_a + $sum_b) % 10; + if ($checksumDigit > 0) { + $checksumDigit = (10 - $checksumDigit); + } + + return $checksumDigit; + } +} diff --git a/src/Types/TypeITF14.php b/src/Types/TypeITF14.php new file mode 100644 index 00000000..d0fedc44 --- /dev/null +++ b/src/Types/TypeITF14.php @@ -0,0 +1,93 @@ + 14) { + throw new InvalidLengthException(); + } + + if (strlen($code) === 13) { + $code .= $this->getChecksum($code); + } elseif (substr($code, -1) !== $this->getChecksum(substr($code, 0, -1))) { + // If length of given barcode is same as final length, barcode is including checksum + // Make sure that checksum is the same as we calculated + throw new InvalidCheckDigitException(); + } + + $barcode = new Barcode($code); + + // Add start and stop codes + $code = 'AA' . strtolower($code) . 'ZA'; + + // Loop through 2 chars at once + for ($charIndex = 0; $charIndex < strlen($code); $charIndex += 2) { + if (! isset($chr[$code[$charIndex]]) || ! isset($chr[$code[$charIndex + 1]])) { + throw new InvalidCharacterException(); + } + + $drawBar = true; + $pbars = $chr[$code[$charIndex]]; + $pspaces = $chr[$code[$charIndex + 1]]; + $pmixed = ''; + + while (strlen($pbars) > 0) { + $pmixed .= $pbars[0] . $pspaces[0]; + $pbars = substr($pbars, 1); + $pspaces = substr($pspaces, 1); + } + + foreach (str_split($pmixed) as $width) { + $barcode->addBar(new BarcodeBar(intval($width), 1, $drawBar)); + $drawBar = ! $drawBar; + } + } + + return $barcode; + } + + private function getChecksum(string $code): string + { + $total = 0; + + for ($charIndex = 0; $charIndex <= (strlen($code) - 1); $charIndex++) { + $integerOfChar = intval($code[$charIndex]); + $total += $integerOfChar * ($charIndex % 2 === 0 ? 3 : 1); + } + + $checksum = 10 - ($total % 10); + if ($checksum === 10) { + $checksum = 0; + } + + return (string)$checksum; + } +} diff --git a/src/Types/TypeIntelligentMailBarcode.php b/src/Types/TypeIntelligentMailBarcode.php new file mode 100644 index 00000000..9160e142 --- /dev/null +++ b/src/Types/TypeIntelligentMailBarcode.php @@ -0,0 +1,571 @@ +
  • The Barcode Identifier shall be assigned by USPS to encode the + * presort identification that is currently printed in human readable form on the optional endorsement line (OEL) + * as well as for future USPS use. This shall be two digits, with the second digit in the range of 0–4. The + * allowable encoding ranges shall be 00–04, 10–14, 20–24, 30–34, 40–44, 50–54, 60–64, 70–74, 80–84, and + * 90–94.
  • The Service Type Identifier shall be assigned by USPS for any combination of services requested + * on the mailpiece. The allowable encoding range shall be 000http://it2.php.net/manual/en/function.dechex.php–999. + * Each 3-digit value shall correspond to a particular mail class with a particular combination of service(s). Each + * service program, such as OneCode Confirm and OneCode ACS, shall provide the list of Service Type Identifier + * values.
  • The Mailer or Customer Identifier shall be assigned by USPS as a unique, 6 or 9 digit number + * that identifies a business entity. The allowable encoding range for the 6 digit Mailer ID shall be 000000- + * 899999, while the allowable encoding range for the 9 digit Mailer ID shall be 900000000-999999999.
  • The + * Serial or Sequence Number shall be assigned by the mailer for uniquely identifying and tracking mailpieces. The + * allowable encoding range shall be 000000000–999999999 when used with a 6 digit Mailer ID and 000000-999999 when + * used with a 9 digit Mailer ID. e. The Delivery Point ZIP Code shall be assigned by the mailer for routing the + * mailpiece. This shall replace POSTNET for routing the mailpiece to its final delivery point. The length may be + * 0, 5, 9, or 11 digits. The allowable encoding ranges shall be no ZIP Code, 00000–99999, 000000000–999999999, + * and 00000000000–99999999999.
  • + * + * code to print, separate the ZIP (routing code) from the rest using a minus char '-' + * (BarcodeID_ServiceTypeID_MailerID_SerialNumber-RoutingCode) + */ + +class TypeIntelligentMailBarcode implements TypeInterface +{ + public function getBarcode(string $code): Barcode + { + $asc_chr = [ + 4, + 0, + 2, + 6, + 3, + 5, + 1, + 9, + 8, + 7, + 1, + 2, + 0, + 6, + 4, + 8, + 2, + 9, + 5, + 3, + 0, + 1, + 3, + 7, + 4, + 6, + 8, + 9, + 2, + 0, + 5, + 1, + 9, + 4, + 3, + 8, + 6, + 7, + 1, + 2, + 4, + 3, + 9, + 5, + 7, + 8, + 3, + 0, + 2, + 1, + 4, + 0, + 9, + 1, + 7, + 0, + 2, + 4, + 6, + 3, + 7, + 1, + 9, + 5, + 8 + ]; + $dsc_chr = [ + 7, + 1, + 9, + 5, + 8, + 0, + 2, + 4, + 6, + 3, + 5, + 8, + 9, + 7, + 3, + 0, + 6, + 1, + 7, + 4, + 6, + 8, + 9, + 2, + 5, + 1, + 7, + 5, + 4, + 3, + 8, + 7, + 6, + 0, + 2, + 5, + 4, + 9, + 3, + 0, + 1, + 6, + 8, + 2, + 0, + 4, + 5, + 9, + 6, + 7, + 5, + 2, + 6, + 3, + 8, + 5, + 1, + 9, + 8, + 7, + 4, + 0, + 2, + 6, + 3 + ]; + $asc_pos = [ + 3, + 0, + 8, + 11, + 1, + 12, + 8, + 11, + 10, + 6, + 4, + 12, + 2, + 7, + 9, + 6, + 7, + 9, + 2, + 8, + 4, + 0, + 12, + 7, + 10, + 9, + 0, + 7, + 10, + 5, + 7, + 9, + 6, + 8, + 2, + 12, + 1, + 4, + 2, + 0, + 1, + 5, + 4, + 6, + 12, + 1, + 0, + 9, + 4, + 7, + 5, + 10, + 2, + 6, + 9, + 11, + 2, + 12, + 6, + 7, + 5, + 11, + 0, + 3, + 2 + ]; + $dsc_pos = [ + 2, + 10, + 12, + 5, + 9, + 1, + 5, + 4, + 3, + 9, + 11, + 5, + 10, + 1, + 6, + 3, + 4, + 1, + 10, + 0, + 2, + 11, + 8, + 6, + 1, + 12, + 3, + 8, + 6, + 4, + 4, + 11, + 0, + 6, + 1, + 9, + 11, + 5, + 3, + 7, + 3, + 10, + 7, + 11, + 8, + 2, + 10, + 3, + 5, + 8, + 0, + 3, + 12, + 11, + 8, + 4, + 5, + 1, + 3, + 0, + 7, + 12, + 9, + 8, + 10 + ]; + $code_arr = explode('-', $code); + $tracking_number = $code_arr[0]; + if (isset($code_arr[1])) { + $routing_code = $code_arr[1]; + } else { + $routing_code = ''; + } + // Conversion of Routing Code + switch (strlen($routing_code)) { + case 0: + $binary_code = 0; + break; + + case 5: + $binary_code = bcadd($routing_code, '1'); + break; + + case 9: + $binary_code = bcadd($routing_code, '100001'); + break; + + case 11: + $binary_code = bcadd($routing_code, '1000100001'); + break; + + default: + throw new BarcodeException('Routing code unknown'); + } + + $binary_code = bcmul($binary_code, strval(10)); + $binary_code = bcadd($binary_code, $tracking_number[0]); + $binary_code = bcmul($binary_code, strval(5)); + $binary_code = bcadd($binary_code, $tracking_number[1]); + $binary_code .= substr($tracking_number, 2, 18); + + // convert to hexadecimal + $binary_code = $this->dec_to_hex($binary_code); + + // pad to get 13 bytes + $binary_code = str_pad($binary_code, 26, '0', STR_PAD_LEFT); + + // convert string to array of bytes + $binary_code_arr = chunk_split($binary_code, 2, "\r"); + $binary_code_arr = substr($binary_code_arr, 0, -1); + $binary_code_arr = explode("\r", $binary_code_arr); + + // calculate frame check sequence + $fcs = $this->imb_crc11fcs($binary_code_arr); + + // exclude first 2 bits from first byte + $first_byte = sprintf('%2s', dechex((hexdec($binary_code_arr[0]) << 2) >> 2)); + $binary_code_102bit = $first_byte . substr($binary_code, 2); + + // convert binary data to codewords + $codewords = []; + $data = $this->hex_to_dec($binary_code_102bit); + $codewords[0] = bcmod($data, strval(636)) * 2; + $data = bcdiv($data, strval(636)); + for ($i = 1; $i < 9; ++$i) { + $codewords[$i] = bcmod($data, strval(1365)); + $data = bcdiv($data, strval(1365)); + } + $codewords[9] = $data; + if (($fcs >> 10) == 1) { + $codewords[9] += 659; + } + + // generate lookup tables + $table2of13 = $this->imb_tables(2, 78); + $table5of13 = $this->imb_tables(5, 1287); + + // convert codewords to characters + $characters = []; + $bitmask = 512; + foreach ($codewords as $val) { + if ($val <= 1286) { + $chrcode = (int)$table5of13[$val]; + } else { + $chrcode = (int)$table2of13[($val - 1287)]; + } + if (($fcs & $bitmask) > 0) { + // bitwise invert + $chrcode = ((~$chrcode) & 8191); + } + $characters[] = $chrcode; + $bitmask /= 2; + } + $characters = array_reverse($characters); + + // build bars + $barcode = new Barcode($code); + for ($i = 0; $i < 65; ++$i) { + $asc = (($characters[$asc_chr[$i]] & pow(2, $asc_pos[$i])) > 0); + $dsc = (($characters[$dsc_chr[$i]] & pow(2, $dsc_pos[$i])) > 0); + if ($asc AND $dsc) { + // full bar (F) + $p = 0; + $h = 3; + } elseif ($asc) { + // ascender (A) + $p = 0; + $h = 2; + } elseif ($dsc) { + // descender (D) + $p = 1; + $h = 2; + } else { + // tracker (T) + $p = 1; + $h = 1; + } + $barcode->addBar(new BarcodeBar(1, $h, true, $p)); + if ($i < 64) { + $barcode->addBar(new BarcodeBar(1, 2, false, 0)); + } + } + + return $barcode; + } + + /** + * Convert large integer number to hexadecimal representation. + * (requires PHP bcmath extension) + * + * @param $number (string) number to convert specified as a string + * @return string hexadecimal representation + */ + protected function dec_to_hex($number) + { + if ($number == 0) { + return '00'; + } + + $hex = []; + + while ($number > 0) { + $hex[] = strtoupper(dechex(intval(bcmod(strval($number), '16')))); + $number = bcdiv($number, '16', 0); + } + $hex = array_reverse($hex); + + return implode($hex); + } + + + /** + * Intelligent Mail Barcode calculation of Frame Check Sequence + * + * @param $code_arr (string) array of hexadecimal values (13 bytes holding 102 bits right justified). + * @return int 11 bit Frame Check Sequence as integer (decimal base) + * @protected + */ + protected function imb_crc11fcs($code_arr) + { + $genpoly = 0x0F35; // generator polynomial + $fcs = 0x07FF; // Frame Check Sequence + // do most significant byte skipping the 2 most significant bits + $data = hexdec($code_arr[0]) << 5; + for ($bit = 2; $bit < 8; ++$bit) { + if (($fcs ^ $data) & 0x400) { + $fcs = ($fcs << 1) ^ $genpoly; + } else { + $fcs = ($fcs << 1); + } + $fcs &= 0x7FF; + $data <<= 1; + } + // do rest of bytes + for ($byte = 1; $byte < 13; ++$byte) { + $data = hexdec($code_arr[$byte]) << 3; + for ($bit = 0; $bit < 8; ++$bit) { + if (($fcs ^ $data) & 0x400) { + $fcs = ($fcs << 1) ^ $genpoly; + } else { + $fcs = ($fcs << 1); + } + $fcs &= 0x7FF; + $data <<= 1; + } + } + + return $fcs; + } + + /** + * Convert large hexadecimal number to decimal representation (string). + * (requires PHP bcmath extension) + * + * @param $hex (string) hexadecimal number to convert specified as a string + * @return string hexadecimal representation + */ + protected function hex_to_dec($hex) + { + $dec = 0; + $bitval = 1; + $len = strlen($hex); + for ($pos = ($len - 1); $pos >= 0; --$pos) { + $dec = bcadd($dec, bcmul(strval(hexdec($hex[$pos])), strval($bitval))); + $bitval = bcmul($bitval, '16'); + } + + return $dec; + } + + + /** + * generate Nof13 tables used for Intelligent Mail Barcode + * + * @param $n (int) is the type of table: 2 for 2of13 table, 5 for 5of13table + * @param $size (int) size of table (78 for n=2 and 1287 for n=5) + * @return array requested table + * @protected + */ + protected function imb_tables(int $n, int $size): array + { + $table = []; + $lli = 0; // LUT lower index + $lui = $size - 1; // LUT upper index + for ($count = 0; $count < 8192; ++$count) { + $bit_count = 0; + for ($bit_index = 0; $bit_index < 13; ++$bit_index) { + $bit_count += intval(($count & (1 << $bit_index)) != 0); + } + // if we don't have the right number of bits on, go on to the next value + if ($bit_count == $n) { + $reverse = ($this->imb_reverse_us($count) >> 3); + // if the reverse is less than count, we have already visited this pair before + if ($reverse >= $count) { + // If count is symmetric, place it at the first free slot from the end of the list. + // Otherwise, place it at the first free slot from the beginning of the list AND place $reverse ath the next free slot from the beginning of the list + if ($reverse == $count) { + $table[$lui] = $count; + --$lui; + } else { + $table[$lli] = $count; + ++$lli; + $table[$lli] = $reverse; + ++$lli; + } + } + } + } + + return $table; + } + + /** + * Reverse unsigned short value + * + * @param $num (int) value to reversr + * @return int reversed value + * @protected + */ + protected function imb_reverse_us($num) + { + $rev = 0; + for ($i = 0; $i < 16; ++$i) { + $rev <<= 1; + $rev |= ($num & 1); + $num >>= 1; + } + + return $rev; + } +} diff --git a/src/Types/TypeInterface.php b/src/Types/TypeInterface.php new file mode 100644 index 00000000..1fa946db --- /dev/null +++ b/src/Types/TypeInterface.php @@ -0,0 +1,10 @@ +getChecksum($code); + + if ((strlen($code) % 2) != 0) { + // add leading zero if code-length is odd + $code = '0' . $code; + } + + // add start and stop codes + $code = 'AA' . strtolower($code) . 'ZA'; + + $barcode = new Barcode($code); + for ($i = 0; $i < strlen($code); $i = ($i + 2)) { + $char_bar = $code[$i]; + $char_space = $code[$i + 1]; + if (! isset($chr[$char_bar]) || ! isset($chr[$char_space])) { + throw new InvalidCharacterException(); + } + + // create a bar-space sequence + $seq = ''; + $chrlen = strlen($chr[$char_bar]); + for ($s = 0; $s < $chrlen; $s++) { + $seq .= $chr[$char_bar][$s] . $chr[$char_space][$s]; + } + + for ($j = 0; $j < strlen($seq); ++$j) { + if (($j % 2) == 0) { + $t = true; // bar + } else { + $t = false; // space + } + $w = intval($seq[$j]); + $barcode->addBar(new BarcodeBar($w, 1, $t)); + } + } + + return $barcode; + } + + protected function getChecksum(string $code): string + { + $len = strlen($code); + $sum = 0; + for ($i = 0; $i < $len; $i += 2) { + $sum += intval($code[$i]); + } + $sum *= 3; + for ($i = 1; $i < $len; $i += 2) { + $sum += intval($code[$i]); + } + $r = $sum % 10; + if ($r > 0) { + $r = (10 - $r); + } + + return (string)$r; + } +} diff --git a/src/Types/TypeKix.php b/src/Types/TypeKix.php new file mode 100644 index 00000000..de20cf8b --- /dev/null +++ b/src/Types/TypeKix.php @@ -0,0 +1,16 @@ +checksum) { + // add checksum + $clen = strlen($code); + $p = 2; + $check = 0; + for ($i = ($clen - 1); $i >= 0; --$i) { + $check += (hexdec($code[$i]) * $p); + ++$p; + if ($p > 7) { + $p = 2; + } + } + $check %= 11; + if ($check > 0) { + $check = 11 - $check; + } + $code .= $check; + } + $seq = '110'; // left guard + $clen = strlen($code); + for ($i = 0; $i < $clen; ++$i) { + $digit = $code[$i]; + if (! isset($chr[$digit])) { + throw new InvalidCharacterException('Char ' . $digit . ' is unsupported'); + } + $seq .= $chr[$digit]; + } + $seq .= '1001'; // right guard + + return BinarySequenceConverter::convert($code, $seq); + } +} diff --git a/src/Types/TypePharmacode.php b/src/Types/TypePharmacode.php new file mode 100644 index 00000000..9054967f --- /dev/null +++ b/src/Types/TypePharmacode.php @@ -0,0 +1,38 @@ + 0) { + if (($code % 2) == 0) { + $seq .= '11100'; + $code -= 2; + } else { + $seq .= '100'; + $code -= 1; + } + $code /= 2; + } + + $seq = substr($seq, 0, -2); + $seq = strrev($seq); + + return BinarySequenceConverter::convert($originalCode, $seq); + } +} diff --git a/src/Types/TypePharmacodeTwoCode.php b/src/Types/TypePharmacodeTwoCode.php new file mode 100644 index 00000000..f71d2a16 --- /dev/null +++ b/src/Types/TypePharmacodeTwoCode.php @@ -0,0 +1,81 @@ +addBar(new BarcodeBar(1, $h, true, $p)); + if ($i < (strlen($seq) - 1)) { + $barcode->addBar(new BarcodeBar(1, 2, false, 0)); + } + } + + return $barcode; + } +} diff --git a/src/Types/TypePlanet.php b/src/Types/TypePlanet.php new file mode 100644 index 00000000..c3b7f5ed --- /dev/null +++ b/src/Types/TypePlanet.php @@ -0,0 +1,28 @@ + [1, 1, 2, 2, 2], + 1 => [2, 2, 2, 1, 1], + 2 => [2, 2, 1, 2, 1], + 3 => [2, 2, 1, 1, 2], + 4 => [2, 1, 2, 2, 1], + 5 => [2, 1, 2, 1, 2], + 6 => [2, 1, 1, 2, 2], + 7 => [1, 2, 2, 2, 1], + 8 => [1, 2, 2, 1, 2], + 9 => [1, 2, 1, 2, 2] + ]; +} diff --git a/src/Types/TypePostnet.php b/src/Types/TypePostnet.php new file mode 100644 index 00000000..0d235d86 --- /dev/null +++ b/src/Types/TypePostnet.php @@ -0,0 +1,69 @@ + [2, 2, 1, 1, 1], + 1 => [1, 1, 1, 2, 2], + 2 => [1, 1, 2, 1, 2], + 3 => [1, 1, 2, 2, 1], + 4 => [1, 2, 1, 1, 2], + 5 => [1, 2, 1, 2, 1], + 6 => [1, 2, 2, 1, 1], + 7 => [2, 1, 1, 1, 2], + 8 => [2, 1, 1, 2, 1], + 9 => [2, 1, 2, 1, 1] + ]; + + public function getBarcode(string $code): Barcode + { + $code = str_replace(['-', ' '], '', $code); + $len = strlen($code); + + $barcode = new Barcode($code); + + // calculate checksum + $sum = 0; + for ($i = 0; $i < $len; ++$i) { + $sum += intval($code[$i]); + } + $chkd = ($sum % 10); + if ($chkd > 0) { + $chkd = (10 - $chkd); + } + $code .= $chkd; + $len = strlen($code); + + // start bar + $barcode->addBar(new BarcodeBar(1, 2, true)); + $barcode->addBar(new BarcodeBar(1, 2, false)); + + for ($i = 0; $i < $len; ++$i) { + for ($j = 0; $j < 5; ++$j) { + $h = $this->barlen[$code[$i]][$j]; + $p = (int)floor(1 / $h); + $barcode->addBar(new BarcodeBar(1, (int)$h, true, $p)); + $barcode->addBar(new BarcodeBar(1, 2, false)); + } + } + + // end bar + $barcode->addBar(new BarcodeBar(1, 2, true)); + + return $barcode; + } +} diff --git a/src/Types/TypeRms4cc.php b/src/Types/TypeRms4cc.php new file mode 100644 index 00000000..684a6248 --- /dev/null +++ b/src/Types/TypeRms4cc.php @@ -0,0 +1,165 @@ + [3, 3, 2, 2], + '1' => [3, 4, 1, 2], + '2' => [3, 4, 2, 1], + '3' => [4, 3, 1, 2], + '4' => [4, 3, 2, 1], + '5' => [4, 4, 1, 1], + '6' => [3, 1, 4, 2], + '7' => [3, 2, 3, 2], + '8' => [3, 2, 4, 1], + '9' => [4, 1, 3, 2], + 'A' => [4, 1, 4, 1], + 'B' => [4, 2, 3, 1], + 'C' => [3, 1, 2, 4], + 'D' => [3, 2, 1, 4], + 'E' => [3, 2, 2, 3], + 'F' => [4, 1, 1, 4], + 'G' => [4, 1, 2, 3], + 'H' => [4, 2, 1, 3], + 'I' => [1, 3, 4, 2], + 'J' => [1, 4, 3, 2], + 'K' => [1, 4, 4, 1], + 'L' => [2, 3, 3, 2], + 'M' => [2, 3, 4, 1], + 'N' => [2, 4, 3, 1], + 'O' => [1, 3, 2, 4], + 'P' => [1, 4, 1, 4], + 'Q' => [1, 4, 2, 3], + 'R' => [2, 3, 1, 4], + 'S' => [2, 3, 2, 3], + 'T' => [2, 4, 1, 3], + 'U' => [1, 1, 4, 4], + 'V' => [1, 2, 3, 4], + 'W' => [1, 2, 4, 3], + 'X' => [2, 1, 3, 4], + 'Y' => [2, 1, 4, 3], + 'Z' => [2, 2, 3, 3] + ]; + + $code = strtoupper($code); + $len = strlen($code); + + $barcode = new Barcode($code); + + if (! $this->kix) { + // table for checksum calculation (row,col) + $checktable = [ + '0' => [1, 1], + '1' => [1, 2], + '2' => [1, 3], + '3' => [1, 4], + '4' => [1, 5], + '5' => [1, 0], + '6' => [2, 1], + '7' => [2, 2], + '8' => [2, 3], + '9' => [2, 4], + 'A' => [2, 5], + 'B' => [2, 0], + 'C' => [3, 1], + 'D' => [3, 2], + 'E' => [3, 3], + 'F' => [3, 4], + 'G' => [3, 5], + 'H' => [3, 0], + 'I' => [4, 1], + 'J' => [4, 2], + 'K' => [4, 3], + 'L' => [4, 4], + 'M' => [4, 5], + 'N' => [4, 0], + 'O' => [5, 1], + 'P' => [5, 2], + 'Q' => [5, 3], + 'R' => [5, 4], + 'S' => [5, 5], + 'T' => [5, 0], + 'U' => [0, 1], + 'V' => [0, 2], + 'W' => [0, 3], + 'X' => [0, 4], + 'Y' => [0, 5], + 'Z' => [0, 0] + ]; + + $row = 0; + $col = 0; + for ($i = 0; $i < $len; ++$i) { + $row += $checktable[$code[$i]][0]; + $col += $checktable[$code[$i]][1]; + } + $row %= 6; + $col %= 6; + $chk = array_keys($checktable, [$row, $col]); + $code .= $chk[0]; + ++$len; + + // start bar + $barcode->addBar(new BarcodeBar(1, 2, true)); + $barcode->addBar(new BarcodeBar(1, 2, false)); + } + + for ($i = 0; $i < $len; ++$i) { + for ($j = 0; $j < 4; ++$j) { + switch ($barmode[$code[$i]][$j]) { + case 1: + $p = 0; + $h = 2; + break; + + case 2: + $p = 0; + $h = 3; + break; + + case 3: + $p = 1; + $h = 1; + break; + + case 4: + $p = 1; + $h = 2; + break; + } + + $barcode->addBar(new BarcodeBar(1, $h, true, $p)); + $barcode->addBar(new BarcodeBar(1, 2, false)); + } + } + + if (! $this->kix) { + // stop bar + $barcode->addBar(new BarcodeBar(1, 3, true)); + } + + return $barcode; + } +} diff --git a/src/Types/TypeStandard2of5.php b/src/Types/TypeStandard2of5.php new file mode 100644 index 00000000..1445a3ca --- /dev/null +++ b/src/Types/TypeStandard2of5.php @@ -0,0 +1,74 @@ +checksum) { + // add checksum + $code .= $this->checksum_s25($code); + } + $seq = '11011010'; + + for ($i = 0; $i < strlen($code); ++$i) { + $digit = $code[$i]; + if (! isset($chr[$digit])) { + throw new InvalidCharacterException('Char ' . $digit . ' is unsupported'); + } + $seq .= $chr[$digit]; + } + $seq .= '1101011'; + + return BinarySequenceConverter::convert($code, $seq); + } + + /** + * Checksum for standard 2 of 5 barcodes. + * + * @param $code (string) code to process. + * @return string checksum. + * @protected + */ + protected function checksum_s25(string $code): string + { + $len = strlen($code); + $sum = 0; + for ($i = 0; $i < $len; $i += 2) { + $sum += intval($code[$i]); + } + $sum *= 3; + for ($i = 1; $i < $len; $i += 2) { + $sum += intval($code[$i]); + } + $r = $sum % 10; + if ($r > 0) { + $r = (10 - $r); + } + + return strval($r); + } +} diff --git a/src/Types/TypeStandard2of5Checksum.php b/src/Types/TypeStandard2of5Checksum.php new file mode 100644 index 00000000..c1b6fc52 --- /dev/null +++ b/src/Types/TypeStandard2of5Checksum.php @@ -0,0 +1,14 @@ + + * from Java implementation of Telepen by Robin Stuart + * at https://github.com/woo-j/OkapiBarcode which uses the + * Apache License 2.0 http://www.apache.org/licenses/LICENSE-2.0 + * + * Implements Telepen (also known as Telepen Alpha), and Telepen Numeric. + * + * Telepen can encode ASCII text input and includes a modulo-127 check digit. + * Telepen Numeric allows compression of numeric data into a Telepen symbol. Data + * can consist of pairs of numbers or pairs consisting of a numerical digit followed + * by an X character. Telepen Numeric also includes a mod-127 check digit. + */ + +namespace Picqer\Barcode\Types; + +use Picqer\Barcode\Barcode; +use Picqer\Barcode\BarcodeBar; +use Picqer\Barcode\Exceptions\InvalidFormatException; + +class TypeTelepen implements TypeInterface +{ + private const TELEPEN_START_CHAR = '_'; + private const TELEPEN_STOP_CHAR = 'z'; + + private const TELEPEN_ALPHA = 'alpha'; + private const TELEPEN_NUMERIC = 'numeric'; + + private array $telepen_lookup_table; + private string $mode; + + public function __construct($m = 'alpha') + { + $this->mode = self::TELEPEN_ALPHA; + if (strtolower($m) == 'numeric') { + $this->mode = self::TELEPEN_NUMERIC; + } + $this->createTelepenConversionTable(); + } + + public function getBarcode(string $code): Barcode + { + /* The stream we get from the telepen output gives us the + * width of alternating black/white stripes + */ + + $encoded = $this->encode($code); //binary string + $barcode = new Barcode($code); + + $drawBar = true; + for ($i = 0; $i < strlen($encoded); ++$i) { + $barWidth = $encoded[$i]; + $barcode->addBar(new BarcodeBar(intval($barWidth), 250, $drawBar)); + $drawBar = !$drawBar; //flip to other colour + } + + return $barcode; + } + + protected function encode($code) : string + { + if ($this->mode == self::TELEPEN_ALPHA) { + $result = $this->encodeAlpha($code); + } else { + $result = $this->encodeNumeric($code); + } + + return $result; + } + + protected function encodeAlpha($code) : string + { + // We aren't interested in the non-printable parts of the ASCII set + if ( + !preg_match('/[ -~]+/', $code) + ) { // everything from ASCII32-ASCII127 + throw new InvalidFormatException("Invalid characters in data"); + } + + $count = 0; + + /* other implementations use the byte-chr-int type equivalence to work + * with array indices in the conversion/lookup table. It's probably + * better to be more explicit with php, hence the use of ord and chr here. + */ + + // begin with start char + $dest = $this->telepen_lookup_table[ord(self::TELEPEN_START_CHAR)]; + + for ($i = 0; $i < strlen($code); $i++) { + //$ascii_code = ord(substr($code, $i, 1)); + $ascii_code = ord($code[$i]); + $dest .= ($this->telepen_lookup_table[$ascii_code]); + $count += $ascii_code; + } + + // Now add check and terminator + $check_digit = 127 - ($count % 127); + if ($check_digit == 127) { + $check_digit = 0; + } + + $dest .= $this->telepen_lookup_table[ord(strval($check_digit))]; + $dest .= $this->telepen_lookup_table[ord(self::TELEPEN_STOP_CHAR)]; // Stop + + return $dest; + } + + private function encodeNumeric(string $code) : string + { + + /* If input contains non-numeric or X, exit */ + if (!preg_match('/^[0-9X]+$/', $code)) { + throw new InvalidFormatException("Invalid characters in data"); + } + + /* If input is an odd length, exit */ + $t = ''; + if (strlen($code) % 2 > 0) { + throw new InvalidFormatException("There must be an even number of digits"); + } + + $count = 0; + $dest = $this->telepen_lookup_table[ord(self::TELEPEN_START_CHAR)]; // begin with the start character _ + + for ($i = 0; $i < strlen($code); $i += 2) { + $c1 = $code[$i]; + $c2 = $code[$i+1]; + /* Input nX is allowed, but Xn is not */ + if ($c1 == 'X') { + throw new InvalidFormatException("Invalid position of X in data"); + } + $glyph = null; + if ($c2 == 'X') { + $glyph = (ord($c1) - ord('0')) + 17; + } else { + $glyph = ((10 * (ord($c1) - ord('0'))) + (ord($c2) - ord('0'))) + 27; + } + $count += $glyph; + $dest .= $this->telepen_lookup_table[$glyph]; + } + + $check_digit = 127 - ($count % 127); + if ($check_digit == 127) { + $check_digit = 0; + } + + $dest .= $this->telepen_lookup_table[$check_digit]; + $dest .= $this->telepen_lookup_table[ord(self::TELEPEN_STOP_CHAR)]; // Stop + + return $dest; + } + + /** + * The table provides a representation of barcode patterns + * for each character in the ASCII set. from ASCII0-127 + * Each barcode starts with "_" - ASCII95 0x5F, + * and ends with "z" - ASCII122 0xFA. + * More information about Telepen symbology is available from + * https://v4l237.n3cdn1.secureserver.net/wp-content/uploads/2022/05/Barcode-Symbology-information-and-History.pdf + */ + private function createTelepenConversionTable(): void + { + $this->telepen_lookup_table = [ + "1111111111111111", "1131313111", "33313111", "1111313131", + "3111313111", "11333131", "13133131", "111111313111", + "31333111", "1131113131", "33113131", "1111333111", + "3111113131", "1113133111", "1311133111", "111111113131", + "3131113111", "11313331", "333331", "111131113111", + "31113331", "1133113111", "1313113111", "1111113331", + "31131331", "113111113111", "3311113111", "1111131331", + "311111113111", "1113111331", "1311111331", "11111111113111", + "31313311", "1131311131", "33311131", "1111313311", + "3111311131", "11333311", "13133311", "111111311131", + "31331131", "1131113311", "33113311", "1111331131", + "3111113311", "1113131131", "1311131131", "111111113311", + "3131111131", "1131131311", "33131311", "111131111131", + "3111131311", "1133111131", "1313111131", "111111131311", + "3113111311", "113111111131", "3311111131", "111113111311", + "311111111131", "111311111311", "131111111311", "11111111111131", + "3131311111", "11313133", "333133", "111131311111", + "31113133", "1133311111", "1313311111", "1111113133", + "313333", "113111311111", "3311311111", "11113333", + "311111311111", "11131333", "13111333", "11111111311111", + "31311133", "1131331111", "33331111", "1111311133", + "3111331111", "11331133", "13131133", "111111331111", + "3113131111", "1131111133", "33111133", "111113131111", + "3111111133", "111311131111", "131111131111", "111111111133", + "31311313", "113131111111", "3331111111", "1111311313", + "311131111111", "11331313", "13131313", "11111131111111", + "3133111111", "1131111313", "33111313", "111133111111", + "3111111313", "111313111111", "131113111111", "111111111313", + "313111111111", "1131131113", "33131113", "11113111111111", + "3111131113", "113311111111", "131311111111", "111111131113", + "3113111113", "11311111111111", "331111111111", "111113111113", + "31111111111111", "111311111113", "131111111113" + ]; + } +} diff --git a/src/Types/TypeTelepenNumeric.php b/src/Types/TypeTelepenNumeric.php new file mode 100644 index 00000000..519d077e --- /dev/null +++ b/src/Types/TypeTelepenNumeric.php @@ -0,0 +1,24 @@ + + * from Java implementation of Telepen by Robin Stuart + * at https://github.com/woo-j/OkapiBarcode which uses the + * Apache License 2.0 http://www.apache.org/licenses/LICENSE-2.0 + * + * Implements Telepen (also known as Telepen Alpha), and Telepen Numeric. + * + * Telepen can encode ASCII text input and includes a modulo-127 check digit. + * Telepen Numeric allows compression of numeric data into a Telepen symbol. Data + * can consist of pairs of numbers or pairs consisting of a numerical digit followed + * by an X character. Telepen Numeric also includes a mod-127 check digit. + */ + +namespace Picqer\Barcode\Types; + +class TypeTelepenNumeric extends TypeTelepen +{ + public function __construct() + { + parent::__construct('numeric'); + } +} diff --git a/src/Types/TypeUpcA.php b/src/Types/TypeUpcA.php new file mode 100644 index 00000000..11c21a8c --- /dev/null +++ b/src/Types/TypeUpcA.php @@ -0,0 +1,20 @@ +length; + + // Padding + $code = str_pad($code, $len, '0', STR_PAD_LEFT); + + // Calculate check digit + if ($len == 2) { + $r = (int)$code % 4; + } elseif ($len == 5) { + $r = (3 * intval($code[0] . $code[2] . $code[4])) + (9 * intval($code[1] . $code[3])); + $r %= 10; + } else { + throw new InvalidCheckDigitException(); + } + + // Convert digits to bars + $codes = [ + 'A' => [ // left odd parity + '0' => '0001101', + '1' => '0011001', + '2' => '0010011', + '3' => '0111101', + '4' => '0100011', + '5' => '0110001', + '6' => '0101111', + '7' => '0111011', + '8' => '0110111', + '9' => '0001011' + ], + 'B' => [ // left even parity + '0' => '0100111', + '1' => '0110011', + '2' => '0011011', + '3' => '0100001', + '4' => '0011101', + '5' => '0111001', + '6' => '0000101', + '7' => '0010001', + '8' => '0001001', + '9' => '0010111' + ] + ]; + + $parities = [ + 2 =>[ + '0' => ['A', 'A'], + '1' => ['A', 'B'], + '2' => ['B', 'A'], + '3' => ['B', 'B'] + ], + 5 => [ + '0' => ['B', 'B', 'A', 'A', 'A'], + '1' => ['B', 'A', 'B', 'A', 'A'], + '2' => ['B', 'A', 'A', 'B', 'A'], + '3' => ['B', 'A', 'A', 'A', 'B'], + '4' => ['A', 'B', 'B', 'A', 'A'], + '5' => ['A', 'A', 'B', 'B', 'A'], + '6' => ['A', 'A', 'A', 'B', 'B'], + '7' => ['A', 'B', 'A', 'B', 'A'], + '8' => ['A', 'B', 'A', 'A', 'B'], + '9' => ['A', 'A', 'B', 'A', 'B'] + ] + ]; + + $p = $parities[$len][$r]; + $seq = '1011'; // left guard bar + $seq .= $codes[$p[0]][$code[0]]; + for ($i = 1; $i < $len; ++$i) { + $seq .= '01'; // separator + $seq .= $codes[$p[$i]][$code[$i]]; + } + + return BinarySequenceConverter::convert($code, $seq); + } +} diff --git a/src/Types/TypeUpcExtension5.php b/src/Types/TypeUpcExtension5.php new file mode 100644 index 00000000..c9e3065a --- /dev/null +++ b/src/Types/TypeUpcExtension5.php @@ -0,0 +1,14 @@ +getBarcode('081231723897', $generator::TYPE_CODE_128); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-dynamic-code128.html', $generated); + } + + public function test_dynamic_html_barcode_generator_can_generate_imb_barcode_to_test_heights() + { + $generator = new Picqer\Barcode\BarcodeGeneratorDynamicHTML(); + $generated = $generator->getBarcode('12345678903', $generator::TYPE_IMB); + + $this->assertStringEqualsFile('tests/verified-files/12345678903-dynamic-imb.html', $generated); + } +} diff --git a/tests/BarcodeHtmlTest.php b/tests/BarcodeHtmlTest.php new file mode 100644 index 00000000..ecdf6177 --- /dev/null +++ b/tests/BarcodeHtmlTest.php @@ -0,0 +1,22 @@ +getBarcode('081231723897', $generator::TYPE_CODE_128); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-code128.html', $generated); + } + + public function test_html_barcode_generator_can_generate_imb_barcode_to_test_heights() + { + $generator = new Picqer\Barcode\BarcodeGeneratorHTML(); + $generated = $generator->getBarcode('12345678903', $generator::TYPE_IMB); + + $this->assertStringEqualsFile('tests/verified-files/12345678903-imb.html', $generated); + } +} diff --git a/tests/BarcodeJpgTest.php b/tests/BarcodeJpgTest.php new file mode 100644 index 00000000..d597f4ad --- /dev/null +++ b/tests/BarcodeJpgTest.php @@ -0,0 +1,135 @@ +useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_generate_code_39_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_39, 1); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_height() + { + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 2, 45); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_width_factor() + { + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 5); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + + // Copied as Imagick + + public function test_jpg_barcode_generator_can_generate_code_128_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_generate_code_39_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_39, 1); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_height_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 2, 45); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_width_factor_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 5); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } +} diff --git a/tests/BarcodePngTest.php b/tests/BarcodePngTest.php new file mode 100644 index 00000000..700baa26 --- /dev/null +++ b/tests/BarcodePngTest.php @@ -0,0 +1,125 @@ +useGd(); + $generated = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); + + $this->assertEquals('PNG', substr($generated, 1, 3)); + } + + public function test_png_barcode_generator_can_generate_code_39_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_39, 1); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_height() + { + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 2, 45); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_width_factor() + { + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useGd(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 5); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/png', $imageInfo['mime']); + } + + + // Copied as Imagick + + public function test_png_barcode_generator_can_generate_code_128_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useImagick(); + $generated = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); + + $this->assertEquals('PNG', substr($generated, 1, 3)); + } + + public function test_png_barcode_generator_can_generate_code_39_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_39, 1); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_height_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 2, 45); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_width_factor_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $generator = new Picqer\Barcode\BarcodeGeneratorPNG(); + $generator->useImagick(); + $result = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128, 5); + + $imageInfo = getimagesizefromstring($result); + + $this->assertGreaterThan(100, strlen($result)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/png', $imageInfo['mime']); + } +} diff --git a/tests/BarcodeSvgTest.php b/tests/BarcodeSvgTest.php new file mode 100644 index 00000000..5a080ffa --- /dev/null +++ b/tests/BarcodeSvgTest.php @@ -0,0 +1,22 @@ +getBarcode('081231723897', $generator::TYPE_EAN_13); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13.svg', $generated); + } + + public function test_svg_barcode_generator_can_generate_ean_13_barcode_with_fractional_width() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generated = $generator->getBarcode('081231723897', $generator::TYPE_EAN_13, 0.25, 25.75); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13-fractional-width.svg', $generated); + } +} diff --git a/tests/BarcodeTest.php b/tests/BarcodeTest.php deleted file mode 100644 index 8f532b89..00000000 --- a/tests/BarcodeTest.php +++ /dev/null @@ -1,109 +0,0 @@ -getBarcode('081231723897', $generator::TYPE_CODE_128); - - $this->assertEquals('PNG', substr($generated, 1, 3)); - } - - /** - * @test - */ - public function svg_barcode_generator_can_generate_ean_13_barcode() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generated = $generator->getBarcode('081231723897', $generator::TYPE_EAN_13); - - $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13.svg', $generated); - } - - /** - * @test - */ - public function html_barcode_generator_can_generate_code_128_barcode() - { - $generator = new Picqer\Barcode\BarcodeGeneratorHTML(); - $generated = $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); - - $this->assertStringEqualsFile('tests/verified-files/081231723897-code128.html', $generated); - } - - /** - * @test - */ - public function jpg_barcode_generator_can_generate_code_128_barcode() - { - $generator = new Picqer\Barcode\BarcodeGeneratorJPG(); - $generator->getBarcode('081231723897', $generator::TYPE_CODE_128); - } - - /** - * @test - * @expectedException \Picqer\Barcode\Exceptions\InvalidCharacterException - */ - public function ean13_generator_throws_exception_if_invalid_chars_are_used() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generator->getBarcode('A123', $generator::TYPE_EAN_13); - } - - /** - * @test - */ - public function ean13_generator_accepting_13_chars() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generated = $generator->getBarcode('0049000004632', $generator::TYPE_EAN_13); - - $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); - } - - /** - * @test - */ - public function ean13_generator_accepting_12_chars_and_generates_13th_check_digit() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generated = $generator->getBarcode('004900000463', $generator::TYPE_EAN_13); - - $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); - } - - /** - * @test - */ - public function ean13_generator_accepting_11_chars_and_generates_13th_check_digit_and_adds_leading_zero() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generated = $generator->getBarcode('04900000463', $generator::TYPE_EAN_13); - - $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); - } - - /** - * @test - * @expectedException \Picqer\Barcode\Exceptions\InvalidCheckDigitException - */ - public function ean13_generator_throws_exception_when_wrong_check_digit_is_given() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generator->getBarcode('0049000004633', $generator::TYPE_EAN_13); - } - - /** - * @test - * @expectedException \Picqer\Barcode\Exceptions\UnknownTypeException - */ - public function generator_throws_unknown_type_exceptions() - { - $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); - $generator->getBarcode('0049000004633', 'vladimir'); - } -} \ No newline at end of file diff --git a/tests/ChecksumBarcodeTest.php b/tests/ChecksumBarcodeTest.php new file mode 100644 index 00000000..c9daf2d2 --- /dev/null +++ b/tests/ChecksumBarcodeTest.php @@ -0,0 +1,60 @@ + TypeEan13::class, 'barcodes' => [ + '081231723897' => '0812317238973', + '004900000463' => '0049000004632', + ]], + ['type' => TypeITF14::class, 'barcodes' => [ + '0001234560001' => '00012345600012', + '0540014128876' => '05400141288766', + ]], + ]; + private const INVALID_BARCODES = [ + ['type' => TypeEan13::class, 'barcodes' => ['0812317238972', '0049000004633']], + ['type' => TypeITF14::class, 'barcodes' => ['00012345600016', '05400141288762']], + ]; + + public function testAllValidBarcodeTypes() + { + foreach (self::VALID_BARCODES as $barcodeTestSet) { + $barcodeType = $this->getBarcodeType($barcodeTestSet['type']); + + foreach ($barcodeTestSet['barcodes'] as $testBarcode => $validBarcode) { + $this->assertEquals($validBarcode, $barcodeType->getBarcode($testBarcode)->getBarcode()); + } + } + } + + public function testAllInvalidBarcodeTypes() + { + foreach (self::INVALID_BARCODES as $barcodeTestSet) { + $barcodeType = $this->getBarcodeType($barcodeTestSet['type']); + + foreach ($barcodeTestSet['barcodes'] as $invalidBarcode) { + try { + $barcodeType->getBarcode($invalidBarcode)->getBarcode(); + } catch (BarcodeException $exception) { + $this->assertInstanceOf(InvalidCheckDigitException::class, $exception); + continue; + } + + $this->assertTrue(false, sprintf('Exception was not thrown for barcode "%s".', $invalidBarcode)); + } + } + } + + private function getBarcodeType(string $typeClass): TypeInterface + { + return new $typeClass; + } +} diff --git a/tests/DynamicHtmlRendererTest.php b/tests/DynamicHtmlRendererTest.php new file mode 100644 index 00000000..9597874d --- /dev/null +++ b/tests/DynamicHtmlRendererTest.php @@ -0,0 +1,26 @@ +getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\DynamicHtmlRenderer(); + $generated = $renderer->render($barcode); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-dynamic-code128.html', $generated); + } + + public function test_dynamic_html_barcode_generator_can_generate_imb_barcode_to_test_heights() + { + $barcode = (new Picqer\Barcode\Types\TypeIntelligentMailBarcode())->getBarcode('12345678903'); + + $renderer = new Picqer\Barcode\Renderers\DynamicHtmlRenderer(); + $generated = $renderer->render($barcode); + + $this->assertStringEqualsFile('tests/verified-files/12345678903-dynamic-imb.html', $generated); + } +} diff --git a/tests/GeneratorTest.php b/tests/GeneratorTest.php new file mode 100644 index 00000000..293efc09 --- /dev/null +++ b/tests/GeneratorTest.php @@ -0,0 +1,70 @@ +expectException(Picqer\Barcode\Exceptions\InvalidLengthException::class); + + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generator->getBarcode('', $generator::TYPE_EAN_13); + } + + public function test_throws_exception_if_empty_barcode_is_used_in_code128() + { + $this->expectException(Picqer\Barcode\Exceptions\InvalidLengthException::class); + + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generator->getBarcode('', $generator::TYPE_CODE_128); + } + + public function test_ean13_generator_throws_exception_if_invalid_chars_are_used() + { + $this->expectException(Picqer\Barcode\Exceptions\InvalidCharacterException::class); + + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generator->getBarcode('A123', $generator::TYPE_EAN_13); + } + + public function test_ean13_generator_accepting_13_chars() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generated = $generator->getBarcode('0049000004632', $generator::TYPE_EAN_13); + + $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); + } + + public function test_ean13_generator_accepting_12_chars_and_generates_13th_check_digit() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generated = $generator->getBarcode('004900000463', $generator::TYPE_EAN_13); + + $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); + } + + public function test_ean13_generator_accepting_11_chars_and_generates_13th_check_digit_and_adds_leading_zero() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generated = $generator->getBarcode('04900000463', $generator::TYPE_EAN_13); + + $this->assertStringEqualsFile('tests/verified-files/0049000004632-ean13.svg', $generated); + } + + public function test_ean13_generator_throws_exception_when_wrong_check_digit_is_given() + { + $this->expectException(Picqer\Barcode\Exceptions\InvalidCheckDigitException::class); + + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generator->getBarcode('0049000004633', $generator::TYPE_EAN_13); + } + + public function test_generator_throws_unknown_type_exceptions() + { + $this->expectException(Picqer\Barcode\Exceptions\UnknownTypeException::class); + + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $generator->getBarcode('0049000004633', 'vladimir'); + } +} \ No newline at end of file diff --git a/tests/Helpers/ColorHelperTest.php b/tests/Helpers/ColorHelperTest.php new file mode 100644 index 00000000..1fa483b3 --- /dev/null +++ b/tests/Helpers/ColorHelperTest.php @@ -0,0 +1,45 @@ +assertEquals([0, 0, 0], $result); + } + + public function test_convert_color_hex_345() + { + $result = ColorHelper::getArrayFromColorString('#345'); + + $this->assertEquals([51, 68, 85], $result); + } + + public function test_convert_color_hex_334455() + { + $result = ColorHelper::getArrayFromColorString('#334455'); + + $this->assertEquals([51, 68, 85], $result); + } + + public function test_convert_color_hex_632343() + { + $result = ColorHelper::getArrayFromColorString('#632343'); + + $this->assertEquals([99, 35, 67], $result); + } + + public function test_throws_exception_on_unknown_color() + { + $this->expectException(UnknownColorException::class); + + ColorHelper::getArrayFromColorString('aubergine'); + } +} diff --git a/tests/HtmlRendererTest.php b/tests/HtmlRendererTest.php new file mode 100644 index 00000000..2435886d --- /dev/null +++ b/tests/HtmlRendererTest.php @@ -0,0 +1,37 @@ +getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\HtmlRenderer(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-code128.html', $generated); + } + + public function test_html_barcode_generator_can_generate_imb_barcode_to_test_heights() + { + $barcode = (new Picqer\Barcode\Types\TypeIntelligentMailBarcode())->getBarcode('12345678903'); + + $renderer = new Picqer\Barcode\Renderers\HtmlRenderer(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile('tests/verified-files/12345678903-imb.html', $generated); + } + + public function test_html_barcode_generator_with_background() + { + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\HtmlRenderer(); + $renderer->setBackgroundColor([255, 0, 0]); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-code128-red-background.html', $generated); + } +} diff --git a/tests/JpgRendererTest.php b/tests/JpgRendererTest.php new file mode 100644 index 00000000..c6ca4c6d --- /dev/null +++ b/tests/JpgRendererTest.php @@ -0,0 +1,152 @@ +getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + + public function test_jpg_barcode_generator_can_generate_code_39_barcode() + { + $barcode = (new Picqer\Barcode\Types\TypeCode39())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth()); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_height() + { + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2, 45); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_width_factor() + { + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 5); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + + // Copied as Imagick + + public function test_jpg_barcode_generator_can_generate_code_128_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useImagick(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_generate_code_39_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode39())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useImagick(); + $generated = $renderer->render($barcode, $barcode->getWidth()); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_height_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2, 45); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } + + public function test_jpg_barcode_generator_can_use_different_width_factor_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\JpgRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 5); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/jpeg', $imageInfo['mime']); + } +} diff --git a/tests/PharmacodeTest.php b/tests/PharmacodeTest.php new file mode 100644 index 00000000..e2f4ed5d --- /dev/null +++ b/tests/PharmacodeTest.php @@ -0,0 +1,15 @@ +expectException(Picqer\Barcode\Exceptions\InvalidLengthException::class); + + $pharmacode->getBarcode('0'); + } +} diff --git a/tests/PngRendererTest.php b/tests/PngRendererTest.php new file mode 100644 index 00000000..fbb3e3cc --- /dev/null +++ b/tests/PngRendererTest.php @@ -0,0 +1,142 @@ +getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode); + + $this->assertEquals('PNG', substr($generated, 1, 3)); + } + + + public function test_png_barcode_generator_can_generate_code_39_barcode() + { + $barcode = (new Picqer\Barcode\Types\TypeCode39())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth()); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_height() + { + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2, 45); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_width_factor() + { + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 5); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/png', $imageInfo['mime']); + } + + + // Copied as Imagick + + public function test_png_barcode_generator_can_generate_code_128_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useImagick(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertEquals('PNG', substr($generated, 1, 3)); + } + + public function test_png_barcode_generator_can_generate_code_39_barcode_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode39())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useImagick(); + $generated = $renderer->render($barcode, $barcode->getWidth()); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(224, $imageInfo[0]); // Image width + $this->assertEquals(30, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_height_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 2, 45); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(202, $imageInfo[0]); // Image width + $this->assertEquals(45, $imageInfo[1]); // Image height + $this->assertEquals('image/png', $imageInfo['mime']); + } + + public function test_png_barcode_generator_can_use_different_width_factor_imagick() + { + if (! extension_loaded('imagick')) { + $this->markTestSkipped(); + } + + $barcode = (new Picqer\Barcode\Types\TypeCode128())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\PngRenderer(); + $renderer->useGd(); + $generated = $renderer->render($barcode, $barcode->getWidth() * 5); + + $imageInfo = getimagesizefromstring($generated); + + $this->assertGreaterThan(100, strlen($generated)); + $this->assertEquals(505, $imageInfo[0]); // Image width + $this->assertEquals('image/png', $imageInfo['mime']); + } +} diff --git a/tests/SvgRendererTest.php b/tests/SvgRendererTest.php new file mode 100644 index 00000000..984d034e --- /dev/null +++ b/tests/SvgRendererTest.php @@ -0,0 +1,70 @@ +getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $generated = $renderer->render($barcode, 190); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13.svg', $generated); + } + + public function test_svg_barcode_generator_can_generate_ean_13_barcode_with_fractional_width() + { + $barcode = (new Picqer\Barcode\Types\TypeEan13())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $generated = $renderer->render($barcode, 23.75, 25.75); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13-fractional-width.svg', $generated); + } + + public function test_svg_barcode_generator_as_standalone() + { + $barcode = (new Picqer\Barcode\Types\TypeEan13())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $renderer->setSvgType(\Picqer\Barcode\Renderers\SvgRenderer::TYPE_SVG_STANDALONE); + $generated = $renderer->render($barcode); + + $this->assertStringStartsWith('assertStringContainsString('getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $renderer->setSvgType(\Picqer\Barcode\Renderers\SvgRenderer::TYPE_SVG_INLINE); + $generated = $renderer->render($barcode); + + $this->assertStringStartsWith('assertStringNotContainsString('expectException(\Picqer\Barcode\Exceptions\InvalidOptionException::class); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $renderer->setSvgType('other'); + } + + public function test_svg_barcode_generator_can_use_background_color() + { + $barcode = (new Picqer\Barcode\Types\TypeEan13())->getBarcode('081231723897'); + + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + $renderer->setBackgroundColor([255, 0, 0]); + $generated = $renderer->render($barcode, 190); + + $this->assertStringEqualsFile('tests/verified-files/081231723897-ean13-red-background.svg', $generated); + } +} diff --git a/tests/TypesTest.php b/tests/TypesTest.php new file mode 100644 index 00000000..2c6ef81e --- /dev/null +++ b/tests/TypesTest.php @@ -0,0 +1,261 @@ +getBarcode('1234567890ABC'); + $renderer = new SvgRenderer(); + $result = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile('tests/verified-files/TypeCode39-1234567890ABC.svg', $result); + } + + public function test_generator_can_generate_code_39_checksum_barcode() + { + $barcode = (new TypeCode39Checksum())->getBarcode('1234567890ABC'); + $renderer = new SvgRenderer(); + $result = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_code_39_extended_barcode() + { + $barcode = (new TypeCode39Extended())->getBarcode('1234567890abcABC'); + $renderer = new SvgRenderer(); + $result = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile('tests/verified-files/TypeCode39Extended-1234567890abcABC.svg', $result); + } + + public function test_generator_can_generate_code_39_extended_checksum_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890abcABC', $generator::TYPE_CODE_39E_CHECKSUM); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_code_93_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890abcABC', $generator::TYPE_CODE_93); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_standard_2_5_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_STANDARD_2_5); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_standard_2_5_checksum_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_STANDARD_2_5_CHECKSUM); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_interleaved_2_5_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_INTERLEAVED_2_5); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_interleaved_2_5_checksum_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_INTERLEAVED_2_5_CHECKSUM); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_code_128_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890abcABC-283*33', $generator::TYPE_CODE_128); + + $this->assertStringEqualsFile('tests/verified-files/TypeCode128-1234567890abcABC-283-33.svg', $result); + } + + public function test_generator_can_generate_code_128_a_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_CODE_128_A); + + $this->assertStringEqualsFile('tests/verified-files/TypeCode128A-1234567890.svg', $result); + } + + public function test_generator_can_generate_code_128_b_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890abcABC-283*33', $generator::TYPE_CODE_128_B); + + $this->assertStringEqualsFile('tests/verified-files/TypeCode128B-1234567890abcABC-283-33.svg', $result); + } + + public function test_generator_can_generate_ean_2_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('22', $generator::TYPE_EAN_2); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_ean_5_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890abcABC-283*33', $generator::TYPE_EAN_5); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_ean_8_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234568', $generator::TYPE_EAN_8); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_ean_13_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_EAN_13); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_upc_a_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_UPC_A); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_upc_e_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_UPC_E); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_msi_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_MSI); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_msi_checksum_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_MSI_CHECKSUM); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_postnet_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_POSTNET); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_planet_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_PLANET); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_rms4cc_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_RMS4CC); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_kix_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_KIX); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_imb_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_IMB); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_codabar_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_CODABAR); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_code_11_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_CODE_11); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_pharma_code_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_PHARMA_CODE); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_pharma_code_2_tracks_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('123456789', $generator::TYPE_PHARMA_CODE_TWO_TRACKS); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_telepen_alpha_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890ASCD', $generator::TYPE_TELEPEN_ALPHA); + + $this->assertGreaterThan(100, strlen($result)); + } + + public function test_generator_can_generate_telepen_numeric_barcode() + { + $generator = new Picqer\Barcode\BarcodeGeneratorSVG(); + $result = $generator->getBarcode('1234567890', $generator::TYPE_TELEPEN_NUMERIC); + + $this->assertGreaterThan(100, strlen($result)); + } +} diff --git a/tests/VerifiedBarcodeTest.php b/tests/VerifiedBarcodeTest.php new file mode 100644 index 00000000..0e0267e4 --- /dev/null +++ b/tests/VerifiedBarcodeTest.php @@ -0,0 +1,68 @@ + \Picqer\Barcode\Types\TypeCode39::class, 'barcodes' => ['1234567890ABC']], + ['type' => \Picqer\Barcode\Types\TypeCode39Checksum::class, 'barcodes' => ['1234567890ABC']], + ['type' => \Picqer\Barcode\Types\TypeCode39Extended::class, 'barcodes' => ['1234567890abcABC']], + ['type' => \Picqer\Barcode\Types\TypeCode39ExtendedChecksum::class, 'barcodes' => ['1234567890abcABC']], + ['type' => \Picqer\Barcode\Types\TypeCode93::class, 'barcodes' => ['1234567890abcABC']], + ['type' => \Picqer\Barcode\Types\TypeStandard2of5::class, 'barcodes' => ['1234567890']], + ['type' => \Picqer\Barcode\Types\TypeStandard2of5Checksum::class, 'barcodes' => ['1234567890']], + ['type' => \Picqer\Barcode\Types\TypeInterleaved25::class, 'barcodes' => ['1234567890']], + ['type' => \Picqer\Barcode\Types\TypeInterleaved25Checksum::class, 'barcodes' => ['1234567890']], + ['type' => \Picqer\Barcode\Types\TypeEan13::class, 'barcodes' => ['081231723897', '0049000004632', '004900000463']], + ['type' => \Picqer\Barcode\Types\TypeITF14::class, 'barcodes' => ['00012345600012', '05400141288766']], + ['type' => \Picqer\Barcode\Types\TypeCode128::class, 'barcodes' => ['081231723897', '1234567890abcABC-283*33']], + ['type' => \Picqer\Barcode\Types\TypeCode128A::class, 'barcodes' => ['1234567890']], + ['type' => \Picqer\Barcode\Types\TypeCode128B::class, 'barcodes' => ['081231723897', '1234567890abcABC-283*33']], + ['type' => \Picqer\Barcode\Types\TypeUpcExtension2::class, 'barcodes' => ['22']], + ['type' => \Picqer\Barcode\Types\TypeUpcExtension5::class, 'barcodes' => ['1234567890abcABC-283*33']], + ['type' => \Picqer\Barcode\Types\TypeEan8::class, 'barcodes' => ['1234568']], + ['type' => \Picqer\Barcode\Types\TypeUpcA::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeUpcE::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeMsi::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeMsiChecksum::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypePostnet::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypePlanet::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeRms4cc::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeKix::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeIntelligentMailBarcode::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeCodabar::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeCode11::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypePharmacode::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypePharmacodeTwoCode::class, 'barcodes' => ['123456789']], + ['type' => \Picqer\Barcode\Types\TypeTelepen::class, 'barcodes' => ['1234567890ASCD']], + ['type' => \Picqer\Barcode\Types\TypeTelepenNumeric::class, 'barcodes' => ['1234567890']] + ]; + + public function testAllSupportedBarcodeTypes() + { + $renderer = new Picqer\Barcode\Renderers\SvgRenderer(); + + foreach ($this::$supportedBarcodes as $barcodeTestSet) { + foreach ($barcodeTestSet['barcodes'] as $barcodeText) { + $barcode = (new $barcodeTestSet['type']())->getBarcode($barcodeText); + $result = $renderer->render($barcode, $barcode->getWidth() * 2); + + $this->assertStringEqualsFile( + sprintf('tests/verified-files/%s.svg', StringHelpers::getSafeFilenameFrom($barcodeTestSet['type'] . '-' . $barcodeText)), + $result, + sprintf('%s x %s dynamic test failed', $barcodeTestSet['type'], $barcodeText) + ); + } + } + } +} diff --git a/tests/verified-files/0049000004632-ean13.svg b/tests/verified-files/0049000004632-ean13.svg index 427ff401..5c46159d 100644 --- a/tests/verified-files/0049000004632-ean13.svg +++ b/tests/verified-files/0049000004632-ean13.svg @@ -2,7 +2,7 @@ 0049000004632 - + diff --git a/tests/verified-files/081231723897-code128-red-background.html b/tests/verified-files/081231723897-code128-red-background.html new file mode 100644 index 00000000..45f6b312 --- /dev/null +++ b/tests/verified-files/081231723897-code128-red-background.html @@ -0,0 +1,30 @@ +
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
    diff --git a/tests/verified-files/081231723897-code128.html b/tests/verified-files/081231723897-code128.html index 4a332e57..7b3f9f20 100644 --- a/tests/verified-files/081231723897-code128.html +++ b/tests/verified-files/081231723897-code128.html @@ -1,32 +1,30 @@
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    -
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    diff --git a/tests/verified-files/081231723897-dynamic-code128.html b/tests/verified-files/081231723897-dynamic-code128.html new file mode 100644 index 00000000..6d88107c --- /dev/null +++ b/tests/verified-files/081231723897-dynamic-code128.html @@ -0,0 +1,30 @@ +
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
    diff --git a/tests/verified-files/081231723897-ean13-fractional-width.svg b/tests/verified-files/081231723897-ean13-fractional-width.svg new file mode 100644 index 00000000..2ff29675 --- /dev/null +++ b/tests/verified-files/081231723897-ean13-fractional-width.svg @@ -0,0 +1,37 @@ + + + + 0812317238973 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/081231723897-ean13-red-background.svg b/tests/verified-files/081231723897-ean13-red-background.svg new file mode 100644 index 00000000..52bad4d7 --- /dev/null +++ b/tests/verified-files/081231723897-ean13-red-background.svg @@ -0,0 +1,38 @@ + + + + 0812317238973 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/081231723897-ean13.svg b/tests/verified-files/081231723897-ean13.svg index 96e16193..a3964392 100644 --- a/tests/verified-files/081231723897-ean13.svg +++ b/tests/verified-files/081231723897-ean13.svg @@ -2,7 +2,7 @@ 0812317238973 - + diff --git a/tests/verified-files/12345678903-dynamic-imb.html b/tests/verified-files/12345678903-dynamic-imb.html new file mode 100644 index 00000000..7142156f --- /dev/null +++ b/tests/verified-files/12345678903-dynamic-imb.html @@ -0,0 +1,67 @@ +
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
    diff --git a/tests/verified-files/12345678903-imb.html b/tests/verified-files/12345678903-imb.html new file mode 100644 index 00000000..7620abc9 --- /dev/null +++ b/tests/verified-files/12345678903-imb.html @@ -0,0 +1,67 @@ +
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
     
    +
    diff --git a/tests/verified-files/TypeCodabar-123456789.svg b/tests/verified-files/TypeCodabar-123456789.svg new file mode 100644 index 00000000..0bbe6d1c --- /dev/null +++ b/tests/verified-files/TypeCodabar-123456789.svg @@ -0,0 +1,51 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode11-123456789.svg b/tests/verified-files/TypeCode11-123456789.svg new file mode 100644 index 00000000..8c8eb3fb --- /dev/null +++ b/tests/verified-files/TypeCode11-123456789.svg @@ -0,0 +1,43 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode128-081231723897.svg b/tests/verified-files/TypeCode128-081231723897.svg new file mode 100644 index 00000000..d7267abf --- /dev/null +++ b/tests/verified-files/TypeCode128-081231723897.svg @@ -0,0 +1,35 @@ + + + + 081231723897 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode128-1234567890abcABC-283-33.svg b/tests/verified-files/TypeCode128-1234567890abcABC-283-33.svg new file mode 100644 index 00000000..32b8c1fc --- /dev/null +++ b/tests/verified-files/TypeCode128-1234567890abcABC-283-33.svg @@ -0,0 +1,74 @@ + + + + 1234567890abcABC-283*33 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode128A-1234567890.svg b/tests/verified-files/TypeCode128A-1234567890.svg new file mode 100644 index 00000000..9964f3dc --- /dev/null +++ b/tests/verified-files/TypeCode128A-1234567890.svg @@ -0,0 +1,47 @@ + + + + 1234567890 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode128B-081231723897.svg b/tests/verified-files/TypeCode128B-081231723897.svg new file mode 100644 index 00000000..99133e9d --- /dev/null +++ b/tests/verified-files/TypeCode128B-081231723897.svg @@ -0,0 +1,53 @@ + + + + 081231723897 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode128B-1234567890abcABC-283-33.svg b/tests/verified-files/TypeCode128B-1234567890abcABC-283-33.svg new file mode 100644 index 00000000..9d79a944 --- /dev/null +++ b/tests/verified-files/TypeCode128B-1234567890abcABC-283-33.svg @@ -0,0 +1,86 @@ + + + + 1234567890abcABC-283*33 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode39-1234567890ABC.svg b/tests/verified-files/TypeCode39-1234567890ABC.svg new file mode 100644 index 00000000..a0285647 --- /dev/null +++ b/tests/verified-files/TypeCode39-1234567890ABC.svg @@ -0,0 +1,82 @@ + + + + *1234567890ABC* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode39Checksum-1234567890ABC.svg b/tests/verified-files/TypeCode39Checksum-1234567890ABC.svg new file mode 100644 index 00000000..00a99a7a --- /dev/null +++ b/tests/verified-files/TypeCode39Checksum-1234567890ABC.svg @@ -0,0 +1,87 @@ + + + + *1234567890ABCZ* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode39Extended-1234567890abcABC.svg b/tests/verified-files/TypeCode39Extended-1234567890abcABC.svg new file mode 100644 index 00000000..39fc3e3c --- /dev/null +++ b/tests/verified-files/TypeCode39Extended-1234567890abcABC.svg @@ -0,0 +1,112 @@ + + + + *1234567890+A+B+CABC* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode39ExtendedChecksum-1234567890abcABC.svg b/tests/verified-files/TypeCode39ExtendedChecksum-1234567890abcABC.svg new file mode 100644 index 00000000..a6880d6a --- /dev/null +++ b/tests/verified-files/TypeCode39ExtendedChecksum-1234567890abcABC.svg @@ -0,0 +1,117 @@ + + + + *1234567890+A+B+CABCJ* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeCode93-1234567890abcABC.svg b/tests/verified-files/TypeCode93-1234567890abcABC.svg new file mode 100644 index 00000000..efa0b6e7 --- /dev/null +++ b/tests/verified-files/TypeCode93-1234567890abcABC.svg @@ -0,0 +1,77 @@ + + + + *1234567890dAdBdCABC6-* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeEan13-004900000463.svg b/tests/verified-files/TypeEan13-004900000463.svg new file mode 100644 index 00000000..5c46159d --- /dev/null +++ b/tests/verified-files/TypeEan13-004900000463.svg @@ -0,0 +1,37 @@ + + + + 0049000004632 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeEan13-0049000004632.svg b/tests/verified-files/TypeEan13-0049000004632.svg new file mode 100644 index 00000000..5c46159d --- /dev/null +++ b/tests/verified-files/TypeEan13-0049000004632.svg @@ -0,0 +1,37 @@ + + + + 0049000004632 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeEan13-081231723897.svg b/tests/verified-files/TypeEan13-081231723897.svg new file mode 100644 index 00000000..a3964392 --- /dev/null +++ b/tests/verified-files/TypeEan13-081231723897.svg @@ -0,0 +1,37 @@ + + + + 0812317238973 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeEan8-1234568.svg b/tests/verified-files/TypeEan8-1234568.svg new file mode 100644 index 00000000..2dc590c9 --- /dev/null +++ b/tests/verified-files/TypeEan8-1234568.svg @@ -0,0 +1,29 @@ + + + + 12345687 + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeITF14-00012345600012.svg b/tests/verified-files/TypeITF14-00012345600012.svg new file mode 100644 index 00000000..c2a317be --- /dev/null +++ b/tests/verified-files/TypeITF14-00012345600012.svg @@ -0,0 +1,46 @@ + + + + 00012345600012 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeITF14-05400141288766.svg b/tests/verified-files/TypeITF14-05400141288766.svg new file mode 100644 index 00000000..dd1a0b9c --- /dev/null +++ b/tests/verified-files/TypeITF14-05400141288766.svg @@ -0,0 +1,46 @@ + + + + 05400141288766 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeIntelligentMailBarcode-123456789.svg b/tests/verified-files/TypeIntelligentMailBarcode-123456789.svg new file mode 100644 index 00000000..8637b41e --- /dev/null +++ b/tests/verified-files/TypeIntelligentMailBarcode-123456789.svg @@ -0,0 +1,72 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeInterleaved25-1234567890.svg b/tests/verified-files/TypeInterleaved25-1234567890.svg new file mode 100644 index 00000000..3d0c1ed3 --- /dev/null +++ b/tests/verified-files/TypeInterleaved25-1234567890.svg @@ -0,0 +1,36 @@ + + + + AA1234567890ZA + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeInterleaved25Checksum-1234567890.svg b/tests/verified-files/TypeInterleaved25Checksum-1234567890.svg new file mode 100644 index 00000000..394bf7d1 --- /dev/null +++ b/tests/verified-files/TypeInterleaved25Checksum-1234567890.svg @@ -0,0 +1,41 @@ + + + + AA012345678905ZA + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeKix-123456789.svg b/tests/verified-files/TypeKix-123456789.svg new file mode 100644 index 00000000..953a3309 --- /dev/null +++ b/tests/verified-files/TypeKix-123456789.svg @@ -0,0 +1,43 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeMsi-123456789.svg b/tests/verified-files/TypeMsi-123456789.svg new file mode 100644 index 00000000..a48095ed --- /dev/null +++ b/tests/verified-files/TypeMsi-123456789.svg @@ -0,0 +1,46 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeMsiChecksum-123456789.svg b/tests/verified-files/TypeMsiChecksum-123456789.svg new file mode 100644 index 00000000..6cd6062f --- /dev/null +++ b/tests/verified-files/TypeMsiChecksum-123456789.svg @@ -0,0 +1,50 @@ + + + + 1234567892 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypePharmacode-123456789.svg b/tests/verified-files/TypePharmacode-123456789.svg new file mode 100644 index 00000000..3dac6de4 --- /dev/null +++ b/tests/verified-files/TypePharmacode-123456789.svg @@ -0,0 +1,33 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypePharmacodeTwoCode-123456789.svg b/tests/verified-files/TypePharmacodeTwoCode-123456789.svg new file mode 100644 index 00000000..6a409d35 --- /dev/null +++ b/tests/verified-files/TypePharmacodeTwoCode-123456789.svg @@ -0,0 +1,24 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypePlanet-123456789.svg b/tests/verified-files/TypePlanet-123456789.svg new file mode 100644 index 00000000..a6a3fe8f --- /dev/null +++ b/tests/verified-files/TypePlanet-123456789.svg @@ -0,0 +1,59 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypePostnet-123456789.svg b/tests/verified-files/TypePostnet-123456789.svg new file mode 100644 index 00000000..5cefece1 --- /dev/null +++ b/tests/verified-files/TypePostnet-123456789.svg @@ -0,0 +1,59 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeRms4cc-123456789.svg b/tests/verified-files/TypeRms4cc-123456789.svg new file mode 100644 index 00000000..5e7c698d --- /dev/null +++ b/tests/verified-files/TypeRms4cc-123456789.svg @@ -0,0 +1,49 @@ + + + + 123456789 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeStandard2of5-1234567890.svg b/tests/verified-files/TypeStandard2of5-1234567890.svg new file mode 100644 index 00000000..8aee6352 --- /dev/null +++ b/tests/verified-files/TypeStandard2of5-1234567890.svg @@ -0,0 +1,63 @@ + + + + 1234567890 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeStandard2of5Checksum-1234567890.svg b/tests/verified-files/TypeStandard2of5Checksum-1234567890.svg new file mode 100644 index 00000000..650cc9f3 --- /dev/null +++ b/tests/verified-files/TypeStandard2of5Checksum-1234567890.svg @@ -0,0 +1,68 @@ + + + + 12345678905 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeTelepen-1234567890ASCD.svg b/tests/verified-files/TypeTelepen-1234567890ASCD.svg new file mode 100644 index 00000000..22429b29 --- /dev/null +++ b/tests/verified-files/TypeTelepen-1234567890ASCD.svg @@ -0,0 +1,95 @@ + + + + 1234567890ASCD + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeTelepenNumeric-1234567890.svg b/tests/verified-files/TypeTelepenNumeric-1234567890.svg new file mode 100644 index 00000000..34c44795 --- /dev/null +++ b/tests/verified-files/TypeTelepenNumeric-1234567890.svg @@ -0,0 +1,54 @@ + + + + 1234567890 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeUpcA-123456789.svg b/tests/verified-files/TypeUpcA-123456789.svg new file mode 100644 index 00000000..38b420c1 --- /dev/null +++ b/tests/verified-files/TypeUpcA-123456789.svg @@ -0,0 +1,37 @@ + + + + 0001234567895 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeUpcE-123456789.svg b/tests/verified-files/TypeUpcE-123456789.svg new file mode 100644 index 00000000..4333be74 --- /dev/null +++ b/tests/verified-files/TypeUpcE-123456789.svg @@ -0,0 +1,24 @@ + + + + 012349 + + + + + + + + + + + + + + + + + + + + diff --git a/tests/verified-files/TypeUpcExtension2-22.svg b/tests/verified-files/TypeUpcExtension2-22.svg new file mode 100644 index 00000000..1fb84e7d --- /dev/null +++ b/tests/verified-files/TypeUpcExtension2-22.svg @@ -0,0 +1,14 @@ + + + + 22 + + + + + + + + + + diff --git a/tests/verified-files/TypeUpcExtension5-1234567890abcABC-283-33.svg b/tests/verified-files/TypeUpcExtension5-1234567890abcABC-283-33.svg new file mode 100644 index 00000000..a4287a51 --- /dev/null +++ b/tests/verified-files/TypeUpcExtension5-1234567890abcABC-283-33.svg @@ -0,0 +1,23 @@ + + + + 1234567890abcABC-283*33 + + + + + + + + + + + + + + + + + + +