/* postal.c - Handles PostNet, PLANET, FIM. RM4SCC and Flattermarken */ /* Zint - A barcode generating program using libpng Copyright (C) 2008 Robin Stuart Including bug fixes by Bryan Hatton This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include "common.h" #define BESET "ABCD" #define DAFTSET "DAFT" #define KRSET "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ" #define KASUTSET "1234567890-abcdefgh" #define CHKASUTSET "0123456789-abcdefgh" #define SHKASUTSET "1234567890-ABCDEFGHIJKLMNOPQRSTUVWXYZ" /* PostNet number encoding table - In this table L is long as S is short */ static char *PNTable[10] = {"LLSSS", "SSSLL", "SSLSL", "SSLLS", "SLSSL", "SLSLS", "SLLSS", "LSSSL", "LSSLS", "LSLSS"}; static char *PLTable[10] = {"SSLLL", "LLLSS", "LLSLS", "LLSSL", "LSLLS", "LSLSL", "LSSLL", "SLLLS", "SLLSL", "SLSLL"}; static char *FIMTable[4] = {"12121112121", "111112111211111", "121111111111121", "13111111131"}; static char *RoyalValues[36] = {"11", "12", "13", "14", "15", "10", "21", "22", "23", "24", "25", "20", "31", "32", "33", "34", "35", "30", "41", "42", "43", "44", "45", "40", "51", "52", "53", "54", "55", "50", "01", "02", "03", "04", "05", "00"}; /* 0 = Full, 1 = Ascender, 2 = Descender, 3 = Tracker */ static char *RoyalTable[36] = {"3300", "3210", "3201", "2310", "2301", "2211", "3120", "3030", "3021", "2130", "2121", "2031", "3102", "3012", "3003", "2112", "2103", "2013", "1320", "1230", "1221", "0330", "0321", "0231", "1302", "1212", "1203", "0312", "0303", "0213", "1122", "1032", "1023", "0132", "0123", "0033"}; static char *FlatTable[10] = {"0504", "18", "0117", "0216", "0315", "0414", "0513", "0612", "0711", "0810"}; static char *KoreaTable[10] = {"1313150613", "0713131313", "0417131313", "1506131313", "0413171313", "17171313", "1315061313", "0413131713", "17131713", "13171713"}; static char *JapanTable[19] = {"114", "132", "312", "123", "141", "321", "213", "231", "411", "144", "414", "324", "342", "234", "432", "243", "423", "441", "111"}; int postnet(struct zint_symbol *symbol, unsigned char source[], char dest[]) { /* Handles the PostNet system used for Zip codes in the US */ unsigned int i, sum, check_digit; int error_number; error_number = 0; if(ustrlen(source) > 38) { strcpy(symbol->errtxt, "Input too long [401]"); return ERROR_TOO_LONG; } error_number = is_sane(NESET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [402]"); return error_number; } sum = 0; /* start character */ concat (dest, "L"); for (i=0; i < ustrlen(source); i++) { lookup(NESET, PNTable, source[i], dest); sum += ctoi(source[i]); } check_digit = (10 - (sum%10))%10; concat(dest, PNTable[check_digit]); /* stop character */ concat (dest, "L"); return error_number; } int post_plot(struct zint_symbol *symbol, unsigned char source[]) { /* Puts PostNet barcodes into the pattern matrix */ char height_pattern[200]; unsigned int loopey; int writer; strcpy(height_pattern, ""); int error_number; error_number = 0; error_number = postnet(symbol, source, height_pattern); if(error_number != 0) { return error_number; } writer = 0; for(loopey = 0; loopey < strlen(height_pattern); loopey++) { if(height_pattern[loopey] == 'L') { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; writer += 3; } symbol->row_height[0] = 6; symbol->row_height[1] = 6; symbol->rows = 2; symbol->width = writer - 1; return error_number; } int planet(struct zint_symbol *symbol, unsigned char source[], char dest[]) { /* Handles the PLANET system used for item tracking in the US */ unsigned int i, sum, check_digit; int error_number; error_number = 0; if(ustrlen(source) > 38) { strcpy(symbol->errtxt, "Input too long [821]"); return ERROR_TOO_LONG; } error_number = is_sane(NESET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [822]"); return error_number; } sum = 0; /* start character */ concat (dest, "L"); for (i=0; i < ustrlen(source); i++) { lookup(NESET, PLTable, source[i], dest); sum += ctoi(source[i]); } check_digit = (10 - (sum%10))%10; concat(dest, PLTable[check_digit]); /* stop character */ concat (dest, "L"); return error_number; } int planet_plot(struct zint_symbol *symbol, unsigned char source[]) { /* Puts PLANET barcodes into the pattern matrix */ char height_pattern[200]; unsigned int loopey; int writer; int error_number; strcpy(height_pattern, ""); error_number = 0; error_number = planet(symbol, source, height_pattern); if(error_number != 0) { return error_number; } writer = 0; for(loopey = 0; loopey < strlen(height_pattern); loopey++) { if(height_pattern[loopey] == 'L') { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; writer += 3; } symbol->row_height[0] = 6; symbol->row_height[1] = 6; symbol->rows = 2; symbol->width = writer - 1; return error_number; } int korea_post(struct zint_symbol *symbol, unsigned char source[]) { /* Korean Postal Authority */ int total, h, loop, check, zeroes, error_number; char localstr[7], checkstr[3], dest[80]; error_number = 0; h = ustrlen(source); if(h > 6) { strcpy(symbol->errtxt, "Input too long [771]"); return ERROR_TOO_LONG; } error_number = is_sane(NESET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [772]"); return error_number; } strcpy(localstr, ""); zeroes = 6 - h; for(loop = 0; loop < zeroes; loop++) concat(localstr, "0"); concat(localstr, (char *)source); total = 0; for(loop = 0; loop < 6; loop++) { total += ctoi(localstr[loop]); } check = 10 - (total % 10); if(check == 10) { check = 0; } checkstr[0] = itoc(check); checkstr[1] = '\0'; concat(localstr, checkstr); strcpy(dest, ""); for(loop = 5; loop >= 0; loop--) { lookup(NESET, KoreaTable, localstr[loop], dest); } lookup(NESET, KoreaTable, localstr[6], dest); expand(symbol, dest); ustrcpy(symbol->text, (unsigned char*)localstr); return error_number; } int fim(struct zint_symbol *symbol, unsigned char source[]) { /* The simplest barcode symbology ever! Supported by MS Word, so here it is! */ /* glyphs from http://en.wikipedia.org/wiki/Facing_Identification_Mark */ int error_number; char dest[17]; error_number = 0; strcpy(dest, ""); to_upper(source); if(ustrlen(source) > 1) { strcpy(symbol->errtxt, "Input too long [491]"); return ERROR_TOO_LONG; } error_number = is_sane(BESET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [492]"); return error_number; } lookup(BESET, FIMTable, source[0], dest); expand(symbol, dest); return error_number; } char rm4scc(char source[], unsigned char dest[]) { /* Handles the 4 State barcodes used in the UK by Royal Mail */ unsigned int i; int top, bottom, row, column, check_digit; char values[3], set_copy[38]; strcpy(set_copy, KRSET); top = 0; bottom = 0; /* start character */ concat ((char*)dest, "1"); for (i=0; i < strlen(source); i++) { lookup(KRSET, RoyalTable, source[i], (char*)dest); strcpy(values, RoyalValues[posn(KRSET, source[i])]); top += ctoi(values[0]); bottom += ctoi(values[1]); } /* Calculate the check digit */ row = (top % 6) - 1; column = (bottom % 6) - 1; if(row == -1) { row = 5; } if(column == -1) { column = 5; } check_digit = (6 * row) + column; concat((char*)dest, RoyalTable[check_digit]); /* stop character */ concat ((char*)dest, "0"); return set_copy[check_digit]; } int royal_plot(struct zint_symbol *symbol, unsigned char source[]) { /* Puts RM4SCC into the data matrix */ char height_pattern[200], check; unsigned int loopey; int writer; int error_number; strcpy(height_pattern, ""); error_number = 0; to_upper(source); if(ustrlen(source) > 120) { strcpy(symbol->errtxt, "Input too long [701]"); return ERROR_TOO_LONG; } error_number = is_sane(KRSET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [702]"); return error_number; } check = rm4scc((char*)source, (unsigned char*)height_pattern); writer = 0; for(loopey = 0; loopey < strlen(height_pattern); loopey++) { if((height_pattern[loopey] == '1') || (height_pattern[loopey] == '0')) { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; if((height_pattern[loopey] == '2') || (height_pattern[loopey] == '0')) { symbol->encoded_data[2][writer] = '1'; } writer += 2; } symbol->row_height[0] = 4; symbol->row_height[1] = 2; symbol->row_height[2] = 4; symbol->rows = 3; symbol->width = writer - 1; return error_number; } int kix_code(struct zint_symbol *symbol, unsigned char source[]) { /* Handles Dutch Post TNT KIX symbols */ /* The same as RM4SCC but without check digit */ /* Specification at http://www.tntpost.nl/zakelijk/klantenservice/downloads/kIX_code/download.aspx */ char height_pattern[50], localstr[13]; unsigned int loopey; int writer, i; int error_number, zeroes; strcpy(height_pattern, ""); error_number = 0; to_upper(source); if(ustrlen(source) > 11) { strcpy(symbol->errtxt, "Input too long [901]"); return ERROR_TOO_LONG; } error_number = is_sane(KRSET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [902]"); return error_number; } /* Add leading zeroes */ strcpy(localstr, ""); zeroes = 11 - ustrlen(source); for(i = 0; i < zeroes; i++) concat(localstr, "0"); concat(localstr, (char *)source); /* Encode data */ for (i = 0; i < 11; i++) { lookup(KRSET, RoyalTable, localstr[i], height_pattern); } writer = 0; for(loopey = 0; loopey < strlen(height_pattern); loopey++) { if((height_pattern[loopey] == '1') || (height_pattern[loopey] == '0')) { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; if((height_pattern[loopey] == '2') || (height_pattern[loopey] == '0')) { symbol->encoded_data[2][writer] = '1'; } writer += 2; } symbol->row_height[0] = 4; symbol->row_height[1] = 2; symbol->row_height[2] = 4; symbol->rows = 3; symbol->width = writer - 1; return error_number; } int daft_code(struct zint_symbol *symbol, unsigned char source[]) { /* Handles DAFT Code symbols */ /* Presumably 'daft' doesn't mean the same thing in Germany as it does in the UK! */ int input_length; char height_pattern[100], local_source[55]; unsigned int loopey; int writer, i, error_number; strcpy(height_pattern, ""); error_number = 0; input_length = ustrlen(source); strcpy(local_source, (char*)source); if(input_length > 50) { strcpy(symbol->errtxt, "Input too long"); return ERROR_TOO_LONG; } to_upper((unsigned char*)local_source); error_number = is_sane(DAFTSET, (unsigned char*)local_source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data"); return error_number; } for (i = 0; i < input_length; i++) { if(local_source[i] == 'D') { concat(height_pattern, "2"); } if(local_source[i] == 'A') { concat(height_pattern, "1"); } if(local_source[i] == 'F') { concat(height_pattern, "0"); } if(local_source[i] == 'T') { concat(height_pattern, "3"); } } writer = 0; for(loopey = 0; loopey < strlen(height_pattern); loopey++) { if((height_pattern[loopey] == '1') || (height_pattern[loopey] == '0')) { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; if((height_pattern[loopey] == '2') || (height_pattern[loopey] == '0')) { symbol->encoded_data[2][writer] = '1'; } writer += 2; } symbol->row_height[0] = 4; symbol->row_height[1] = 2; symbol->row_height[2] = 4; symbol->rows = 3; symbol->width = writer - 1; return error_number; } int flattermarken(struct zint_symbol *symbol, unsigned char source[]) { /* Flattermarken - Not really a barcode symbology and (in my opinion) probably not much use but it's supported by TBarCode so it's supported by Zint! */ int loop, error_number; char dest[1000]; error_number = 0; strcpy(dest, ""); if(ustrlen(source) > 90) { strcpy(symbol->errtxt, "Input too long [281]"); return ERROR_TOO_LONG; } error_number = is_sane(NESET, source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data [282]"); return error_number; } for(loop = 0; loop < ustrlen(source); loop++) { lookup(NESET, FlatTable, source[loop], dest); } expand(symbol, dest); return error_number; } int japan_post(struct zint_symbol *symbol, unsigned char source[]) { /* Japanese Postal Code (Kasutama Barcode) */ int input_length, error_number; char pattern[65]; int writer, loopey, inter_posn, i, inter_length, sum, check; char check_char; input_length = ustrlen(source); inter_length = input_length * 2; if(inter_length < 20) { inter_length = 20; } char inter[inter_length]; char local_source[input_length]; inter_posn = 0; error_number = 0; strcpy(local_source, (char*)source); to_upper((unsigned char*)local_source); error_number = is_sane(SHKASUTSET, (unsigned char*)local_source); if(error_number == ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "Invalid characters in data"); return error_number; } for(i = 0; i < inter_length; i++) { inter[i] = 'd'; /* Pad character CC4 */ } for(i = 0; i < input_length; i++) { if(((local_source[i] >= '0') && (local_source[i] <= '9')) || (local_source[i] == '-')) { inter[inter_posn] = local_source[i]; inter_posn++; } else { if((local_source[i] >= 'A') && (local_source[i] <= 'J')) { inter[inter_posn] = 'a'; inter[inter_posn + 1] = local_source[i] - 'A' + '0'; inter_posn += 2; } if((local_source[i] >= 'K') && (local_source[i] <= 'T')) { inter[inter_posn] = 'b'; inter[inter_posn + 1] = local_source[i] - 'K' + '0'; inter_posn += 2; } if((local_source[i] >= 'U') && (local_source[i] <= 'Z')) { inter[inter_posn] = 'c'; inter[inter_posn + 1] = local_source[i] - 'U' + '0'; inter_posn += 2; } } } strcpy(pattern, "13"); /* Start */ sum = 0; for(i = 0; i < 20; i++) { concat(pattern, JapanTable[posn(KASUTSET, inter[i])]); sum += posn(CHKASUTSET, inter[i]); /* printf("%c (%d)\n", inter[i], posn(CHKASUTSET, inter[i])); */ } /* Calculate check digit */ check = 19 - (sum % 19); if(check == 19) { check = 0; } if(check <= 9) { check_char = check + '0'; } if(check == 10) { check_char = '-'; } if(check >= 11) { check_char = (check - 11) + 'a'; } concat(pattern, JapanTable[posn(KASUTSET, check_char)]); /* printf("check %c (%d)\n", check_char, check); */ concat(pattern, "31"); /* Stop */ /* Resolve pattern to 4-state symbols */ writer = 0; for(loopey = 0; loopey < strlen(pattern); loopey++) { if((pattern[loopey] == '2') || (pattern[loopey] == '1')) { symbol->encoded_data[0][writer] = '1'; } symbol->encoded_data[1][writer] = '1'; if((pattern[loopey] == '3') || (pattern[loopey] == '1')) { symbol->encoded_data[2][writer] = '1'; } writer += 2; } symbol->row_height[0] = 2; symbol->row_height[1] = 2; symbol->row_height[2] = 2; symbol->rows = 3; symbol->width = writer - 1; return error_number; }