2008-07-14 01:15:55 +04:00
/* upcean.c - Handles UPC, EAN and ISBN
libzint - the open source barcode library
2008-11-17 11:47:42 +03:00
Copyright ( C ) 2008 Robin Stuart < robin @ zint . org . uk >
2008-07-14 01:15:55 +04:00
2013-05-16 21:26:38 +04:00
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2. Redistributions in binary form must reproduce the above copyright
notice , this list of conditions and the following disclaimer in the
documentation and / or other materials provided with the distribution .
3. Neither the name of the project nor the names of its contributors
may be used to endorse or promote products derived from this software
without specific prior written permission .
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS " AND
ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE .
2008-07-14 01:15:55 +04:00
*/
2009-10-06 23:03:00 +04:00
# define SODIUM "0123456789+"
2008-07-14 01:15:55 +04:00
# define EAN2 102
# define EAN5 105
# include <stdio.h>
# include <string.h>
# include <stdlib.h>
# include "common.h"
/* UPC and EAN tables checked against EN 797:1996 */
2012-12-29 22:37:03 +04:00
static const char * UPCParity0 [ 10 ] = { " BBBAAA " , " BBABAA " , " BBAABA " , " BBAAAB " , " BABBAA " , " BAABBA " , " BAAABB " ,
2008-07-14 01:15:55 +04:00
" BABABA " , " BABAAB " , " BAABAB " } ; /* Number set for UPC-E symbol (EN Table 4) */
2012-12-29 22:37:03 +04:00
static const char * UPCParity1 [ 10 ] = { " AAABBB " , " AABABB " , " AABBAB " , " AABBBA " , " ABAABB " , " ABBAAB " , " ABBBAA " ,
2008-07-14 01:15:55 +04:00
" ABABAB " , " ABABBA " , " ABBABA " } ; /* Not covered by BS EN 797:1995 */
2012-12-29 22:37:03 +04:00
static const char * EAN2Parity [ 4 ] = { " AA " , " AB " , " BA " , " BB " } ; /* Number sets for 2-digit add-on (EN Table 6) */
static const char * EAN5Parity [ 10 ] = { " BBAAA " , " BABAA " , " BAABA " , " BAAAB " , " ABBAA " , " AABBA " , " AAABB " , " ABABA " ,
2008-07-14 01:15:55 +04:00
" ABAAB " , " AABAB " } ; /* Number set for 5-digit add-on (EN Table 7) */
2012-12-29 22:37:03 +04:00
static const char * EAN13Parity [ 10 ] = { " AAAAA " , " ABABB " , " ABBAB " , " ABBBA " , " BAABB " , " BBAAB " , " BBBAA " , " BABAB " ,
2008-07-14 01:15:55 +04:00
" BABBA " , " BBABA " } ; /* Left hand of the EAN-13 symbol (EN Table 3) */
2012-12-29 22:37:03 +04:00
static const char * EANsetA [ 10 ] = { " 3211 " , " 2221 " , " 2122 " , " 1411 " , " 1132 " , " 1231 " , " 1114 " , " 1312 " , " 1213 " ,
2008-07-14 01:15:55 +04:00
" 3112 " } ; /* Representation set A and C (EN Table 1) */
2012-12-29 22:37:03 +04:00
static const char * EANsetB [ 10 ] = { " 1123 " , " 1222 " , " 2212 " , " 1141 " , " 2311 " , " 1321 " , " 4111 " , " 2131 " , " 3121 " ,
2008-07-14 01:15:55 +04:00
" 2113 " } ; /* Representation set B (EN Table 1) */
2008-10-06 11:15:07 +04:00
char upc_check ( char source [ ] )
2008-07-14 01:15:55 +04:00
{ /* Calculate the correct check digit for a UPC barcode */
unsigned int i , count , check_digit ;
count = 0 ;
2010-09-14 14:36:00 +04:00
for ( i = 0 ; i < strlen ( source ) ; i + + ) {
2008-07-14 01:15:55 +04:00
count + = ctoi ( source [ i ] ) ;
2010-09-14 14:36:00 +04:00
if ( ! ( i & 1 ) ) {
2008-07-14 01:15:55 +04:00
count + = 2 * ( ctoi ( source [ i ] ) ) ;
}
}
check_digit = 10 - ( count % 10 ) ;
if ( check_digit = = 10 ) { check_digit = 0 ; }
return itoc ( check_digit ) ;
}
2008-10-06 11:15:07 +04:00
void upca_draw ( char source [ ] , char dest [ ] )
2008-07-14 01:15:55 +04:00
{ /* UPC A is usually used for 12 digit numbers, but this function takes a source of any length */
unsigned int i , half_way ;
2008-10-06 11:15:07 +04:00
half_way = strlen ( source ) / 2 ;
2008-07-14 01:15:55 +04:00
/* start character */
concat ( dest , " 111 " ) ;
2008-10-06 11:15:07 +04:00
for ( i = 0 ; i < = strlen ( source ) ; i + + )
2008-07-14 01:15:55 +04:00
{
if ( i = = half_way )
{
/* middle character - separates manufacturer no. from product no. */
/* also inverts right hand characters */
concat ( dest , " 11111 " ) ;
}
2009-10-06 23:03:00 +04:00
lookup ( NEON , EANsetA , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
/* stop character */
concat ( dest , " 111 " ) ;
}
void upca ( struct zint_symbol * symbol , unsigned char source [ ] , char dest [ ] )
{ /* Make a UPC A barcode when we haven't been given the check digit */
int length ;
2008-10-06 11:15:07 +04:00
char gtin [ 15 ] ;
strcpy ( gtin , ( char * ) source ) ;
length = strlen ( gtin ) ;
gtin [ length ] = upc_check ( gtin ) ;
gtin [ length + 1 ] = ' \0 ' ;
upca_draw ( gtin , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) gtin ) ;
2008-07-14 01:15:55 +04:00
}
void upce ( struct zint_symbol * symbol , unsigned char source [ ] , char dest [ ] )
{ /* UPC E is a zero-compressed version of UPC A */
unsigned int i , num_system ;
char emode , equivalent [ 12 ] , check_digit , parity [ 8 ] , temp [ 8 ] ;
2008-10-03 12:05:56 +04:00
char hrt [ 9 ] ;
2008-07-14 01:15:55 +04:00
/* Two number systems can be used - system 0 and system 1 */
2008-09-30 19:05:53 +04:00
if ( ustrlen ( source ) = = 7 ) {
2008-07-14 01:15:55 +04:00
switch ( source [ 0 ] ) {
case ' 0 ' : num_system = 0 ; break ;
case ' 1 ' : num_system = 1 ; break ;
default : num_system = 0 ; source [ 0 ] = ' 0 ' ; break ;
}
2008-10-03 12:05:56 +04:00
strcpy ( temp , ( char * ) source ) ;
strcpy ( hrt , ( char * ) source ) ;
2008-07-14 01:15:55 +04:00
for ( i = 1 ; i < = 7 ; i + + ) {
source [ i - 1 ] = temp [ i ] ;
}
}
else {
num_system = 0 ;
hrt [ 0 ] = ' 0 ' ;
hrt [ 1 ] = ' \0 ' ;
2008-10-03 12:05:56 +04:00
concat ( hrt , ( char * ) source ) ;
2008-07-14 01:15:55 +04:00
}
/* Expand the zero-compressed UPCE code to make a UPCA equivalent (EN Table 5) */
emode = source [ 5 ] ;
for ( i = 0 ; i < 11 ; i + + ) {
equivalent [ i ] = ' 0 ' ;
}
2009-07-05 00:49:17 +04:00
if ( num_system = = 1 ) { equivalent [ 0 ] = temp [ 0 ] ; }
2008-07-14 01:15:55 +04:00
equivalent [ 1 ] = source [ 0 ] ;
equivalent [ 2 ] = source [ 1 ] ;
equivalent [ 11 ] = ' \0 ' ;
switch ( emode )
{
case ' 0 ' :
case ' 1 ' :
case ' 2 ' :
equivalent [ 3 ] = emode ;
equivalent [ 8 ] = source [ 2 ] ;
equivalent [ 9 ] = source [ 3 ] ;
equivalent [ 10 ] = source [ 4 ] ;
break ;
case ' 3 ' :
equivalent [ 3 ] = source [ 2 ] ;
equivalent [ 9 ] = source [ 3 ] ;
equivalent [ 10 ] = source [ 4 ] ;
if ( ( ( source [ 2 ] = = ' 0 ' ) | | ( source [ 2 ] = = ' 1 ' ) ) | | ( source [ 2 ] = = ' 2 ' ) ) {
/* Note 1 - "X3 shall not be equal to 0, 1 or 2" */
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid UPC-E data " ) ;
2008-07-14 01:15:55 +04:00
}
break ;
case ' 4 ' :
equivalent [ 3 ] = source [ 2 ] ;
equivalent [ 4 ] = source [ 3 ] ;
equivalent [ 10 ] = source [ 4 ] ;
if ( source [ 3 ] = = ' 0 ' ) {
/* Note 2 - "X4 shall not be equal to 0" */
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid UPC-E data " ) ;
2008-07-14 01:15:55 +04:00
}
break ;
case ' 5 ' :
case ' 6 ' :
case ' 7 ' :
case ' 8 ' :
case ' 9 ' :
equivalent [ 3 ] = source [ 2 ] ;
equivalent [ 4 ] = source [ 3 ] ;
equivalent [ 5 ] = source [ 4 ] ;
equivalent [ 10 ] = emode ;
if ( source [ 4 ] = = ' 0 ' ) {
/* Note 3 - "X5 shall not be equal to 0" */
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid UPC-E data " ) ;
2008-07-14 01:15:55 +04:00
}
break ;
}
/* Get the check digit from the expanded UPCA code */
2008-10-06 11:15:07 +04:00
check_digit = upc_check ( equivalent ) ;
2008-07-14 01:15:55 +04:00
/* Use the number system and check digit information to choose a parity scheme */
if ( num_system = = 1 ) {
strcpy ( parity , UPCParity1 [ ctoi ( check_digit ) ] ) ;
} else {
strcpy ( parity , UPCParity0 [ ctoi ( check_digit ) ] ) ;
}
/* Take all this information and make the barcode pattern */
/* start character */
concat ( dest , " 111 " ) ;
2008-09-30 19:05:53 +04:00
for ( i = 0 ; i < = ustrlen ( source ) ; i + + ) {
2008-07-14 01:15:55 +04:00
switch ( parity [ i ] ) {
2009-10-06 23:03:00 +04:00
case ' A ' : lookup ( NEON , EANsetA , source [ i ] , dest ) ; break ;
case ' B ' : lookup ( NEON , EANsetB , source [ i ] , dest ) ; break ;
2008-07-14 01:15:55 +04:00
}
}
/* stop character */
concat ( dest , " 111111 " ) ;
hrt [ 7 ] = check_digit ;
hrt [ 8 ] = ' \0 ' ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) hrt ) ;
2008-07-14 01:15:55 +04:00
}
void add_on ( unsigned char source [ ] , char dest [ ] , int mode )
{ /* EAN-2 and EAN-5 add-on codes */
char parity [ 6 ] ;
unsigned int i , code_type ;
/* If an add-on then append with space */
if ( mode ! = 0 )
{
concat ( dest , " 9 " ) ;
}
/* Start character */
concat ( dest , " 112 " ) ;
/* Determine EAN2 or EAN5 add-on */
2008-09-30 19:05:53 +04:00
if ( ustrlen ( source ) = = 2 )
2008-07-14 01:15:55 +04:00
{
code_type = EAN2 ;
}
else
{
code_type = EAN5 ;
}
/* Calculate parity for EAN2 */
if ( code_type = = EAN2 )
{
int code_value , parity_bit ;
code_value = ( 10 * ctoi ( source [ 0 ] ) ) + ctoi ( source [ 1 ] ) ;
parity_bit = code_value % 4 ;
strcpy ( parity , EAN2Parity [ parity_bit ] ) ;
}
if ( code_type = = EAN5 )
{
int values [ 6 ] , parity_sum , parity_bit ;
for ( i = 0 ; i < 6 ; i + + )
{
values [ i ] = ctoi ( source [ i ] ) ;
}
parity_sum = ( 3 * ( values [ 0 ] + values [ 2 ] + values [ 4 ] ) ) ;
parity_sum + = ( 9 * ( values [ 1 ] + values [ 3 ] ) ) ;
parity_bit = parity_sum % 10 ;
strcpy ( parity , EAN5Parity [ parity_bit ] ) ;
}
2008-09-30 19:05:53 +04:00
for ( i = 0 ; i < ustrlen ( source ) ; i + + )
2008-07-14 01:15:55 +04:00
{
switch ( parity [ i ] ) {
2009-10-06 23:03:00 +04:00
case ' A ' : lookup ( NEON , EANsetA , source [ i ] , dest ) ; break ;
case ' B ' : lookup ( NEON , EANsetB , source [ i ] , dest ) ; break ;
2008-07-14 01:15:55 +04:00
}
/* Glyph separator */
2008-09-30 19:05:53 +04:00
if ( i ! = ( ustrlen ( source ) - 1 ) )
2008-07-14 01:15:55 +04:00
{
concat ( dest , " 11 " ) ;
}
}
}
/* ************************ EAN-13 ****************** */
2008-10-06 11:15:07 +04:00
char ean_check ( char source [ ] )
2008-07-14 01:15:55 +04:00
{ /* Calculate the correct check digit for a EAN-13 barcode */
int i ;
unsigned int h , count , check_digit ;
count = 0 ;
2008-10-06 11:15:07 +04:00
h = strlen ( source ) ;
2010-09-14 14:36:00 +04:00
for ( i = h - 1 ; i > = 0 ; i - - ) {
2008-07-14 01:15:55 +04:00
count + = ctoi ( source [ i ] ) ;
2010-09-14 14:36:00 +04:00
if ( i & 1 ) {
2008-07-14 01:15:55 +04:00
count + = 2 * ctoi ( source [ i ] ) ;
}
}
check_digit = 10 - ( count % 10 ) ;
if ( check_digit = = 10 ) { check_digit = 0 ; }
return itoc ( check_digit ) ;
}
void ean13 ( struct zint_symbol * symbol , unsigned char source [ ] , char dest [ ] )
{
unsigned int length , i , half_way ;
char parity [ 6 ] ;
2008-10-06 11:15:07 +04:00
char gtin [ 15 ] ;
2008-07-14 01:15:55 +04:00
strcpy ( parity , " " ) ;
2008-10-06 11:15:07 +04:00
strcpy ( gtin , ( char * ) source ) ;
2008-07-14 01:15:55 +04:00
/* Add the appropriate check digit */
2008-10-06 11:15:07 +04:00
length = strlen ( gtin ) ;
gtin [ length ] = ean_check ( gtin ) ;
gtin [ length + 1 ] = ' \0 ' ;
2008-07-14 01:15:55 +04:00
/* Get parity for first half of the symbol */
2009-10-06 23:03:00 +04:00
lookup ( SODIUM , EAN13Parity , gtin [ 0 ] , parity ) ;
2008-07-14 01:15:55 +04:00
/* Now get on with the cipher */
half_way = 7 ;
/* start character */
concat ( dest , " 111 " ) ;
2009-10-06 23:03:00 +04:00
length = strlen ( gtin ) ;
for ( i = 1 ; i < = length ; i + + )
2008-07-14 01:15:55 +04:00
{
if ( i = = half_way )
{
/* middle character - separates manufacturer no. from product no. */
/* also inverses right hand characters */
concat ( dest , " 11111 " ) ;
}
if ( ( ( i > 1 ) & & ( i < 7 ) ) & & ( parity [ i - 2 ] = = ' B ' ) )
{
2009-10-06 23:03:00 +04:00
lookup ( NEON , EANsetB , gtin [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
else
{
2009-10-06 23:03:00 +04:00
lookup ( NEON , EANsetA , gtin [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
}
/* stop character */
concat ( dest , " 111 " ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) gtin ) ;
2008-07-14 01:15:55 +04:00
}
void ean8 ( struct zint_symbol * symbol , unsigned char source [ ] , char dest [ ] )
{ /* Make an EAN-8 barcode when we haven't been given the check digit */
/* EAN-8 is basically the same as UPC-A but with fewer digits */
int length ;
2008-10-06 11:15:07 +04:00
char gtin [ 10 ] ;
strcpy ( gtin , ( char * ) source ) ;
length = strlen ( gtin ) ;
gtin [ length ] = upc_check ( gtin ) ;
gtin [ length + 1 ] = ' \0 ' ;
upca_draw ( gtin , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) gtin ) ;
2008-07-14 01:15:55 +04:00
}
char isbn13_check ( unsigned char source [ ] ) /* For ISBN(13) only */
{
2009-10-06 23:03:00 +04:00
unsigned int i , weight , sum , check , h ;
2008-07-14 01:15:55 +04:00
sum = 0 ;
weight = 1 ;
2009-10-06 23:03:00 +04:00
h = ustrlen ( source ) - 1 ;
2008-07-14 01:15:55 +04:00
2009-10-06 23:03:00 +04:00
for ( i = 0 ; i < h ; i + + )
2008-07-14 01:15:55 +04:00
{
sum + = ctoi ( source [ i ] ) * weight ;
if ( weight = = 1 ) weight = 3 ; else weight = 1 ;
}
check = sum % 10 ;
check = 10 - check ;
2011-05-09 12:02:01 +04:00
if ( check = = 10 ) check = 0 ;
2008-07-14 01:15:55 +04:00
return itoc ( check ) ;
}
char isbn_check ( unsigned char source [ ] ) /* For ISBN(10) and SBN only */
{
2009-10-06 23:03:00 +04:00
unsigned int i , weight , sum , check , h ;
2010-05-17 01:58:18 +04:00
char check_char ;
2008-07-14 01:15:55 +04:00
sum = 0 ;
weight = 1 ;
2009-10-06 23:03:00 +04:00
h = ustrlen ( source ) - 1 ;
for ( i = 0 ; i < h ; i + + )
2008-07-14 01:15:55 +04:00
{
sum + = ctoi ( source [ i ] ) * weight ;
weight + + ;
}
check = sum % 11 ;
2010-05-17 01:58:18 +04:00
check_char = itoc ( check ) ;
if ( check = = 10 ) { check_char = ' X ' ; }
return check_char ;
2008-07-14 01:15:55 +04:00
}
2009-10-06 23:03:00 +04:00
int isbn ( struct zint_symbol * symbol , unsigned char source [ ] , const unsigned int src_len , char dest [ ] ) /* Make an EAN-13 barcode from an SBN or ISBN */
2008-07-14 01:15:55 +04:00
{
2009-10-06 23:03:00 +04:00
int i , error_number ;
2008-07-14 01:15:55 +04:00
char check_digit ;
2009-10-06 23:03:00 +04:00
to_upper ( source ) ;
error_number = is_sane ( " 0123456789X " , source , src_len ) ;
2016-02-17 13:37:20 +03:00
if ( error_number = = ZINT_ERROR_INVALID_DATA ) {
2009-10-06 23:03:00 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in input " ) ;
return error_number ;
}
2008-07-14 01:15:55 +04:00
/* Input must be 9, 10 or 13 characters */
2009-10-06 23:03:00 +04:00
if ( ( ( src_len < 9 ) | | ( src_len > 13 ) ) | | ( ( src_len > 10 ) & & ( src_len < 13 ) ) )
2008-07-14 01:15:55 +04:00
{
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Input wrong length " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
2009-10-06 23:03:00 +04:00
if ( src_len = = 13 ) /* Using 13 character ISBN */
2008-07-14 01:15:55 +04:00
{
if ( ! ( ( ( source [ 0 ] = = ' 9 ' ) & & ( source [ 1 ] = = ' 7 ' ) ) & &
( ( source [ 2 ] = = ' 8 ' ) | | ( source [ 2 ] = = ' 9 ' ) ) ) )
{
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid ISBN " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_INVALID_DATA ;
2008-07-14 01:15:55 +04:00
}
check_digit = isbn13_check ( source ) ;
2009-10-06 23:03:00 +04:00
if ( source [ src_len - 1 ] ! = check_digit )
2008-07-14 01:15:55 +04:00
{
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Incorrect ISBN check " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_INVALID_CHECK ;
2008-07-14 01:15:55 +04:00
}
source [ 12 ] = ' \0 ' ;
ean13 ( symbol , source , dest ) ;
}
2009-10-06 23:03:00 +04:00
if ( src_len = = 10 ) /* Using 10 digit ISBN */
2008-07-14 01:15:55 +04:00
{
check_digit = isbn_check ( source ) ;
2009-10-06 23:03:00 +04:00
if ( check_digit ! = source [ src_len - 1 ] )
2008-07-14 01:15:55 +04:00
{
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Incorrect ISBN check " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_INVALID_CHECK ;
2008-07-14 01:15:55 +04:00
}
for ( i = 13 ; i > 0 ; i - - )
{
source [ i ] = source [ i - 3 ] ;
}
source [ 0 ] = ' 9 ' ;
source [ 1 ] = ' 7 ' ;
source [ 2 ] = ' 8 ' ;
source [ 12 ] = ' \0 ' ;
ean13 ( symbol , source , dest ) ;
}
2009-10-06 23:03:00 +04:00
if ( src_len = = 9 ) /* Using 9 digit SBN */
2008-07-14 01:15:55 +04:00
{
/* Add leading zero */
for ( i = 10 ; i > 0 ; i - - )
{
source [ i ] = source [ i - 1 ] ;
}
source [ 0 ] = ' 0 ' ;
/* Verify check digit */
check_digit = isbn_check ( source ) ;
2008-09-30 19:05:53 +04:00
if ( check_digit ! = source [ ustrlen ( source ) - 1 ] )
2008-07-14 01:15:55 +04:00
{
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Incorrect SBN check " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_INVALID_CHECK ;
2008-07-14 01:15:55 +04:00
}
/* Convert to EAN-13 number */
for ( i = 13 ; i > 0 ; i - - )
{
source [ i ] = source [ i - 3 ] ;
}
source [ 0 ] = ' 9 ' ;
source [ 1 ] = ' 7 ' ;
source [ 2 ] = ' 8 ' ;
source [ 12 ] = ' \0 ' ;
ean13 ( symbol , source , dest ) ;
}
2009-09-29 13:45:46 +04:00
return 0 ;
2008-07-14 01:15:55 +04:00
}
2009-05-27 13:06:28 +04:00
void ean_leading_zeroes ( struct zint_symbol * symbol , unsigned char source [ ] , unsigned char local_source [ ] ) {
/* Add leading zeroes to EAN and UPC strings */
unsigned char first_part [ 20 ] , second_part [ 20 ] , zfirst_part [ 20 ] , zsecond_part [ 20 ] ;
int with_addon = 0 ;
2009-10-06 23:03:00 +04:00
int first_len = 0 , second_len = 0 , zfirst_len = 0 , zsecond_len = 0 , i , h ;
2009-05-27 13:06:28 +04:00
2009-10-06 23:03:00 +04:00
h = ustrlen ( source ) ;
for ( i = 0 ; i < h ; i + + ) {
2009-05-27 13:06:28 +04:00
if ( source [ i ] = = ' + ' ) {
with_addon = 1 ;
} else {
if ( with_addon = = 0 ) {
first_len + + ;
} else {
second_len + + ;
}
}
}
ustrcpy ( first_part , ( unsigned char * ) " " ) ;
ustrcpy ( second_part , ( unsigned char * ) " " ) ;
ustrcpy ( zfirst_part , ( unsigned char * ) " " ) ;
ustrcpy ( zsecond_part , ( unsigned char * ) " " ) ;
/* Split input into two strings */
for ( i = 0 ; i < first_len ; i + + ) {
first_part [ i ] = source [ i ] ;
first_part [ i + 1 ] = ' \0 ' ;
}
for ( i = 0 ; i < second_len ; i + + ) {
second_part [ i ] = source [ i + first_len + 1 ] ;
second_part [ i + 1 ] = ' \0 ' ;
}
/* Calculate target lengths */
2009-07-08 13:14:15 +04:00
if ( second_len < = 5 ) { zsecond_len = 5 ; }
if ( second_len < = 2 ) { zsecond_len = 2 ; }
if ( second_len = = 0 ) { zsecond_len = 0 ; }
2009-05-27 13:06:28 +04:00
switch ( symbol - > symbology ) {
case BARCODE_EANX :
case BARCODE_EANX_CC :
if ( first_len < = 12 ) { zfirst_len = 12 ; }
if ( first_len < = 7 ) { zfirst_len = 7 ; }
2009-07-08 13:14:15 +04:00
if ( second_len = = 0 ) {
if ( first_len < = 5 ) { zfirst_len = 5 ; }
if ( first_len < = 2 ) { zfirst_len = 2 ; }
}
2009-05-27 13:06:28 +04:00
break ;
case BARCODE_UPCA :
case BARCODE_UPCA_CC :
zfirst_len = 11 ;
break ;
case BARCODE_UPCE :
case BARCODE_UPCE_CC :
if ( first_len = = 7 ) { zfirst_len = 7 ; }
if ( first_len < = 6 ) { zfirst_len = 6 ; }
break ;
case BARCODE_ISBNX :
2009-06-07 13:43:37 +04:00
if ( first_len < = 9 ) { zfirst_len = 9 ; }
2009-05-27 13:06:28 +04:00
break ;
}
2009-07-08 13:14:15 +04:00
2009-05-27 13:06:28 +04:00
/* Add leading zeroes */
for ( i = 0 ; i < ( zfirst_len - first_len ) ; i + + ) {
uconcat ( zfirst_part , ( unsigned char * ) " 0 " ) ;
}
uconcat ( zfirst_part , first_part ) ;
for ( i = 0 ; i < ( zsecond_len - second_len ) ; i + + ) {
uconcat ( zsecond_part , ( unsigned char * ) " 0 " ) ;
}
uconcat ( zsecond_part , second_part ) ;
/* Copy adjusted data back to local_source */
uconcat ( local_source , zfirst_part ) ;
if ( zsecond_len ! = 0 ) {
uconcat ( local_source , ( unsigned char * ) " + " ) ;
uconcat ( local_source , zsecond_part ) ;
}
}
2009-10-06 23:03:00 +04:00
int eanx ( struct zint_symbol * symbol , unsigned char source [ ] , int src_len )
2008-07-14 01:15:55 +04:00
{
/* splits string to parts before and after '+' parts */
2009-10-06 23:03:00 +04:00
unsigned char first_part [ 20 ] = { 0 } , second_part [ 20 ] = { 0 } , dest [ 1000 ] = { 0 } ;
unsigned char local_source [ 20 ] = { 0 } ;
2008-07-14 01:15:55 +04:00
unsigned int latch , reader , writer , with_addon ;
2009-09-29 13:45:46 +04:00
int error_number , i ;
2008-07-14 01:15:55 +04:00
with_addon = FALSE ;
latch = FALSE ;
writer = 0 ;
2009-10-06 23:03:00 +04:00
if ( src_len > 19 ) {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
if ( symbol - > symbology ! = BARCODE_ISBNX ) {
/* ISBN has it's own checking routine */
2009-11-09 01:17:11 +03:00
error_number = is_sane ( " 0123456789+ " , source , src_len ) ;
2016-02-17 13:37:20 +03:00
if ( error_number = = ZINT_ERROR_INVALID_DATA ) {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2009-09-29 13:45:46 +04:00
return error_number ;
}
} else {
2009-10-06 23:03:00 +04:00
error_number = is_sane ( " 0123456789Xx " , source , src_len ) ;
2016-02-17 13:37:20 +03:00
if ( error_number = = ZINT_ERROR_INVALID_DATA ) {
2009-09-29 13:45:46 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in input " ) ;
return error_number ;
2008-07-14 01:15:55 +04:00
}
}
2009-05-27 13:06:28 +04:00
2009-09-29 13:45:46 +04:00
2009-05-27 13:06:28 +04:00
/* Add leading zeroes */
ustrcpy ( local_source , ( unsigned char * ) " " ) ;
2009-09-29 13:45:46 +04:00
if ( symbol - > symbology = = BARCODE_ISBNX ) {
to_upper ( local_source ) ;
}
2009-05-27 13:06:28 +04:00
ean_leading_zeroes ( symbol , source , local_source ) ;
2008-07-14 01:15:55 +04:00
2009-05-27 13:06:28 +04:00
for ( reader = 0 ; reader < = ustrlen ( local_source ) ; reader + + )
2008-07-14 01:15:55 +04:00
{
if ( source [ reader ] = = ' + ' ) { with_addon = TRUE ; }
}
reader = 0 ;
if ( with_addon ) {
do {
2009-05-27 13:06:28 +04:00
if ( local_source [ reader ] = = ' + ' ) {
2008-07-14 01:15:55 +04:00
first_part [ writer ] = ' \0 ' ;
latch = TRUE ;
reader + + ;
writer = 0 ;
}
if ( latch ) {
2009-05-27 13:06:28 +04:00
second_part [ writer ] = local_source [ reader ] ;
2008-07-14 01:15:55 +04:00
reader + + ;
writer + + ;
} else {
2009-05-27 13:06:28 +04:00
first_part [ writer ] = local_source [ reader ] ;
2008-07-14 01:15:55 +04:00
reader + + ;
writer + + ;
}
2009-05-27 13:06:28 +04:00
} while ( reader < = ustrlen ( local_source ) ) ;
2008-07-14 01:15:55 +04:00
} else {
2009-05-27 13:06:28 +04:00
strcpy ( ( char * ) first_part , ( char * ) local_source ) ;
2008-07-14 01:15:55 +04:00
}
switch ( symbol - > symbology )
{
case BARCODE_EANX :
2008-10-03 12:05:56 +04:00
switch ( ustrlen ( first_part ) )
2008-07-14 01:15:55 +04:00
{
2009-02-19 22:09:57 +03:00
case 2 : add_on ( first_part , ( char * ) dest , 0 ) ; ustrcpy ( symbol - > text , first_part ) ; break ;
case 5 : add_on ( first_part , ( char * ) dest , 0 ) ; ustrcpy ( symbol - > text , first_part ) ; break ;
2008-10-03 12:05:56 +04:00
case 7 : ean8 ( symbol , first_part , ( char * ) dest ) ; break ;
case 12 : ean13 ( symbol , first_part , ( char * ) dest ) ; break ;
2016-02-17 13:37:20 +03:00
default : strcpy ( symbol - > errtxt , " Invalid length input " ) ; return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_EANX_CC :
2008-10-03 12:05:56 +04:00
switch ( ustrlen ( first_part ) )
2008-07-14 01:15:55 +04:00
{ /* Adds vertical separator bars according to ISO/IEC 24723 section 11.4 */
2009-06-01 00:33:54 +04:00
case 7 : set_module ( symbol , symbol - > rows , 1 ) ;
set_module ( symbol , symbol - > rows , 67 ) ;
set_module ( symbol , symbol - > rows + 1 , 0 ) ;
set_module ( symbol , symbol - > rows + 1 , 68 ) ;
set_module ( symbol , symbol - > rows + 2 , 1 ) ;
set_module ( symbol , symbol - > rows + 1 , 67 ) ;
2008-07-14 01:15:55 +04:00
symbol - > row_height [ symbol - > rows ] = 2 ;
symbol - > row_height [ symbol - > rows + 1 ] = 2 ;
symbol - > row_height [ symbol - > rows + 2 ] = 2 ;
symbol - > rows + = 3 ;
2008-10-03 12:05:56 +04:00
ean8 ( symbol , first_part , ( char * ) dest ) ; break ;
2009-06-01 00:33:54 +04:00
case 12 : set_module ( symbol , symbol - > rows , 1 ) ;
set_module ( symbol , symbol - > rows , 95 ) ;
set_module ( symbol , symbol - > rows + 1 , 0 ) ;
set_module ( symbol , symbol - > rows + 1 , 96 ) ;
set_module ( symbol , symbol - > rows + 2 , 1 ) ;
set_module ( symbol , symbol - > rows + 2 , 95 ) ;
2008-07-14 01:15:55 +04:00
symbol - > row_height [ symbol - > rows ] = 2 ;
symbol - > row_height [ symbol - > rows + 1 ] = 2 ;
symbol - > row_height [ symbol - > rows + 2 ] = 2 ;
symbol - > rows + = 3 ;
2008-10-03 12:05:56 +04:00
ean13 ( symbol , first_part , ( char * ) dest ) ; break ;
2016-02-17 13:37:20 +03:00
default : strcpy ( symbol - > errtxt , " Invalid length EAN input " ) ; return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_UPCA :
2008-10-03 12:05:56 +04:00
if ( ustrlen ( first_part ) = = 11 ) {
upca ( symbol , first_part , ( char * ) dest ) ;
2008-07-14 01:15:55 +04:00
} else {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Input wrong length " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_UPCA_CC :
2008-10-03 12:05:56 +04:00
if ( ustrlen ( first_part ) = = 11 ) {
2009-06-01 00:33:54 +04:00
set_module ( symbol , symbol - > rows , 1 ) ;
set_module ( symbol , symbol - > rows , 95 ) ;
set_module ( symbol , symbol - > rows + 1 , 0 ) ;
set_module ( symbol , symbol - > rows + 1 , 96 ) ;
set_module ( symbol , symbol - > rows + 2 , 1 ) ;
set_module ( symbol , symbol - > rows + 2 , 95 ) ;
2008-07-14 01:15:55 +04:00
symbol - > row_height [ symbol - > rows ] = 2 ;
symbol - > row_height [ symbol - > rows + 1 ] = 2 ;
symbol - > row_height [ symbol - > rows + 2 ] = 2 ;
symbol - > rows + = 3 ;
2008-10-03 12:05:56 +04:00
upca ( symbol , first_part , ( char * ) dest ) ;
2008-07-14 01:15:55 +04:00
} else {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " UPCA input wrong length " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_UPCE :
2008-10-03 12:05:56 +04:00
if ( ( ustrlen ( first_part ) > = 6 ) & & ( ustrlen ( first_part ) < = 7 ) ) {
upce ( symbol , first_part , ( char * ) dest ) ;
2008-07-14 01:15:55 +04:00
} else {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Input wrong length " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_UPCE_CC :
2008-10-03 12:05:56 +04:00
if ( ( ustrlen ( first_part ) > = 6 ) & & ( ustrlen ( first_part ) < = 7 ) ) {
2009-06-01 00:33:54 +04:00
set_module ( symbol , symbol - > rows , 1 ) ;
set_module ( symbol , symbol - > rows , 51 ) ;
set_module ( symbol , symbol - > rows + 1 , 0 ) ;
set_module ( symbol , symbol - > rows + 1 , 52 ) ;
set_module ( symbol , symbol - > rows + 2 , 1 ) ;
set_module ( symbol , symbol - > rows + 2 , 51 ) ;
2008-07-14 01:15:55 +04:00
symbol - > row_height [ symbol - > rows ] = 2 ;
symbol - > row_height [ symbol - > rows + 1 ] = 2 ;
symbol - > row_height [ symbol - > rows + 2 ] = 2 ;
symbol - > rows + = 3 ;
2008-10-03 12:05:56 +04:00
upce ( symbol , first_part , ( char * ) dest ) ;
2008-07-14 01:15:55 +04:00
} else {
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " UPCE input wrong length " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
break ;
case BARCODE_ISBNX :
2009-10-06 23:03:00 +04:00
error_number = isbn ( symbol , first_part , ustrlen ( first_part ) , ( char * ) dest ) ;
2009-09-29 13:45:46 +04:00
if ( error_number > 4 ) {
return error_number ;
2008-07-14 01:15:55 +04:00
}
break ;
}
2008-10-03 12:05:56 +04:00
switch ( ustrlen ( second_part ) )
2008-07-14 01:15:55 +04:00
{
case 0 : break ;
case 2 :
2008-10-03 12:05:56 +04:00
add_on ( second_part , ( char * ) dest , 1 ) ;
2009-02-19 22:09:57 +03:00
uconcat ( symbol - > text , ( unsigned char * ) " + " ) ;
uconcat ( symbol - > text , second_part ) ;
2008-07-14 01:15:55 +04:00
break ;
case 5 :
2008-10-03 12:05:56 +04:00
add_on ( second_part , ( char * ) dest , 1 ) ;
2009-02-19 22:09:57 +03:00
uconcat ( symbol - > text , ( unsigned char * ) " + " ) ;
uconcat ( symbol - > text , second_part ) ;
2008-07-14 01:15:55 +04:00
break ;
default :
2009-05-27 13:06:28 +04:00
strcpy ( symbol - > errtxt , " Invalid length input " ) ;
2016-02-17 13:37:20 +03:00
return ZINT_ERROR_TOO_LONG ;
2008-07-14 01:15:55 +04:00
}
2008-10-03 12:05:56 +04:00
expand ( symbol , ( char * ) dest ) ;
2008-07-14 01:15:55 +04:00
switch ( symbol - > symbology ) {
case BARCODE_EANX_CC :
case BARCODE_UPCA_CC :
case BARCODE_UPCE_CC :
/* shift the symbol to the right one space to allow for separator bars */
for ( i = ( symbol - > width + 1 ) ; i > = 1 ; i - - ) {
2009-06-01 00:33:54 +04:00
if ( module_is_set ( symbol , symbol - > rows - 1 , i - 1 ) ) {
set_module ( symbol , symbol - > rows - 1 , i ) ;
} else {
unset_module ( symbol , symbol - > rows - 1 , i ) ;
}
2008-07-14 01:15:55 +04:00
}
2009-06-01 00:33:54 +04:00
unset_module ( symbol , symbol - > rows - 1 , 0 ) ;
2008-07-14 01:15:55 +04:00
symbol - > width + = 2 ;
break ;
}
2009-09-29 13:45:46 +04:00
if ( ( symbol - > errtxt [ 0 ] = = ' w ' ) & & ( error_number = = 0 ) ) {
error_number = 1 ; /* flag UPC-E warnings */
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}