2008-07-14 09:15:55 +12:00
|
|
|
/* code128.c - Handles Code 128 and derivatives */
|
|
|
|
/*
|
|
|
|
libzint - the open source barcode library
|
2023-01-15 13:22:43 +13:00
|
|
|
Copyright (C) 2008-2023 Robin Stuart <rstuart114@gmail.com>
|
2008-07-19 02:35:32 +12:00
|
|
|
Bugfixes thanks to Christian Sakowski and BogDan Vatra
|
2013-05-17 05:26:38 +12:00
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
|
2017-10-24 08:37:52 +13:00
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
2013-05-17 05:26:38 +12:00
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
2017-10-24 08:37:52 +13:00
|
|
|
documentation and/or other materials provided with the distribution.
|
2013-05-17 05:26:38 +12:00
|
|
|
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
|
2017-10-24 08:37:52 +13:00
|
|
|
without specific prior written permission.
|
2013-05-17 05:26:38 +12:00
|
|
|
|
|
|
|
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
|
2017-10-24 08:37:52 +13:00
|
|
|
OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
2013-05-17 05:26:38 +12:00
|
|
|
SUCH DAMAGE.
|
2016-02-20 22:38:03 +13:00
|
|
|
*/
|
RMQR: update to ISO/IEC 23941:2022 - R13x77 numeric cclens change 8 -> 7
QRCODE: use stricter interpretation of ZINT_FULL_MULTIBYTE, excluding
certain trailing bytes
libzint: fix some confusing error messages introduced by segment stuff
general: new escape chars \U, \d and \o
backend_qt: fudge rendering of border rectangles due to scaling/translation
rounding errors TODO: better fudge
GUI: foreground/background colours -> text boxes and icon buttons, add swap
button, independently movable picker (NULL parent), preview colour changes,
preview Data Window changes, add clear data (del) buttons, add zap button
and Factory Reset menu option, various other fixes
libzint: remove STATIC_UNLESS_ZINT_TEST, use wrappers
CMake: add find package QtSvg, remove QtXml
manual: split symbology and general specs and sort, move DAFT to 4-state,
UPC/EAN -> EAN/UPC, DataBar -> GS1 DataBar always, expand MAILMARK info,
various other fiddlings
man page: options or -> |, expand MSI Plessey check digit options
README.linux: add packages info
license: add SPDX-License-Identifier to touched files
2022-06-10 08:52:02 +12:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause */
|
2008-07-14 09:15:55 +12:00
|
|
|
|
2020-05-22 05:22:28 +12:00
|
|
|
#include <assert.h>
|
2022-07-15 03:01:30 +12:00
|
|
|
#include <stdio.h>
|
2008-07-14 09:15:55 +12:00
|
|
|
#include "common.h"
|
2020-05-22 05:22:28 +12:00
|
|
|
#include "code128.h"
|
2009-01-08 21:43:25 +13:00
|
|
|
#include "gs1.h"
|
2008-07-14 09:15:55 +12:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
static const char KRSET[] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
|
|
#define KRSET_F (IS_NUM_F | IS_UPR_F)
|
|
|
|
|
2008-07-14 09:15:55 +12:00
|
|
|
/* Code 128 tables checked against ISO/IEC 15417:2007 */
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
INTERNAL_DATA const char C128Table[107][6] = { /* Used by CODABLOCKF and CODE16K also */
|
|
|
|
/* Code 128 character encodation - Table 1 (with final CODE16K-only character in place of Stop character) */
|
|
|
|
{'2','1','2','2','2','2'}, {'2','2','2','1','2','2'}, {'2','2','2','2','2','1'}, {'1','2','1','2','2','3'},
|
|
|
|
{'1','2','1','3','2','2'}, {'1','3','1','2','2','2'}, {'1','2','2','2','1','3'}, {'1','2','2','3','1','2'},
|
|
|
|
{'1','3','2','2','1','2'}, {'2','2','1','2','1','3'}, {'2','2','1','3','1','2'}, {'2','3','1','2','1','2'},
|
|
|
|
{'1','1','2','2','3','2'}, {'1','2','2','1','3','2'}, {'1','2','2','2','3','1'}, {'1','1','3','2','2','2'},
|
|
|
|
{'1','2','3','1','2','2'}, {'1','2','3','2','2','1'}, {'2','2','3','2','1','1'}, {'2','2','1','1','3','2'},
|
|
|
|
{'2','2','1','2','3','1'}, {'2','1','3','2','1','2'}, {'2','2','3','1','1','2'}, {'3','1','2','1','3','1'},
|
|
|
|
{'3','1','1','2','2','2'}, {'3','2','1','1','2','2'}, {'3','2','1','2','2','1'}, {'3','1','2','2','1','2'},
|
|
|
|
{'3','2','2','1','1','2'}, {'3','2','2','2','1','1'}, {'2','1','2','1','2','3'}, {'2','1','2','3','2','1'},
|
|
|
|
{'2','3','2','1','2','1'}, {'1','1','1','3','2','3'}, {'1','3','1','1','2','3'}, {'1','3','1','3','2','1'},
|
|
|
|
{'1','1','2','3','1','3'}, {'1','3','2','1','1','3'}, {'1','3','2','3','1','1'}, {'2','1','1','3','1','3'},
|
|
|
|
{'2','3','1','1','1','3'}, {'2','3','1','3','1','1'}, {'1','1','2','1','3','3'}, {'1','1','2','3','3','1'},
|
|
|
|
{'1','3','2','1','3','1'}, {'1','1','3','1','2','3'}, {'1','1','3','3','2','1'}, {'1','3','3','1','2','1'},
|
|
|
|
{'3','1','3','1','2','1'}, {'2','1','1','3','3','1'}, {'2','3','1','1','3','1'}, {'2','1','3','1','1','3'},
|
|
|
|
{'2','1','3','3','1','1'}, {'2','1','3','1','3','1'}, {'3','1','1','1','2','3'}, {'3','1','1','3','2','1'},
|
|
|
|
{'3','3','1','1','2','1'}, {'3','1','2','1','1','3'}, {'3','1','2','3','1','1'}, {'3','3','2','1','1','1'},
|
|
|
|
{'3','1','4','1','1','1'}, {'2','2','1','4','1','1'}, {'4','3','1','1','1','1'}, {'1','1','1','2','2','4'},
|
|
|
|
{'1','1','1','4','2','2'}, {'1','2','1','1','2','4'}, {'1','2','1','4','2','1'}, {'1','4','1','1','2','2'},
|
|
|
|
{'1','4','1','2','2','1'}, {'1','1','2','2','1','4'}, {'1','1','2','4','1','2'}, {'1','2','2','1','1','4'},
|
|
|
|
{'1','2','2','4','1','1'}, {'1','4','2','1','1','2'}, {'1','4','2','2','1','1'}, {'2','4','1','2','1','1'},
|
|
|
|
{'2','2','1','1','1','4'}, {'4','1','3','1','1','1'}, {'2','4','1','1','1','2'}, {'1','3','4','1','1','1'},
|
|
|
|
{'1','1','1','2','4','2'}, {'1','2','1','1','4','2'}, {'1','2','1','2','4','1'}, {'1','1','4','2','1','2'},
|
|
|
|
{'1','2','4','1','1','2'}, {'1','2','4','2','1','1'}, {'4','1','1','2','1','2'}, {'4','2','1','1','1','2'},
|
|
|
|
{'4','2','1','2','1','1'}, {'2','1','2','1','4','1'}, {'2','1','4','1','2','1'}, {'4','1','2','1','2','1'},
|
|
|
|
{'1','1','1','1','4','3'}, {'1','1','1','3','4','1'}, {'1','3','1','1','4','1'}, {'1','1','4','1','1','3'},
|
|
|
|
{'1','1','4','3','1','1'}, {'4','1','1','1','1','3'}, {'4','1','1','3','1','1'}, {'1','1','3','1','4','1'},
|
|
|
|
{'1','1','4','1','3','1'}, {'3','1','1','1','4','1'}, {'4','1','1','1','3','1'}, {'2','1','1','4','1','2'},
|
|
|
|
{'2','1','1','2','1','4'}, {'2','1','1','2','3','2'}, {/* Only used by CODE16K */ '2','1','1','1','3','3'}
|
2016-02-20 22:38:03 +13:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Determine appropriate mode for a given character */
|
2021-10-21 11:05:30 +13:00
|
|
|
INTERNAL int c128_parunmodd(const unsigned char llyth) {
|
2016-02-20 22:38:03 +13:00
|
|
|
int modd;
|
|
|
|
|
|
|
|
if (llyth <= 31) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_SHIFTA;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else if ((llyth >= 48) && (llyth <= 57)) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_ABORC;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else if (llyth <= 95) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_AORB;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else if (llyth <= 127) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_SHIFTB;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else if (llyth <= 159) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_SHIFTA;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else if (llyth <= 223) {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_AORB;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
modd = C128_SHIFTB;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
return modd;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2013-01-01 02:41:59 +13:00
|
|
|
/**
|
2023-01-15 13:22:43 +13:00
|
|
|
* Bring together same type blocks
|
2013-01-01 02:41:59 +13:00
|
|
|
*/
|
2023-01-15 13:22:43 +13:00
|
|
|
static void c128_grwp(int list[2][C128_MAX], int *p_indexliste) {
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* bring together same type blocks */
|
2023-01-15 13:22:43 +13:00
|
|
|
if (*p_indexliste > 1) {
|
2017-09-11 03:03:09 +12:00
|
|
|
int i = 1;
|
2023-01-15 13:22:43 +13:00
|
|
|
while (i < *p_indexliste) {
|
2017-10-24 08:34:31 +13:00
|
|
|
if (list[1][i - 1] == list[1][i]) {
|
2017-10-17 06:26:54 +13:00
|
|
|
int j;
|
2016-02-20 22:38:03 +13:00
|
|
|
/* bring together */
|
|
|
|
list[0][i - 1] = list[0][i - 1] + list[0][i];
|
2017-10-17 06:26:54 +13:00
|
|
|
j = i + 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2020-05-16 21:22:33 +12:00
|
|
|
/* decrease the list */
|
2023-01-15 13:22:43 +13:00
|
|
|
while (j < *p_indexliste) {
|
2016-02-20 22:38:03 +13:00
|
|
|
list[0][j - 1] = list[0][j];
|
|
|
|
list[1][j - 1] = list[1][j];
|
|
|
|
j++;
|
|
|
|
}
|
2023-01-15 13:22:43 +13:00
|
|
|
*p_indexliste = *p_indexliste - 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
i--;
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2013-01-01 02:41:59 +13:00
|
|
|
/**
|
|
|
|
* Implements rules from ISO 15417 Annex E
|
|
|
|
*/
|
2023-01-15 13:22:43 +13:00
|
|
|
INTERNAL void c128_dxsmooth(int list[2][C128_MAX], int *p_indexliste, const char *manual_set) {
|
2017-09-11 03:03:09 +12:00
|
|
|
int i, last, next;
|
2023-01-15 13:22:43 +13:00
|
|
|
const int indexliste = *p_indexliste;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
for (i = 0; i < indexliste; i++) {
|
2021-10-21 11:05:30 +13:00
|
|
|
int current = list[1][i]; /* Either C128_ABORC, C128_AORB, C128_SHIFTA or C128_SHIFTB */
|
2017-09-11 03:03:09 +12:00
|
|
|
int length = list[0][i];
|
2016-02-20 22:38:03 +13:00
|
|
|
if (i != 0) {
|
|
|
|
last = list[1][i - 1];
|
|
|
|
} else {
|
2022-07-19 07:30:31 +12:00
|
|
|
last = 0;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2023-01-15 13:22:43 +13:00
|
|
|
if (i != indexliste - 1) {
|
2016-02-20 22:38:03 +13:00
|
|
|
next = list[1][i + 1];
|
|
|
|
} else {
|
2022-07-19 07:30:31 +12:00
|
|
|
next = 0;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
if (i == 0) { /* first block */
|
2021-10-21 11:05:30 +13:00
|
|
|
if (current == C128_ABORC) {
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set && manual_set[i]) {
|
|
|
|
list[1][i] = manual_set[i];
|
|
|
|
current = manual_set[i];
|
|
|
|
} else if ((indexliste == 1) && (length == 2)) {
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Rule 1a */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHC;
|
|
|
|
current = C128_LATCHC;
|
2020-05-16 21:22:33 +12:00
|
|
|
} else if (length >= 4) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Rule 1b */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHC;
|
|
|
|
current = C128_LATCHC;
|
2016-02-20 22:38:03 +13:00
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
current = C128_AORB; /* Determine below */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
if (current == C128_AORB) {
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set && (manual_set[i] == 'A' || manual_set[i] == 'B')) {
|
|
|
|
list[1][i] = manual_set[i];
|
|
|
|
} else if (next == C128_SHIFTA) {
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Rule 1c */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHA;
|
2020-05-16 21:22:33 +12:00
|
|
|
} else {
|
|
|
|
/* Rule 1d */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHB;
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
} else if (current == C128_SHIFTA) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Rule 1c */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHA;
|
2023-01-15 13:22:43 +13:00
|
|
|
} else if (current == C128_SHIFTB) { /* Unless C128_LATCHX set above, can only be C128_SHIFTB */
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Rule 1d */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHB;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
if (current == C128_ABORC) {
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set && manual_set[i]) {
|
|
|
|
list[1][i] = manual_set[i];
|
|
|
|
current = manual_set[i];
|
|
|
|
} else if (length >= 4) {
|
|
|
|
/* Rule 3 - note Rule 3b (odd C blocks) dealt with later */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHC;
|
|
|
|
current = C128_LATCHC;
|
2020-05-16 21:22:33 +12:00
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
current = C128_AORB; /* Determine below */
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
if (current == C128_AORB) {
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set && (manual_set[i] == 'A' || manual_set[i] == 'B')) {
|
|
|
|
list[1][i] = manual_set[i];
|
|
|
|
} else if (last == C128_LATCHA || last == C128_SHIFTB) { /* Maintain state */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHA;
|
|
|
|
} else if (last == C128_LATCHB || last == C128_SHIFTA) { /* Maintain state */
|
|
|
|
list[1][i] = C128_LATCHB;
|
|
|
|
} else if (next == C128_SHIFTA) {
|
|
|
|
list[1][i] = C128_LATCHA;
|
2020-05-16 21:22:33 +12:00
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHB;
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
} else if (current == C128_SHIFTA) {
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set && manual_set[i] == 'A') {
|
|
|
|
list[1][i] = C128_LATCHA;
|
|
|
|
} else if (length > 1) {
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Rule 4 */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHA;
|
|
|
|
} else if (last == C128_LATCHA || last == C128_SHIFTB) { /* Maintain state */
|
|
|
|
list[1][i] = C128_LATCHA;
|
|
|
|
} else if (last == C128_LATCHC) {
|
|
|
|
list[1][i] = C128_LATCHA;
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2023-01-15 13:22:43 +13:00
|
|
|
} else if (current == C128_SHIFTB) { /* Unless C128_LATCHX set above, can only be C128_SHIFTB */
|
|
|
|
if (manual_set && manual_set[i] == 'B') {
|
|
|
|
list[1][i] = C128_LATCHB;
|
|
|
|
} else if (length > 1) {
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Rule 5 */
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][i] = C128_LATCHB;
|
|
|
|
} else if (last == C128_LATCHB || last == C128_SHIFTA) { /* Maintain state */
|
|
|
|
list[1][i] = C128_LATCHB;
|
|
|
|
} else if (last == C128_LATCHC) {
|
|
|
|
list[1][i] = C128_LATCHB;
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2020-05-16 21:22:33 +12:00
|
|
|
} /* Rule 2 is implemented elsewhere, Rule 6 is implied */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2008-07-14 09:15:55 +12:00
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
c128_grwp(list, p_indexliste);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2013-01-01 02:41:59 +13:00
|
|
|
/**
|
|
|
|
* Translate Code 128 Set A characters into barcodes.
|
2020-05-16 21:22:33 +12:00
|
|
|
* This set handles all control characters NUL to US.
|
2013-01-01 02:41:59 +13:00
|
|
|
*/
|
2021-10-21 11:05:30 +13:00
|
|
|
INTERNAL void c128_set_a(const unsigned char source, int values[], int *bar_chars) {
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
if (source > 127) {
|
|
|
|
if (source < 160) {
|
|
|
|
values[(*bar_chars)] = (source - 128) + 64;
|
|
|
|
} else {
|
|
|
|
values[(*bar_chars)] = (source - 128) - 32;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (source < 32) {
|
|
|
|
values[(*bar_chars)] = source + 64;
|
|
|
|
} else {
|
|
|
|
values[(*bar_chars)] = source - 32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(*bar_chars)++;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2013-01-01 02:41:59 +13:00
|
|
|
/**
|
|
|
|
* Translate Code 128 Set B characters into barcodes.
|
|
|
|
* This set handles all characters which are not part of long numbers and not
|
|
|
|
* control characters.
|
|
|
|
*/
|
2021-10-21 11:05:30 +13:00
|
|
|
INTERNAL int c128_set_b(const unsigned char source, int values[], int *bar_chars) {
|
2021-07-08 02:46:02 +12:00
|
|
|
if (source >= 128 + 32) {
|
2016-02-20 22:38:03 +13:00
|
|
|
values[(*bar_chars)] = source - 32 - 128;
|
2021-07-08 02:46:02 +12:00
|
|
|
} else if (source >= 128) { /* Should never happen */
|
|
|
|
return 0; /* Not reached */
|
|
|
|
} else if (source >= 32) {
|
2016-02-20 22:38:03 +13:00
|
|
|
values[(*bar_chars)] = source - 32;
|
2021-07-08 00:58:04 +12:00
|
|
|
} else { /* Should never happen */
|
|
|
|
return 0; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
(*bar_chars)++;
|
2021-07-08 00:58:04 +12:00
|
|
|
return 1;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Translate Code 128 Set C characters into barcodes
|
|
|
|
* This set handles numbers in a compressed form
|
|
|
|
*/
|
2021-10-21 11:05:30 +13:00
|
|
|
INTERNAL void c128_set_c(const unsigned char source_a, const unsigned char source_b, int values[], int *bar_chars) {
|
2016-02-20 22:38:03 +13:00
|
|
|
int weight;
|
|
|
|
|
|
|
|
weight = (10 * ctoi(source_a)) + ctoi(source_b);
|
|
|
|
values[(*bar_chars)] = weight;
|
|
|
|
(*bar_chars)++;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Put set data into set[]. If source given (GS1_MODE) then resolves odd C blocks */
|
|
|
|
INTERNAL void c128_put_in_set(int list[2][C128_MAX], const int indexliste, char set[C128_MAX],
|
2023-01-15 13:22:43 +13:00
|
|
|
const unsigned char *source) {
|
2021-10-21 11:05:30 +13:00
|
|
|
int read = 0;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < indexliste; i++) {
|
|
|
|
for (j = 0; j < list[0][i]; j++) {
|
2023-01-15 13:22:43 +13:00
|
|
|
set[read++] = list[1][i];
|
2021-10-21 11:05:30 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (source) {
|
|
|
|
/* Watch out for odd-length Mode C blocks */
|
|
|
|
int c_count = 0;
|
|
|
|
for (i = 0; i < read; i++) {
|
|
|
|
if (set[i] == 'C') {
|
|
|
|
if (source[i] == '[') {
|
|
|
|
if (c_count & 1) {
|
|
|
|
if ((i - c_count) != 0) {
|
|
|
|
set[i - c_count] = 'B';
|
|
|
|
} else {
|
|
|
|
set[i - 1] = 'B';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c_count = 0;
|
|
|
|
} else {
|
|
|
|
c_count++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (c_count & 1) {
|
|
|
|
if ((i - c_count) != 0) {
|
|
|
|
set[i - c_count] = 'B';
|
|
|
|
} else {
|
|
|
|
set[i - 1] = 'B';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c_count = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (c_count & 1) {
|
|
|
|
if ((i - c_count) != 0) {
|
|
|
|
set[i - c_count] = 'B';
|
|
|
|
} else {
|
|
|
|
set[i - 1] = 'B';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 1; i < read - 1; i++) {
|
|
|
|
if ((set[i] == 'C') && ((set[i - 1] == 'B') && (set[i + 1] == 'B'))) {
|
|
|
|
set[i] = 'B';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Handle Code 128, 128B and HIBC 128 */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int code128(struct zint_symbol *symbol, unsigned char source[], int length) {
|
2022-10-14 01:33:59 +13:00
|
|
|
int i, j, k, values[C128_MAX] = {0}, bar_characters = 0, read, total_sum;
|
|
|
|
int error_number = 0, indexchaine, indexliste, f_state = 0;
|
2023-01-15 13:22:43 +13:00
|
|
|
unsigned char src_buf[C128_MAX + 1];
|
|
|
|
unsigned char *src = source;
|
|
|
|
char manual_set[C128_MAX] = {0};
|
2020-05-22 05:22:28 +12:00
|
|
|
int list[2][C128_MAX] = {{0}};
|
|
|
|
char set[C128_MAX] = {0}, fset[C128_MAX], mode, last_set, current_set = ' ';
|
2022-10-15 03:05:33 +13:00
|
|
|
int glyph_count = 0; /* Codeword estimate times 2 */
|
2016-02-20 22:38:03 +13:00
|
|
|
char dest[1000];
|
2021-10-21 11:05:30 +13:00
|
|
|
char *d = dest;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2020-05-22 05:22:28 +12:00
|
|
|
/* Suppresses clang-analyzer-core.UndefinedBinaryOperatorResult warning on fset which is fully set */
|
|
|
|
assert(length > 0);
|
|
|
|
|
2022-10-14 01:33:59 +13:00
|
|
|
if (length > C128_MAX) {
|
2020-04-03 09:55:58 +13:00
|
|
|
/* This only blocks ridiculously long input - the actual length of the
|
2016-02-20 22:38:03 +13:00
|
|
|
resulting barcode depends on the type of data, so this is trapped later */
|
2021-07-07 06:53:31 +12:00
|
|
|
sprintf(symbol->errtxt, "340: Input too long (%d character maximum)", C128_MAX);
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
/* Detect special Code Set escapes for Code 128 in extra escape mode only */
|
|
|
|
if ((symbol->input_mode & EXTRA_ESCAPE_MODE) && symbol->symbology == BARCODE_CODE128) {
|
|
|
|
char manual_ch = '\0';
|
|
|
|
j = 0;
|
|
|
|
for (i = 0; i < length; i++) {
|
|
|
|
if (source[i] == '\\' && i + 2 < length && source[i + 1] == '^'
|
|
|
|
&& ((source[i + 2] >= 'A' && source[i + 2] <= 'C') || source[i + 2] == '^')) {
|
|
|
|
if (source[i + 2] != '^') {
|
|
|
|
i += 2;
|
|
|
|
manual_ch = source[i];
|
|
|
|
} else { /* Escape sequence '\^^' */
|
|
|
|
manual_set[j] = manual_ch;
|
|
|
|
src_buf[j++] = source[i++];
|
|
|
|
manual_set[j] = manual_ch;
|
|
|
|
src_buf[j++] = source[i++];
|
|
|
|
/* Drop second '^' */
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
manual_set[j] = manual_ch;
|
|
|
|
src_buf[j++] = source[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (j != length) {
|
|
|
|
length = j;
|
|
|
|
if (length == 0) {
|
|
|
|
strcpy(symbol->errtxt, "842: No input data");
|
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
|
|
|
}
|
|
|
|
src = src_buf;
|
|
|
|
src[length] = '\0';
|
|
|
|
if (symbol->debug & ZINT_DEBUG_PRINT) {
|
2023-06-12 12:25:55 +12:00
|
|
|
fputs("MSet: ", stdout);
|
2023-01-15 13:22:43 +13:00
|
|
|
for (i = 0; i < length; i++) printf("%c", manual_set[i] ? manual_set[i] : '.');
|
2023-06-12 12:25:55 +12:00
|
|
|
fputc('\n', stdout);
|
2023-01-15 13:22:43 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Detect extended ASCII characters */
|
2022-10-14 01:33:59 +13:00
|
|
|
for (i = 0; i < length; i++) {
|
2023-01-15 13:22:43 +13:00
|
|
|
fset[i] = src[i] >= 128 ? 'f' : ' ';
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Decide when to latch to extended mode - Annex E note 3 */
|
|
|
|
j = 0;
|
2022-10-14 01:33:59 +13:00
|
|
|
for (i = 0; i < length; i++) {
|
2016-02-20 22:38:03 +13:00
|
|
|
if (fset[i] == 'f') {
|
|
|
|
j++;
|
|
|
|
} else {
|
|
|
|
j = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (j >= 5) {
|
|
|
|
for (k = i; k > (i - 5); k--) {
|
|
|
|
fset[k] = 'F';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-14 01:33:59 +13:00
|
|
|
if ((j >= 3) && (i == (length - 1))) {
|
2016-02-20 22:38:03 +13:00
|
|
|
for (k = i; k > (i - 3); k--) {
|
|
|
|
fset[k] = 'F';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Decide if it is worth reverting to 646 encodation for a few characters as described in 4.3.4.2 (d) */
|
2022-10-14 01:33:59 +13:00
|
|
|
for (i = 1; i < length; i++) {
|
2016-02-20 22:38:03 +13:00
|
|
|
if ((fset[i - 1] == 'F') && (fset[i] == ' ')) {
|
|
|
|
/* Detected a change from 8859-1 to 646 - count how long for */
|
2022-10-14 01:33:59 +13:00
|
|
|
for (j = 0; ((i + j) < length) && (fset[i + j] == ' '); j++);
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Count how many 8859-1 beyond */
|
|
|
|
k = 0;
|
2022-10-14 01:33:59 +13:00
|
|
|
if (i + j < length) {
|
|
|
|
for (k = 1; ((i + j + k) < length) && (fset[i + j + k] != ' '); k++);
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
|
|
|
if (j < 3 || (j < 5 && k > 2)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Change to shifting back rather than latching back */
|
2020-05-16 21:22:33 +12:00
|
|
|
/* Inverts the same figures recommended by Annex E note 3 */
|
2016-02-20 22:38:03 +13:00
|
|
|
for (k = 0; k < j; k++) {
|
|
|
|
fset[i + k] = 'n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Decide on mode using same system as PDF417 and rules of ISO 15417 Annex E */
|
|
|
|
indexliste = 0;
|
|
|
|
indexchaine = 0;
|
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
mode = c128_parunmodd(src[indexchaine]);
|
|
|
|
if (mode == C128_ABORC
|
|
|
|
&& (symbol->symbology == BARCODE_CODE128AB
|
|
|
|
|| (manual_set[indexchaine] == 'A' || manual_set[indexchaine] == 'B'))) {
|
2021-10-21 11:05:30 +13:00
|
|
|
mode = C128_AORB;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
list[1][indexliste] = mode;
|
2022-10-14 01:33:59 +13:00
|
|
|
while ((list[1][indexliste] == mode) && (indexchaine < length)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
list[0][indexliste]++;
|
|
|
|
indexchaine++;
|
2022-10-14 01:33:59 +13:00
|
|
|
if (indexchaine == length) {
|
2020-05-22 05:22:28 +12:00
|
|
|
break;
|
|
|
|
}
|
2023-01-15 13:22:43 +13:00
|
|
|
mode = c128_parunmodd(src[indexchaine]);
|
|
|
|
if (mode == C128_ABORC
|
|
|
|
&& (symbol->symbology == BARCODE_CODE128AB
|
|
|
|
|| (manual_set[indexchaine] == 'A' || manual_set[indexchaine] == 'B'))) {
|
2021-10-21 11:05:30 +13:00
|
|
|
mode = C128_AORB;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2023-01-15 13:22:43 +13:00
|
|
|
if (manual_set[indexchaine] != manual_set[indexchaine - 1]) {
|
|
|
|
break;
|
|
|
|
}
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
indexliste++;
|
2022-10-14 01:33:59 +13:00
|
|
|
} while (indexchaine < length);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
if (src == src_buf) {
|
|
|
|
/* Need to re-index `manual_set` to have sames indexes as `list` blocks for `c128_dxsmooth()` */
|
|
|
|
j = 0;
|
|
|
|
for (i = 1; i < indexliste; i++) {
|
|
|
|
j += list[0][i - 1];
|
|
|
|
manual_set[i] = manual_set[j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c128_dxsmooth(list, &indexliste, src == src_buf ? manual_set : NULL);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Resolve odd length C128_LATCHC blocks */
|
|
|
|
if ((list[1][0] == C128_LATCHC) && (list[0][0] & 1)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Rule 2 */
|
|
|
|
list[0][1]++;
|
|
|
|
list[0][0]--;
|
|
|
|
if (indexliste == 1) {
|
|
|
|
list[0][1] = 1;
|
2021-10-21 11:05:30 +13:00
|
|
|
list[1][1] = C128_LATCHB;
|
2016-02-20 22:38:03 +13:00
|
|
|
indexliste = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (indexliste > 1) {
|
|
|
|
for (i = 1; i < indexliste; i++) {
|
2021-10-21 11:05:30 +13:00
|
|
|
if ((list[1][i] == C128_LATCHC) && (list[0][i] & 1)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Rule 3b */
|
|
|
|
list[0][i - 1]++;
|
|
|
|
list[0][i]--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Put set data into set[]. Giving NULL as source as used to resolve odd C blocks which has been done above */
|
|
|
|
c128_put_in_set(list, indexliste, set, NULL /*source*/);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2020-05-16 21:22:33 +12:00
|
|
|
if (symbol->debug & ZINT_DEBUG_PRINT) {
|
2023-01-15 13:22:43 +13:00
|
|
|
printf("Data: %.*s (%d)\n", length, src, length);
|
2022-10-14 01:33:59 +13:00
|
|
|
printf(" Set: %.*s\n", length, set);
|
|
|
|
printf("FSet: %.*s\n", length, fset);
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Now we can calculate how long the barcode is going to be - and stop it from
|
|
|
|
being too long */
|
2020-05-16 21:22:33 +12:00
|
|
|
last_set = set[0];
|
2022-10-14 01:33:59 +13:00
|
|
|
for (i = 0; i < length; i++) {
|
2016-02-20 22:38:03 +13:00
|
|
|
if ((set[i] == 'a') || (set[i] == 'b')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2; /* 1 codeword */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
if ((fset[i] == 'f') || (fset[i] == 'n')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2; /* May be overestimate if in latch */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
if (((set[i] == 'A') || (set[i] == 'B')) || (set[i] == 'C')) {
|
|
|
|
if (set[i] != last_set) {
|
|
|
|
last_set = set[i];
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == 0) {
|
|
|
|
if (fset[i] == 'F') {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 4; /* 2 codewords */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((fset[i] == 'F') && (fset[i - 1] != 'F')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 4;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
if ((fset[i] != 'F') && (fset[i - 1] == 'F')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 4;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (set[i] == 'C') {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 1; /* Half a codeword */
|
2016-02-20 22:38:03 +13:00
|
|
|
} else {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
2022-10-15 03:05:33 +13:00
|
|
|
if (glyph_count > 120) { /* 60 * 2 */
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "341: Input too long (60 symbol character maximum)");
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* So now we know what start character to use - we can get on with it! */
|
|
|
|
if (symbol->output_options & READER_INIT) {
|
|
|
|
/* Reader Initialisation mode */
|
|
|
|
switch (set[0]) {
|
|
|
|
case 'A': /* Start A */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 103;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'A';
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 96; /* FNC3 */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B': /* Start B */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 104;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'B';
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 96; /* FNC3 */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'C': /* Start C */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 104; /* Start B */
|
|
|
|
values[bar_characters++] = 96; /* FNC3 */
|
|
|
|
values[bar_characters++] = 99; /* Code C */
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'C';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Normal mode */
|
|
|
|
switch (set[0]) {
|
|
|
|
case 'A': /* Start A */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 103;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'A';
|
|
|
|
break;
|
|
|
|
case 'B': /* Start B */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 104;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'B';
|
|
|
|
break;
|
|
|
|
case 'C': /* Start C */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 105;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'C';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fset[0] == 'F') {
|
|
|
|
switch (current_set) {
|
|
|
|
case 'A':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 101;
|
|
|
|
values[bar_characters++] = 101;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 100;
|
|
|
|
values[bar_characters++] = 100;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Encode the data */
|
|
|
|
read = 0;
|
|
|
|
do {
|
|
|
|
|
|
|
|
if ((read != 0) && (set[read] != current_set)) {
|
|
|
|
/* Latch different code set */
|
|
|
|
switch (set[read]) {
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'A':
|
|
|
|
values[bar_characters++] = 101;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'A';
|
|
|
|
break;
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'B':
|
|
|
|
values[bar_characters++] = 100;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'B';
|
|
|
|
break;
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'C':
|
|
|
|
values[bar_characters++] = 99;
|
2016-02-20 22:38:03 +13:00
|
|
|
current_set = 'C';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (read != 0) {
|
|
|
|
if ((fset[read] == 'F') && (f_state == 0)) {
|
|
|
|
/* Latch beginning of extended mode */
|
|
|
|
switch (current_set) {
|
|
|
|
case 'A':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 101;
|
|
|
|
values[bar_characters++] = 101;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 100;
|
|
|
|
values[bar_characters++] = 100;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 1;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((fset[read] == ' ') && (f_state == 1)) {
|
|
|
|
/* Latch end of extended mode */
|
|
|
|
switch (current_set) {
|
|
|
|
case 'A':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 101;
|
|
|
|
values[bar_characters++] = 101;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 0;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 100;
|
|
|
|
values[bar_characters++] = 100;
|
2022-10-14 01:33:59 +13:00
|
|
|
f_state = 0;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-14 01:33:59 +13:00
|
|
|
if ((fset[read] == 'f' && f_state == 0) || (fset[read] == 'n' && f_state == 1)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Shift to or from extended mode */
|
|
|
|
switch (current_set) {
|
|
|
|
case 'A':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 101; /* FNC 4 */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B':
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 100; /* FNC 4 */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((set[read] == 'a') || (set[read] == 'b')) {
|
|
|
|
/* Insert shift character */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 98;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (set[read]) { /* Encode data characters */
|
|
|
|
case 'a':
|
2023-01-15 13:22:43 +13:00
|
|
|
case 'A': c128_set_a(src[read], values, &bar_characters);
|
2016-02-20 22:38:03 +13:00
|
|
|
read++;
|
|
|
|
break;
|
|
|
|
case 'b':
|
2023-01-15 13:22:43 +13:00
|
|
|
case 'B': (void) c128_set_b(src[read], values, &bar_characters);
|
2016-02-20 22:38:03 +13:00
|
|
|
read++;
|
|
|
|
break;
|
2023-01-15 13:22:43 +13:00
|
|
|
case 'C': c128_set_c(src[read], src[read + 1], values, &bar_characters);
|
2016-02-20 22:38:03 +13:00
|
|
|
read += 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-10-14 01:33:59 +13:00
|
|
|
} while (read < length);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Destination setting and check digit calculation */
|
|
|
|
memcpy(d, C128Table[values[0]], 6);
|
|
|
|
d += 6;
|
|
|
|
total_sum = values[0];
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
for (i = 1; i < bar_characters; i++, d += 6) {
|
|
|
|
memcpy(d, C128Table[values[i]], 6);
|
|
|
|
total_sum += values[i] * i; /* Note can't overflow as 106 * 60 * 60 = 381600 */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
total_sum %= 103;
|
|
|
|
memcpy(d, C128Table[total_sum], 6);
|
|
|
|
d += 6;
|
|
|
|
values[bar_characters++] = total_sum;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* Stop character */
|
2021-10-21 11:05:30 +13:00
|
|
|
memcpy(d, "2331112", 7);
|
|
|
|
d += 7;
|
|
|
|
values[bar_characters++] = 106;
|
2020-05-16 21:22:33 +12:00
|
|
|
|
|
|
|
if (symbol->debug & ZINT_DEBUG_PRINT) {
|
2023-06-12 12:25:55 +12:00
|
|
|
fputs("Codewords:", stdout);
|
2020-05-16 21:22:33 +12:00
|
|
|
for (i = 0; i < bar_characters; i++) {
|
|
|
|
printf(" %d", values[i]);
|
|
|
|
}
|
2020-07-11 06:39:32 +12:00
|
|
|
printf(" (%d)\n", bar_characters);
|
2021-11-22 04:49:32 +13:00
|
|
|
printf("Barspaces: %.*s\n", (int) (d - dest), dest);
|
2022-11-11 11:13:41 +13:00
|
|
|
printf("Checksum: %d\n", total_sum);
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
|
|
|
#ifdef ZINT_TEST
|
|
|
|
if (symbol->debug & ZINT_DEBUG_TEST) {
|
|
|
|
debug_test_codeword_dump_int(symbol, values, bar_characters);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
expand(symbol, dest, d - dest);
|
2020-07-19 11:13:03 +12:00
|
|
|
|
2021-06-20 00:11:23 +12:00
|
|
|
/* ISO/IEC 15417:2007 leaves dimensions/height as application specification */
|
|
|
|
|
2023-06-12 12:25:55 +12:00
|
|
|
(void) hrt_cpy_iso8859_1(symbol, src, length); /* Truncation can't happen */
|
2020-07-19 11:13:03 +12:00
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
return error_number;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2021-06-20 00:11:23 +12:00
|
|
|
/* Handle EAN-128 (Now known as GS1-128), and composite version if `cc_mode` set */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int gs1_128_cc(struct zint_symbol *symbol, unsigned char source[], int length, const int cc_mode,
|
2021-06-20 00:11:23 +12:00
|
|
|
const int cc_rows) {
|
2022-10-14 01:33:59 +13:00
|
|
|
int i, values[C128_MAX] = {0}, bar_characters = 0, read, total_sum;
|
2021-06-20 00:11:23 +12:00
|
|
|
int error_number, warn_number = 0, indexchaine, indexliste;
|
2020-05-22 05:22:28 +12:00
|
|
|
int list[2][C128_MAX] = {{0}};
|
|
|
|
char set[C128_MAX] = {0}, mode, last_set;
|
2022-10-15 03:05:33 +13:00
|
|
|
int glyph_count = 0; /* Codeword estimate times 2 */
|
2016-02-20 22:38:03 +13:00
|
|
|
char dest[1000];
|
2021-10-21 11:05:30 +13:00
|
|
|
char *d = dest;
|
2022-10-14 01:33:59 +13:00
|
|
|
int separator_row = 0, linkage_flag = 0;
|
2020-05-16 21:22:33 +12:00
|
|
|
int reduced_length;
|
2022-07-15 03:01:30 +12:00
|
|
|
unsigned char *reduced = (unsigned char *) z_alloca(length + 1);
|
2020-05-22 05:22:28 +12:00
|
|
|
|
|
|
|
if (length > C128_MAX) {
|
2020-04-03 09:55:58 +13:00
|
|
|
/* This only blocks ridiculously long input - the actual length of the
|
2016-02-20 22:38:03 +13:00
|
|
|
resulting barcode depends on the type of data, so this is trapped later */
|
2021-07-07 06:53:31 +12:00
|
|
|
sprintf(symbol->errtxt, "342: Input too long (%d character maximum)", C128_MAX);
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if part of a composite symbol make room for the separator pattern */
|
2020-07-30 07:43:08 +12:00
|
|
|
if (symbol->symbology == BARCODE_GS1_128_CC) {
|
2016-02-20 22:38:03 +13:00
|
|
|
separator_row = symbol->rows;
|
|
|
|
symbol->row_height[symbol->rows] = 1;
|
|
|
|
symbol->rows += 1;
|
|
|
|
}
|
|
|
|
|
2019-10-17 22:06:21 +13:00
|
|
|
error_number = gs1_verify(symbol, source, length, reduced);
|
2021-01-19 09:10:52 +13:00
|
|
|
if (error_number >= ZINT_ERROR) {
|
2019-10-17 22:06:21 +13:00
|
|
|
return error_number;
|
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
|
2020-12-22 08:30:07 +13:00
|
|
|
reduced_length = (int) ustrlen(reduced);
|
2019-10-17 22:06:21 +13:00
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Decide on mode using same system as PDF417 and rules of ISO 15417 Annex E */
|
|
|
|
indexliste = 0;
|
|
|
|
indexchaine = 0;
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
mode = c128_parunmodd(reduced[indexchaine]);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
do {
|
|
|
|
list[1][indexliste] = mode;
|
2020-05-16 21:22:33 +12:00
|
|
|
while ((list[1][indexliste] == mode) && (indexchaine < reduced_length)) {
|
2016-02-20 22:38:03 +13:00
|
|
|
list[0][indexliste]++;
|
|
|
|
indexchaine++;
|
2020-05-22 05:22:28 +12:00
|
|
|
if (indexchaine == reduced_length) {
|
|
|
|
break;
|
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
mode = c128_parunmodd(reduced[indexchaine]);
|
2019-10-17 22:06:21 +13:00
|
|
|
if (reduced[indexchaine] == '[') {
|
2021-10-21 11:05:30 +13:00
|
|
|
mode = C128_ABORC;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
indexliste++;
|
2020-05-16 21:22:33 +12:00
|
|
|
} while (indexchaine < reduced_length);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2023-01-15 13:22:43 +13:00
|
|
|
c128_dxsmooth(list, &indexliste, NULL /*manual_set*/);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Put set data into set[], resolving odd C blocks */
|
|
|
|
c128_put_in_set(list, indexliste, set, reduced);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2020-05-16 21:22:33 +12:00
|
|
|
if (symbol->debug & ZINT_DEBUG_PRINT) {
|
|
|
|
printf("Data: %s (%d)\n", reduced, reduced_length);
|
|
|
|
printf(" Set: %.*s\n", reduced_length, set);
|
|
|
|
}
|
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Now we can calculate how long the barcode is going to be - and stop it from
|
2021-08-06 03:34:45 +12:00
|
|
|
being too long */
|
2020-05-16 21:22:33 +12:00
|
|
|
last_set = set[0];
|
|
|
|
for (i = 0; i < reduced_length; i++) {
|
2021-08-10 23:04:25 +12:00
|
|
|
if ((set[i] == 'A') || (set[i] == 'B') || (set[i] == 'C')) {
|
2016-02-20 22:38:03 +13:00
|
|
|
if (set[i] != last_set) {
|
|
|
|
last_set = set[i];
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2; /* 1 codeword */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2021-08-10 23:04:25 +12:00
|
|
|
} else if ((set[i] == 'a') || (set[i] == 'b')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
2019-10-17 22:06:21 +13:00
|
|
|
if ((set[i] == 'C') && (reduced[i] != '[')) {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 1; /* Half a codeword */
|
2016-02-20 22:38:03 +13:00
|
|
|
} else {
|
2022-10-15 03:05:33 +13:00
|
|
|
glyph_count += 2;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
2022-10-15 03:05:33 +13:00
|
|
|
if (glyph_count > 120) { /* 60 * 2 */
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "344: Input too long (60 symbol character maximum)");
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* So now we know what start character to use - we can get on with it! */
|
|
|
|
switch (set[0]) {
|
|
|
|
case 'A': /* Start A */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 103; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B': /* Start B */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 104;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'C': /* Start C */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 105;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 102;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* Encode the data */
|
|
|
|
read = 0;
|
|
|
|
do {
|
|
|
|
|
|
|
|
if ((read != 0) && (set[read] != set[read - 1])) { /* Latch different code set */
|
|
|
|
switch (set[read]) {
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'A':
|
|
|
|
values[bar_characters++] = 101; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'B':
|
|
|
|
values[bar_characters++] = 100;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
2021-10-21 11:05:30 +13:00
|
|
|
case 'C':
|
|
|
|
values[bar_characters++] = 99;
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((set[read] == 'a') || (set[read] == 'b')) {
|
|
|
|
/* Insert shift character */
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 98; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
2019-10-17 22:06:21 +13:00
|
|
|
if (reduced[read] != '[') {
|
2016-02-20 22:38:03 +13:00
|
|
|
switch (set[read]) { /* Encode data characters */
|
|
|
|
case 'A':
|
|
|
|
case 'a':
|
2021-10-21 11:05:30 +13:00
|
|
|
c128_set_a(reduced[read], values, &bar_characters); /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
read++;
|
|
|
|
break;
|
|
|
|
case 'B':
|
|
|
|
case 'b':
|
2021-10-21 11:05:30 +13:00
|
|
|
(void) c128_set_b(reduced[read], values, &bar_characters);
|
2016-02-20 22:38:03 +13:00
|
|
|
read++;
|
|
|
|
break;
|
|
|
|
case 'C':
|
2021-10-21 11:05:30 +13:00
|
|
|
c128_set_c(reduced[read], reduced[read + 1], values, &bar_characters);
|
2016-02-20 22:38:03 +13:00
|
|
|
read += 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = 102;
|
2016-02-20 22:38:03 +13:00
|
|
|
read++;
|
|
|
|
}
|
2020-05-16 21:22:33 +12:00
|
|
|
} while (read < reduced_length);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* "...note that the linkage flag is an extra code set character between
|
|
|
|
the last data character and the Symbol Check Character" (GS1 Specification) */
|
|
|
|
|
|
|
|
/* Linkage flags in GS1-128 are determined by ISO/IEC 24723 section 7.4 */
|
|
|
|
|
2021-06-20 00:11:23 +12:00
|
|
|
switch (cc_mode) {
|
2016-02-20 22:38:03 +13:00
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
/* CC-A or CC-B 2D component */
|
2020-05-16 21:22:33 +12:00
|
|
|
switch (set[reduced_length - 1]) {
|
2021-08-06 03:34:45 +12:00
|
|
|
case 'A': linkage_flag = 100; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B': linkage_flag = 99;
|
|
|
|
break;
|
|
|
|
case 'C': linkage_flag = 101;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
/* CC-C 2D component */
|
2020-05-16 21:22:33 +12:00
|
|
|
switch (set[reduced_length - 1]) {
|
2021-08-06 03:34:45 +12:00
|
|
|
case 'A': linkage_flag = 99; /* Not reached */
|
2016-02-20 22:38:03 +13:00
|
|
|
break;
|
|
|
|
case 'B': linkage_flag = 101;
|
|
|
|
break;
|
|
|
|
case 'C': linkage_flag = 100;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (linkage_flag != 0) {
|
2021-10-21 11:05:30 +13:00
|
|
|
values[bar_characters++] = linkage_flag;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
/* Destination setting and check digit calculation */
|
|
|
|
memcpy(d, C128Table[values[0]], 6);
|
|
|
|
d += 6;
|
|
|
|
total_sum = values[0];
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
for (i = 1; i < bar_characters; i++, d += 6) {
|
|
|
|
memcpy(d, C128Table[values[i]], 6);
|
|
|
|
total_sum += values[i] * i; /* Note can't overflow as 106 * 60 * 60 = 381600 */
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
2021-10-21 11:05:30 +13:00
|
|
|
total_sum %= 103;
|
|
|
|
memcpy(d, C128Table[total_sum], 6);
|
|
|
|
d += 6;
|
|
|
|
values[bar_characters++] = total_sum;
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* Stop character */
|
2021-10-21 11:05:30 +13:00
|
|
|
memcpy(d, "2331112", 7);
|
|
|
|
d += 7;
|
|
|
|
values[bar_characters++] = 106;
|
2020-05-16 21:22:33 +12:00
|
|
|
|
|
|
|
if (symbol->debug & ZINT_DEBUG_PRINT) {
|
2023-06-12 12:25:55 +12:00
|
|
|
fputs("Codewords:", stdout);
|
2020-05-16 21:22:33 +12:00
|
|
|
for (i = 0; i < bar_characters; i++) {
|
|
|
|
printf(" %d", values[i]);
|
|
|
|
}
|
2020-07-11 06:39:32 +12:00
|
|
|
printf(" (%d)\n", bar_characters);
|
2021-11-22 04:49:32 +13:00
|
|
|
printf("Barspaces: %.*s\n", (int) (d - dest), dest);
|
2020-05-16 21:22:33 +12:00
|
|
|
}
|
2020-05-22 05:22:28 +12:00
|
|
|
#ifdef ZINT_TEST
|
|
|
|
if (symbol->debug & ZINT_DEBUG_TEST) {
|
|
|
|
debug_test_codeword_dump_int(symbol, values, bar_characters);
|
|
|
|
}
|
|
|
|
#endif
|
2020-05-16 21:22:33 +12:00
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
expand(symbol, dest, d - dest);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
/* Add the separator pattern for composite symbols */
|
2020-07-30 07:43:08 +12:00
|
|
|
if (symbol->symbology == BARCODE_GS1_128_CC) {
|
2016-02-20 22:38:03 +13:00
|
|
|
for (i = 0; i < symbol->width; i++) {
|
|
|
|
if (!(module_is_set(symbol, separator_row + 1, i))) {
|
|
|
|
set_module(symbol, separator_row, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
if (symbol->output_options & COMPLIANT_HEIGHT) {
|
|
|
|
/* GS1 General Specifications 21.0.1 5.12.3.2 table 2, including footnote (**):
|
|
|
|
same as ITF-14: "in case of further space constraints" height 5.8mm / 1.016mm (X max) ~ 5.7;
|
|
|
|
default 31.75mm / 0.495mm ~ 64.14 */
|
|
|
|
const float min_height = stripf(5.8f / 1.016f);
|
|
|
|
const float default_height = stripf(31.75f / 0.495f);
|
|
|
|
if (symbol->symbology == BARCODE_GS1_128_CC) {
|
|
|
|
/* Pass back via temporary linear structure */
|
|
|
|
symbol->height = symbol->height ? min_height : default_height;
|
|
|
|
} else {
|
|
|
|
warn_number = set_height(symbol, min_height, default_height, 0.0f, 0 /*no_errtxt*/);
|
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
} else {
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
const float height = 50.0f;
|
|
|
|
if (symbol->symbology == BARCODE_GS1_128_CC) {
|
|
|
|
symbol->height = height - cc_rows * (cc_mode == 3 ? 3 : 2) - 1.0f;
|
|
|
|
} else {
|
|
|
|
(void) set_height(symbol, 0.0f, height, 0.0f, 1 /*no_errtxt*/);
|
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
}
|
|
|
|
|
2020-12-22 08:30:07 +13:00
|
|
|
for (i = 0; i < length; i++) {
|
2016-02-20 22:38:03 +13:00
|
|
|
if (source[i] == '[') {
|
|
|
|
symbol->text[i] = '(';
|
2021-07-14 04:39:03 +12:00
|
|
|
} else if (source[i] == ']') {
|
2016-02-20 22:38:03 +13:00
|
|
|
symbol->text[i] = ')';
|
2021-07-14 04:39:03 +12:00
|
|
|
} else {
|
|
|
|
symbol->text[i] = source[i];
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-20 00:11:23 +12:00
|
|
|
return error_number ? error_number : warn_number;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle EAN-128 (Now known as GS1-128) */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int gs1_128(struct zint_symbol *symbol, unsigned char source[], int length) {
|
|
|
|
return gs1_128_cc(symbol, source, length, 0 /*cc_mode*/, 0 /*cc_rows*/);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
2008-09-19 20:49:30 +12:00
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* Add check digit if encoding an NVE18 symbol */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int nve18(struct zint_symbol *symbol, unsigned char source[], int length) {
|
2021-05-15 03:04:34 +12:00
|
|
|
int error_number, zeroes;
|
|
|
|
unsigned char ean128_equiv[23];
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-05-15 03:04:34 +12:00
|
|
|
if (length > 17) {
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "345: Input too long (17 character maximum)");
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
if (!is_sane(NEON_F, source, length)) {
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "346: Invalid character in data (digits only)");
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
2021-05-15 03:04:34 +12:00
|
|
|
zeroes = 17 - length;
|
2021-05-15 23:23:46 +12:00
|
|
|
ustrcpy(ean128_equiv, symbol->input_mode & GS1PARENS_MODE ? "(00)" : "[00]");
|
2021-05-15 03:04:34 +12:00
|
|
|
memset(ean128_equiv + 4, '0', zeroes);
|
|
|
|
ustrcpy(ean128_equiv + 4 + zeroes, source);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
2021-12-09 07:12:20 +13:00
|
|
|
ean128_equiv[21] = gs1_check_digit(ean128_equiv + 4, 17);
|
2016-02-20 22:38:03 +13:00
|
|
|
ean128_equiv[22] = '\0';
|
|
|
|
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
error_number = gs1_128(symbol, ean128_equiv, 22);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
return error_number;
|
2008-10-13 10:05:53 +13:00
|
|
|
}
|
|
|
|
|
2016-02-20 22:38:03 +13:00
|
|
|
/* EAN-14 - A version of EAN-128 */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int ean14(struct zint_symbol *symbol, unsigned char source[], int length) {
|
2016-02-20 22:38:03 +13:00
|
|
|
int error_number, zeroes;
|
2021-05-15 03:04:34 +12:00
|
|
|
unsigned char ean128_equiv[19];
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
if (length > 13) {
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "347: Input too long (13 character maximum)");
|
2016-02-20 22:38:03 +13:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
2021-10-21 11:05:30 +13:00
|
|
|
if (!is_sane(NEON_F, source, length)) {
|
2021-07-07 06:53:31 +12:00
|
|
|
strcpy(symbol->errtxt, "348: Invalid character in data (digits only)");
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
2016-02-20 22:38:03 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
zeroes = 13 - length;
|
2021-05-15 23:23:46 +12:00
|
|
|
ustrcpy(ean128_equiv, symbol->input_mode & GS1PARENS_MODE ? "(01)" : "[01]");
|
2016-02-20 22:38:03 +13:00
|
|
|
memset(ean128_equiv + 4, '0', zeroes);
|
|
|
|
ustrcpy(ean128_equiv + 4 + zeroes, source);
|
|
|
|
|
2021-12-09 07:12:20 +13:00
|
|
|
ean128_equiv[17] = gs1_check_digit(ean128_equiv + 4, 13);
|
2016-02-20 22:38:03 +13:00
|
|
|
ean128_equiv[18] = '\0';
|
|
|
|
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
error_number = gs1_128(symbol, ean128_equiv, 18);
|
2016-02-20 22:38:03 +13:00
|
|
|
|
|
|
|
return error_number;
|
2008-09-19 20:49:30 +12:00
|
|
|
}
|
2020-08-03 23:13:05 +12:00
|
|
|
|
2022-08-06 10:51:24 +12:00
|
|
|
/* DPD (Deutscher Paketdienst) Code */
|
2022-11-11 11:13:41 +13:00
|
|
|
/* Specification at https://esolutions.dpd.com/dokumente/DPD_Parcel_Label_Specification_2.4.1_EN.pdf
|
|
|
|
* and identification tag info (Barcode ID) at https://esolutions.dpd.com/dokumente/DPD_Routing_Database_1.3_EN.pdf */
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
INTERNAL int dpd(struct zint_symbol *symbol, unsigned char source[], int length) {
|
2020-08-03 23:13:05 +12:00
|
|
|
int error_number = 0;
|
|
|
|
int i, p;
|
2022-11-11 11:13:41 +13:00
|
|
|
unsigned char ident_tag;
|
|
|
|
unsigned char local_source_buf[29];
|
|
|
|
unsigned char *local_source;
|
2020-08-03 23:13:05 +12:00
|
|
|
const int mod = 36;
|
2022-11-11 11:13:41 +13:00
|
|
|
const int relabel = symbol->option_2 == 1; /* A "relabel" has no identification tag */
|
2022-07-15 03:01:30 +12:00
|
|
|
int cd; /* Check digit */
|
2021-09-27 11:55:16 +13:00
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
if ((length != 27 && length != 28) || (length == 28 && relabel)) {
|
|
|
|
if (relabel) {
|
|
|
|
strcpy(symbol->errtxt, "830: DPD relabel input wrong length (27 characters required)");
|
|
|
|
} else {
|
|
|
|
strcpy(symbol->errtxt, "349: DPD input wrong length (27 or 28 characters required)");
|
|
|
|
}
|
2020-08-03 23:13:05 +12:00
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
if (length == 27 && !relabel) {
|
|
|
|
local_source_buf[0] = '%';
|
|
|
|
ustrcpy(local_source_buf + 1, source);
|
|
|
|
local_source = local_source_buf;
|
|
|
|
length++;
|
|
|
|
} else {
|
|
|
|
local_source = source;
|
|
|
|
}
|
|
|
|
|
|
|
|
ident_tag = local_source[0];
|
2021-09-27 11:55:16 +13:00
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
to_upper(local_source + !relabel, length - !relabel);
|
|
|
|
if (!is_sane(KRSET_F, local_source + !relabel, length - !relabel)) {
|
|
|
|
if (local_source == local_source_buf || relabel) {
|
|
|
|
strcpy(symbol->errtxt, "300: Invalid character in data (alphanumerics only)");
|
|
|
|
} else {
|
|
|
|
strcpy(symbol->errtxt, "299: Invalid character in data (alphanumerics only after first character)");
|
|
|
|
}
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
2020-08-03 23:13:05 +12:00
|
|
|
}
|
2021-09-27 11:55:16 +13:00
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
if ((ident_tag < 32) || (ident_tag > 127)) {
|
|
|
|
strcpy(symbol->errtxt, "343: Invalid DPD identification tag (first character), ASCII values 32 to 127 only");
|
2020-08-03 23:13:05 +12:00
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
|
|
|
}
|
2021-09-27 11:55:16 +13:00
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
error_number = code128(symbol, local_source, length); /* Only returns errors, not warnings */
|
2021-06-20 00:11:23 +12:00
|
|
|
|
|
|
|
if (error_number < ZINT_ERROR) {
|
2022-11-11 11:13:41 +13:00
|
|
|
if (!(symbol->output_options & (BARCODE_BOX | BARCODE_BIND | BARCODE_BIND_TOP))) {
|
|
|
|
/* If no option has been selected then uses default bind top option */
|
|
|
|
symbol->output_options |= BARCODE_BIND_TOP; /* Note won't extend over quiet zones for DPD */
|
|
|
|
if (symbol->border_width == 0) { /* Allow override if non-zero */
|
|
|
|
symbol->border_width = 3; /* From examples, not mentioned in spec */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
if (symbol->output_options & COMPLIANT_HEIGHT) {
|
2022-11-11 11:13:41 +13:00
|
|
|
/* DPD Parcel Label Specification Version 2.4.1 (19.01.2021) Section 4.6.1.2
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
25mm / 0.4mm (X max) = 62.5 min, 25mm / 0.375 (X) ~ 66.66 default */
|
2022-11-11 11:13:41 +13:00
|
|
|
if (relabel) { /* If relabel then half-size */
|
|
|
|
error_number = set_height(symbol, 31.25f, stripf(12.5f / 0.375f), 0.0f, 0 /*no_errtxt*/);
|
|
|
|
} else {
|
|
|
|
error_number = set_height(symbol, 62.5f, stripf(25.0f / 0.375f), 0.0f, 0 /*no_errtxt*/);
|
|
|
|
}
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
} else {
|
2022-11-11 11:13:41 +13:00
|
|
|
(void) set_height(symbol, 0.0f, relabel ? 25.0f : 50.0f, 0.0f, 1 /*no_errtxt*/);
|
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 12:13:39 +13:00
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
|
|
|
|
cd = mod;
|
|
|
|
|
|
|
|
p = 0;
|
2022-11-11 11:13:41 +13:00
|
|
|
for (i = !relabel; i < length; i++) {
|
|
|
|
symbol->text[p] = local_source[i];
|
2021-06-20 00:11:23 +12:00
|
|
|
p++;
|
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
cd += posn(KRSET, local_source[i]);
|
2021-06-20 00:11:23 +12:00
|
|
|
if (cd > mod) cd -= mod;
|
|
|
|
cd *= 2;
|
|
|
|
if (cd >= (mod + 1)) cd -= mod + 1;
|
|
|
|
|
2022-11-11 11:13:41 +13:00
|
|
|
switch (i + relabel) {
|
2021-06-20 00:11:23 +12:00
|
|
|
case 4:
|
|
|
|
case 7:
|
|
|
|
case 11:
|
|
|
|
case 15:
|
|
|
|
case 19:
|
|
|
|
case 21:
|
|
|
|
case 24:
|
|
|
|
case 27:
|
|
|
|
symbol->text[p] = ' ';
|
|
|
|
p++;
|
|
|
|
break;
|
|
|
|
}
|
2020-08-03 23:13:05 +12:00
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
|
|
|
|
cd = mod + 1 - cd;
|
|
|
|
if (cd == mod) cd = 0;
|
|
|
|
|
|
|
|
if (cd < 10) {
|
|
|
|
symbol->text[p] = cd + '0';
|
|
|
|
} else {
|
|
|
|
symbol->text[p] = (cd - 10) + 'A';
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
|
|
|
|
symbol->text[p] = '\0';
|
2022-11-11 11:13:41 +13:00
|
|
|
|
|
|
|
if (error_number == 0) {
|
|
|
|
/* Some compliance checks */
|
|
|
|
if (!is_sane(NEON_F, local_source + length - 16, 16)) {
|
|
|
|
if (!is_sane(NEON_F, local_source + length - 3, 3)) { /* 3-digit Country Code (ISO 3166-1) */
|
|
|
|
strcpy(symbol->errtxt, "831: Destination Country Code (last 3 characters) should be numeric");
|
|
|
|
} else if (!is_sane(NEON_F, local_source + length - 6, 3)) { /* 3-digit Service Code */
|
|
|
|
strcpy(symbol->errtxt, "832: Service Code (characters 6-4 from end) should be numeric");
|
|
|
|
} else { /* Last 10 characters of Tracking No. */
|
- API: add new zint_symbol `dpmm` field for output resolution (BMP/
EMF/PCX/PNG/TIF only, i.e. excluding EPS, GIF & SVG)
- Add support for specifying scale by X-dimension and resolution
with new option `--scalexdimdp` for CLI/Tcl & new API function
`ZBarcode_Scale_From_XdimDp()` (+ `ZBarcode_XdimDp_From_Scale()`
& `ZBarcode_Default_Xdim()`) and new GUI popup; manual: document
- BMP/EMF/PCX/PNG/TIF: use new `dpmm` resolution field (for EMF
following Inkscape)
- backend_qt: add `dpmm()`, `vectorWidth()`, `vectorHeight()`,
`noPng()`, `getVersion()`, `takesGS1AIData()`, & `XdimDp` stuff
incl. new `QZintXdimDp` struct for passing around scale vars &
use in `getAsCLI()`; add comments
- Raise `scale` limit to 200 (from 100) to allow for large dpmm
- output: create directories & subdirectories as necessary for
output path using new function `out_fopen()` and use in BMP/EMF/
EPS/GIF/PCX/PNG/SVG/TIF
- DPLEIT/DPIDENT: format HRT according to (incomplete)
documentation, and set default height to 72X (from 50X)
- CODE128B renamed to CODE128AB as can use subsets A and/or B
- CODABAR: fix minimum height calc
- EMF: fix indexing of handles (zero-based not 1-based)
- GUI: fix symbology zap (previous technique of clearing and
re-loading settings without doing a sync no longer works);
fix UPCEAN guard descent enable
- MAILMARK: better error message if input < 14 characters
- GUI: add "Default" button for DAFT tracker ratio & enable/disable
various default buttons; use new `takesGS1AIData()` to
enable/disable GS1-specific checkboxes
- CLI: use new `validate_float()` to parse float options (7
significant digits allowed only, no scientific notation)
- DATAMATRIX/GRIDMATRIX/PDF417/QR/ULTRA: micro-optimize structapp
ID parse
- library/CLI: fiddle with static asserts (make CHAR_BIT sensitive,
supposedly)
- win32/README: update building libpng (assembly removed)
- README.linux: document incompatibility of Qt6 >= 6.3
- manual: expand Barcode Studio waffle
- test suite: change range separator to hyphen and allow multiple
excludes
2022-12-03 10:39:01 +13:00
|
|
|
strcpy(symbol->errtxt,
|
|
|
|
"833: Last 10 characters of Tracking Number (characters 16-7 from end) should be numeric");
|
2022-11-11 11:13:41 +13:00
|
|
|
}
|
|
|
|
error_number = ZINT_WARN_NONCOMPLIANT;
|
|
|
|
}
|
|
|
|
}
|
2020-08-03 23:13:05 +12:00
|
|
|
}
|
2021-06-20 00:11:23 +12:00
|
|
|
|
2020-08-03 23:13:05 +12:00
|
|
|
return error_number;
|
|
|
|
}
|
Add multiple segments support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, HANXIN, MAXICODE, MICROPDF417, PDF417, QRCODE, RMQR, ULTRA
RMQR: fix ECI encoding (wrong bit length for indicator)
MICROQR: check versions M1 and M2 for allowed characters so as to give
better error messages
DOTCODE: some small optimizations
common.c: add is_chr(), segs_length(), segs_cpy()
CODEONE/CODE128/DOTCODE/GRIDMATRIX/HANXIN/MAXICODE/QRCODE/ULTRA: add
namespace prefixes to static funcs/data
includes: use Z_ prefix, unuse double underscore prefixes (guard defines)
manual.txt: compress some tables using double/treble column sets
2022-05-10 06:50:50 +12:00
|
|
|
|
2022-12-10 03:51:26 +13:00
|
|
|
/* Universal Postal Union S10 */
|
|
|
|
/* https://www.upu.int/UPU/media/upu/files/postalSolutions/programmesAndServices/standards/S10-12.pdf */
|
|
|
|
INTERNAL int upu_s10(struct zint_symbol *symbol, unsigned char source[], int length) {
|
|
|
|
int i, j;
|
|
|
|
unsigned char local_source[13 + 1];
|
|
|
|
unsigned char have_check_digit = '\0';
|
|
|
|
int check_digit;
|
2023-06-12 12:25:55 +12:00
|
|
|
static const char weights[8] = { 8, 6, 4, 2, 3, 5, 9, 7 };
|
2022-12-10 03:51:26 +13:00
|
|
|
int error_number = 0, warn_number = 0;
|
|
|
|
|
|
|
|
if (length != 12 && length != 13) {
|
|
|
|
strcpy(symbol->errtxt, "834: Input must be 12 or 13 characters long");
|
|
|
|
return ZINT_ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
if (length == 13) { /* Includes check digit - remove for now */
|
|
|
|
have_check_digit = source[10];
|
|
|
|
memcpy(local_source, source, 10);
|
|
|
|
ustrcpy(local_source + 10, source + 11);
|
|
|
|
} else {
|
|
|
|
ustrcpy(local_source, source);
|
|
|
|
}
|
|
|
|
to_upper(local_source, length);
|
|
|
|
|
|
|
|
if (!z_isupper(local_source[0]) || !z_isupper(local_source[1])) {
|
|
|
|
strcpy(symbol->errtxt, "835: Invalid character in Service Indictor (first 2 characters) (alphabetic only)");
|
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
|
|
|
}
|
|
|
|
if (!is_sane(NEON_F, local_source + 2, 12 - 4) || (have_check_digit && !z_isdigit(have_check_digit))) {
|
|
|
|
sprintf(symbol->errtxt, "836: Invalid character in Serial Number (middle %d characters) (digits only)",
|
|
|
|
have_check_digit ? 9 : 8);
|
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
|
|
|
}
|
|
|
|
if (!z_isupper(local_source[10]) || !z_isupper(local_source[11])) {
|
|
|
|
strcpy(symbol->errtxt, "837: Invalid character in Country Code (last 2 characters) (alphabetic only)");
|
|
|
|
return ZINT_ERROR_INVALID_DATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
check_digit = 0;
|
|
|
|
for (i = 2; i < 10; i++) { /* Serial Number only */
|
|
|
|
check_digit += ctoi(local_source[i]) * weights[i - 2];
|
|
|
|
}
|
|
|
|
check_digit %= 11;
|
|
|
|
check_digit = 11 - check_digit;
|
|
|
|
if (check_digit == 10) {
|
|
|
|
check_digit = 0;
|
|
|
|
} else if (check_digit == 11) {
|
|
|
|
check_digit = 5;
|
|
|
|
}
|
|
|
|
if (have_check_digit && ctoi(have_check_digit) != check_digit) {
|
|
|
|
sprintf(symbol->errtxt, "838: Invalid check digit '%c', expecting '%c'", have_check_digit, itoc(check_digit));
|
|
|
|
return ZINT_ERROR_INVALID_CHECK;
|
|
|
|
}
|
|
|
|
/* Add in (back) check digit */
|
|
|
|
local_source[12] = local_source[11];
|
|
|
|
local_source[11] = local_source[10];
|
|
|
|
local_source[10] = itoc(check_digit);
|
|
|
|
local_source[13] = '\0';
|
|
|
|
|
|
|
|
/* Do some checks on the Service Indicator (first char only) and Country Code */
|
|
|
|
if (strchr("JKSTW", local_source[0]) != NULL) { /* These are reserved & cannot be assigned */
|
|
|
|
strcpy(symbol->errtxt, "839: Invalid Service Indicator (first character should not be any of \"JKSTW\")");
|
|
|
|
error_number = ZINT_WARN_NONCOMPLIANT;
|
|
|
|
} else if (strchr("FHIOXY", local_source[0]) != NULL) { /* These aren't allocated as of spec Oct 2017 */
|
|
|
|
strcpy(symbol->errtxt, "840: Non-standard Service Indicator (first 2 characters)");
|
|
|
|
error_number = ZINT_WARN_NONCOMPLIANT;
|
2022-12-10 04:56:13 +13:00
|
|
|
} else if (!gs1_iso3166_alpha2(local_source + 11)) {
|
2022-12-10 03:51:26 +13:00
|
|
|
strcpy(symbol->errtxt, "841: Country code (last two characters) is not ISO 3166-1");
|
|
|
|
error_number = ZINT_WARN_NONCOMPLIANT;
|
|
|
|
}
|
|
|
|
|
|
|
|
(void) code128(symbol, local_source, 13); /* Only error returned is TOO_LONG which can't happen */
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
for (i = 0; i < 13; i++) {
|
|
|
|
if (i == 2 || i == 5 || i == 8 || i == 11) {
|
|
|
|
symbol->text[j++] = ' ';
|
|
|
|
}
|
|
|
|
symbol->text[j++] = local_source[i];
|
|
|
|
}
|
|
|
|
symbol->text[j] = '\0';
|
|
|
|
|
|
|
|
if (symbol->output_options & COMPLIANT_HEIGHT) {
|
|
|
|
/* Universal Postal Union S10 Section 8, using max X 0.51mm & minimum height 12.5mm or 15% of width */
|
|
|
|
const float min_height_min = stripf(12.5f / 0.51f);
|
|
|
|
float min_height = stripf(symbol->width * 0.15f);
|
|
|
|
if (min_height < min_height_min) {
|
|
|
|
min_height = min_height_min;
|
|
|
|
}
|
|
|
|
/* Using 50 as default as none recommended */
|
|
|
|
warn_number = set_height(symbol, min_height, min_height > 50.0f ? min_height : 50.0f, 0.0f, 0 /*no_errtxt*/);
|
|
|
|
} else {
|
|
|
|
(void) set_height(symbol, 0.0f, 50.0f, 0.0f, 1 /*no_errtxt*/);
|
|
|
|
}
|
|
|
|
|
|
|
|
return error_number ? error_number : warn_number;
|
|
|
|
}
|
|
|
|
|
Add multiple segments support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, HANXIN, MAXICODE, MICROPDF417, PDF417, QRCODE, RMQR, ULTRA
RMQR: fix ECI encoding (wrong bit length for indicator)
MICROQR: check versions M1 and M2 for allowed characters so as to give
better error messages
DOTCODE: some small optimizations
common.c: add is_chr(), segs_length(), segs_cpy()
CODEONE/CODE128/DOTCODE/GRIDMATRIX/HANXIN/MAXICODE/QRCODE/ULTRA: add
namespace prefixes to static funcs/data
includes: use Z_ prefix, unuse double underscore prefixes (guard defines)
manual.txt: compress some tables using double/treble column sets
2022-05-10 06:50:50 +12:00
|
|
|
/* vim: set ts=4 sw=4 et : */
|