zint/backend_tcl/zint.c

1195 lines
40 KiB
C
Raw Normal View History

2015-08-18 14:23:28 +02:00
/* zint_tcl.c TCL binding for zint */
/*
zint - the open source tcl binding to the zint barcode library
Copyright (C) 2014 Harald Oehlmann <oehhar@users.sourceforge.net>
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2015-08-18 14:23:28 +02:00
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
2015-08-18 14:23:28 +02: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
without specific prior written permission.
2015-08-18 14:23:28 +02: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
OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2015-08-18 14:23:28 +02:00
SUCH DAMAGE.
*/
/*
History
2014-06-16 2.5.0 HaO
First implementation
2016-09-14 2.5.1 HaO
- Added Codablock F options "-rows".
- Adopted to new image format of zint
2016-10-14 2.5.2 HaO
2017-10-23 21:34:31 +02:00
- Include the upstream reverted image format
2016-12-12 2.5.3 HaO
- No changes here, take 2.5.1 framework files
2017-10-23 21:34:31 +02:00
2017-05-12 2.6.0 HaO
- No changes here, take 2.6 framework files
2017-10-23 21:34:31 +02:00
2017-08-29 2.6.1 HaO
- Framework 2.6.1 extensions
- EAN/UPC Codes with included check digit
- UPNQR Code
- Misspelled symbology: AztecRunes
2017-10-23 21:34:31 +02:00
2017-10-23 2.6.2 HaO
- Framework 2.6.2 bugfixes
- Allow dll unload
2018-02-13 2.6.3 HaO
- Framework trunk update
- Added VIN and MailMark symbologies.
2018-11-02 2.6.4 HaO
- Framework trunk update
- Add options -bold, -dotted, -dotsize, -dmre, -eci
- Implemented ECI logic
2019-09-01 2.6.5 HaO
- Framework 2.6.5 update
- Add option -gssep
2019-09-18 11:58:57 +02:00
2019-09-18 2.6.6 HaO
- Framework 2.6.6 update
2019-10-07 2.6.7 HaO
- Framework 2.6.7 update
2019-12-05 20:16:39 +01:00
2019-12-05 2.7.0 HaO
- Framework 2.7.0 update
2019-12-05 20:16:39 +01:00
- Add symbology rmqr
2020-02-01 19:36:50 +01:00
2020-02-01 2.7.1 HaO
- Framework 2.7.1 update
2020-04-06 HaO
- Added option -fullmultibyte
2020-04-07 2.8.0 HaO
- Added symbology "UltraCode".
2020-05-19 HaO
- Added option -separator to specify stacked symbology separator width
- -cols maximum changed from 66 to 67
2020-07-27 2.9.0 HaO
- added option "-addongap"
- Renamed symbology names:
- Matrix2of5 -> Standard2of5
- PDF417Trunc -> PDF417Compact
- RSS14Stacked -> GS1DataBarStacked
- RSS14Stacked -> GS1DataBarStacked
- RSS14StackedOmni -> GS1DataBarSstackedOmni
- RSS14ExpandedStacked -> GS1DataBarExpandedStacked
- OneCode -> USPSIntelligentMail
- EAN128-CC -> GS1-128-CC
- RSS14-CC -> GS1DataBarOmni-CC
- RSSLimited-CC -> GS1DataBarLimited-CC
- RSSExpandedStacked-CC -> GS1DataBarExpanded-CC
- RSSEXPanded-CC -> GS1DataBarExpanded-CC
- RSS14Stacked-CC -> GS1DataBarStacked-CC
- RSS14Omni-CC -> GS1DataBarStackedOmni-CC
- RSSExpandedStacked-CC -> GS1DataBarExpandedStacked-CC
*** Potential incompatibility ***
2020-08-04 2.10.0 HaO
- added symbology "DPDCode"
- Alpha channel support added:
- added option -nobackground
- also allow RRGGBBAA for -fg and -bg options
2015-08-18 14:23:28 +02:00
*/
#if defined(__WIN32__) || defined(_WIN32) || defined(WIN32)
2015-08-18 14:23:28 +02:00
#pragma warning(disable : 4201 4214 4514)
#define STRICT
#define WIN32_LEAN_AND_MEAN
/* TCL Defines */
#define DLL_BUILD
#include <windows.h>
/* Define ERROR_INVALID_DATA is also used by zint... */
#ifdef ERROR_INVALID_DATA
#undef ERROR_INVALID_DATA
#endif
#endif
2015-08-18 14:23:28 +02:00
#include <zint.h>
#include <string.h>
2015-08-18 14:23:28 +02:00
#if defined(__WIN32__) || defined(_WIN32) || defined(WIN32)
2015-08-18 14:23:28 +02:00
#define USE_TCL_STUBS
#define USE_TK_STUBS
#endif
2015-08-18 14:23:28 +02:00
#include <tcl.h>
#include <tk.h>
2015-08-18 14:23:28 +02:00
#undef EXPORT
#if defined(__WIN32__) || defined(_WIN32) || defined(WIN32)
#define EXPORT __declspec(dllexport)
#else
#define EXPORT
#endif
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* > File option defines */
#define VERSION "2.9.0"
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* >>>>> Hepler defines */
#define STRING( x ) #x
/*----------------------------------------------------------------------------*/
/* >>>> External Prototypes (exports) */
EXPORT int Zint_Init (Tcl_Interp *interp);
2017-10-23 21:34:31 +02:00
EXPORT int Zint_Unload (Tcl_Interp *Interp, int Flags);
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* >>>> local prototypes */
static int Zint(ClientData unused, Tcl_Interp *interp, int objc,
Tcl_Obj *CONST objv[]);
static int Encode(Tcl_Interp *interp, int objc,
Tcl_Obj *CONST objv[]);
static int is_fullmultibyte(struct zint_symbol* symbol);
static int is_stackable(const int symbology);
static int is_extendable(const int symbology);
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* >>>> File Global Variables */
/* >> List of Codes */
static char *s_code_list[] = {
"Code11",
"Standard2of5",
"Interleaved2of5",
"IATAC2of5",
"Logic2of5",
"Ind2of5",
"Code39",
"Code39Extended",
2017-10-23 21:34:31 +02:00
"EAN",
"EAN+Check",
"GS1-128",
"Codabar",
"Code128",
"DPLeit",
"DPIdent",
"Code16K",
"Code49",
"Code93",
"Flat",
"GS1DataBar",
"GS1DataBarLimited",
"GS1DataBarExpanded",
"Telepen",
2017-10-23 21:34:31 +02:00
"UPC-A",
"UPC-A+Check",
"UPC-E",
2017-10-23 21:34:31 +02:00
"UPC-E+Check",
"Postnet",
"MSIPlessey",
"FIM",
"Logmars",
"Pharma",
"PZN",
"PharmaTwo",
"PDF417",
"PDF417Compact",
"MaxiCode",
"QR",
"Code128B",
"AusPost",
"AusReply",
"AusRoute",
"AusRedirect",
"ISBN",
"RM4SCC",
"Datamatrix",
"EAN14",
"VIN",
"CodablockF",
"NVE18",
"JapanPost",
"KoreaPost",
"GS1DataBarStacked",
"GS1DataBarSstackedOmni",
"GS1DataBarExpandedStacked",
"Planet",
"DPDCode",
"MicroPDF417",
"USPSIntelligentMail",
"Plessey",
"TelepenNum",
"ITF14",
"KIX",
"Aztec",
"DAFT",
"MicroQR",
"HIBC-128",
"HIBC-39",
"HIBC-DM",
"HIBC-QR",
"HIBC-PDF",
"HIBC-MicroPDF",
"HIBC-CodablockF",
2017-10-23 21:34:31 +02:00
"HIBCAztec",
"DotCode",
"HanXin",
"MailMark",
2017-08-29 22:19:55 +02:00
"AztecRunes",
"Code32",
"EAN-CC",
"GS1-128-CC",
"GS1DataBarOmni-CC",
"GS1DataBarLimited-CC",
"GS1DataBarExpanded-CC",
"UPCA-CC",
"UPCE-CC",
"GS1DataBarStacked-CC",
"GS1DataBarStackedOmni-CC",
"GS1DataBarExpandedStacked-CC",
"Channel",
"CodeOne",
"GridMatrix",
"UPNQR",
"UltraCode",
2019-12-05 20:16:39 +01:00
"rMQR",
NULL};
2015-08-18 14:23:28 +02:00
static int s_code_number[] = {
BARCODE_CODE11,
BARCODE_C25STANDARD,
BARCODE_C25INTER,
BARCODE_C25IATA,
BARCODE_C25LOGIC,
BARCODE_C25IND,
BARCODE_CODE39,
BARCODE_EXCODE39,
2017-10-23 21:34:31 +02:00
BARCODE_EANX,
BARCODE_EANX_CHK,
BARCODE_GS1_128,
BARCODE_CODABAR,
BARCODE_CODE128,
BARCODE_DPLEIT,
BARCODE_DPIDENT,
BARCODE_CODE16K,
BARCODE_CODE49,
BARCODE_CODE93,
BARCODE_FLAT,
BARCODE_DBAR_OMN,
BARCODE_DBAR_LTD,
BARCODE_DBAR_EXP,
BARCODE_TELEPEN,
2017-10-23 21:34:31 +02:00
BARCODE_UPCA,
BARCODE_UPCA_CHK,
BARCODE_UPCE,
2017-10-23 21:34:31 +02:00
BARCODE_UPCE_CHK,
BARCODE_POSTNET,
BARCODE_MSI_PLESSEY,
BARCODE_FIM,
BARCODE_LOGMARS,
BARCODE_PHARMA,
BARCODE_PZN,
BARCODE_PHARMA_TWO,
BARCODE_PDF417,
BARCODE_PDF417COMP,
BARCODE_MAXICODE,
BARCODE_QRCODE,
BARCODE_CODE128B,
BARCODE_AUSPOST,
BARCODE_AUSREPLY,
BARCODE_AUSROUTE,
BARCODE_AUSREDIRECT,
BARCODE_ISBNX,
BARCODE_RM4SCC,
BARCODE_DATAMATRIX,
BARCODE_EAN14,
BARCODE_VIN,
BARCODE_CODABLOCKF,
BARCODE_NVE18,
BARCODE_JAPANPOST,
BARCODE_KOREAPOST,
BARCODE_DBAR_STK,
BARCODE_DBAR_OMNSTK,
BARCODE_DBAR_EXPSTK,
BARCODE_PLANET,
BARCODE_DPD,
BARCODE_MICROPDF417,
BARCODE_USPS_IMAIL,
BARCODE_PLESSEY,
BARCODE_TELEPEN_NUM,
BARCODE_ITF14,
BARCODE_KIX,
BARCODE_AZTEC,
BARCODE_DAFT,
BARCODE_MICROQR,
BARCODE_HIBC_128,
BARCODE_HIBC_39,
BARCODE_HIBC_DM,
BARCODE_HIBC_QR,
BARCODE_HIBC_PDF,
BARCODE_HIBC_MICPDF,
BARCODE_HIBC_BLOCKF,
2017-10-23 21:34:31 +02:00
BARCODE_HIBC_AZTEC,
BARCODE_DOTCODE,
BARCODE_HANXIN,
BARCODE_MAILMARK,
BARCODE_AZRUNE,
BARCODE_CODE32,
BARCODE_EANX_CC,
BARCODE_GS1_128_CC,
BARCODE_DBAR_OMN_CC,
BARCODE_DBAR_LTD_CC,
BARCODE_DBAR_EXP_CC,
BARCODE_UPCA_CC,
BARCODE_UPCE_CC,
BARCODE_DBAR_STK_CC,
BARCODE_DBAR_OMNSTK_CC,
BARCODE_DBAR_EXPSTK_CC,
BARCODE_CHANNEL,
BARCODE_CODEONE,
BARCODE_GRIDMATRIX,
BARCODE_UPNQR,
BARCODE_ULTRA,
2019-12-05 20:16:39 +01:00
BARCODE_RMQR,
0};
2015-08-18 14:23:28 +02:00
/* ECI TCL encoding names.
* The ECI comments are given after the name.
* A ** indicates encodings where native data must be delivered and not utf-8
*/
static char *s_eci_list[] = {
"iso8859-1", /* 3: ISO-8859-1 - Latin alphabet No. 1 (default)*/
"iso8859-2", /* 4: ISO-8859-2 - Latin alphabet No. 2*/
"iso8859-3", /* 5: ISO-8859-3 - Latin alphabet No. 3*/
"iso8859-4", /* 6: ISO-8859-4 - Latin alphabet No. 4*/
"iso8859-5", /* 7: ISO-8859-5 - Latin/Cyrillic alphabet*/
"iso8859-6", /* 8: ISO-8859-6 - Latin/Arabic alphabet*/
"iso8859-7", /* 9: ISO-8859-7 - Latin/Greek alphabet*/
"iso8859-9", /*10: ISO-8859-8 - Latin/Hebrew alphabet*/
"iso8859-9", /*11: ISO-8859-9 - Latin alphabet No. 5*/
"iso8859-10", /*12: ISO-8859-10 - Latin alphabet No. 6*/
"iso8859-11", /*13: ISO-8859-11 - Latin/Thai alphabet*/
"iso8859-13", /*15: ISO-8859-13 - Latin alphabet No. 7*/
"iso8859-14", /*16: ISO-8859-14 - Latin alphabet No. 8 (Celtic)*/
"iso8859-15", /*17: ISO-8859-15 - Latin alphabet No. 9*/
"iso8859-16", /*18: ISO-8859-16 - Latin alphabet No. 10*/
"jis0208", /*20: ** Shift-JIS (JISX 0208 amd JISX 0201)*/
"cp1250", /*21: Windows-1250*/
"cp1251", /*22: Windows-1251*/
"cp1252", /*23: Windows-1252*/
"cp1256", /*24: Windows-1256*/
"unicode", /*25: ** UCS-2 Unicode (High order byte first)*/
"utf-8", /*26: Unicode (UTF-8)*/
"ascii", /*27: ISO-646:1991 7-bit character set*/
"big5", /*28: ** Big-5 (Taiwan) Chinese Character Set*/
"euc-cn", /*29: ** GB (PRC) Chinese Character Set*/
"iso2022-kr", /*30: ** Korean Character Set (KSX1001:1998)*/
NULL
};
/* The ECI numerical number to pass to ZINT */
static int s_eci_number[] = {
3,4,5,6,7,8,9,10,11,12,13,15,16,17,18,20,21,22,23,24,25,26,27,28,29,30
};
/* Flag if an encoding is delivered as unicode. If not, native encoding is
* required.
* Those encodings are marked with "**" in the upper comments.
*/
static int s_eci_unicode_input[] = {
/* 3,4,5,6,7,8,9,10,11,12,13,15,16,17,18,20,21,22,23,24,25,26,27,28,29,30 */
1,1,1,1,1,1,1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 0, 0, 0
};
2015-08-18 14:23:28 +02:00
/* Version information */
static char version_string[] = VERSION;
/* Help text */
static char help_message[] = "zint tcl(stub,obj) dll\n"
" Generate barcode in tk images\n"
"Usage:\n"
2017-10-23 21:34:31 +02:00
" zint encode data photo option value...\n"
" data: data to encode in the symbol\n"
" photo: a tcl photo image handle ('p' after 'image create photo p')\n"
" Available options:\n"
" -addongap number: (7..12, default: 9) set add-on gap in multiple of module size (UPC/EAN-CC)\n"
" -barcode choice: symbology, use 'zint symbology' to get a list\n"
" -bind bool: bars above/below the code, size set by -border\n"
" -border integer: width of a border around the symbol. Use with -bind/-box 1\n"
" -box bool: box around bar code, size set be -border\n"
" -height integer: Symbol height in modules\n"
" -whitesp integer: horizontal quiet zone in modules\n"
" -fg color: set foreground color as 6 or 8 hex rrggbbaa\n"
" -bg color: set background color as 6 or 8 hex rrggbbaa\n"
" -nobackground bool: set background transparent\n"
" -cols integer: PDF417, Codablock F: number of columns\n"
" -rows integer: Codablock F: number of rows\n"
" -vers integer: Symbology option\n"
" -dmre bool: Allow Data Matrix Rectangular Extended\n"
" -separator 0..4 (default: 1) : Stacked symbologies: separator width\n"
" -rotate angle: Image rotation by 0,90 or 270 degrees\n"
" -secure integer: EC Level (PDF417, QR)\n"
" -mode: Structured primary data mode (Maxicode, Composite)\n"
" -primary text: Structured primary data (Maxicode, Composite)\n"
" -dotty bool: use dots instead of boxes for matrix codes\n"
" -dotsize number: radius ratio of dots from 0.01 to 1.0\n"
" -scale double: Scale the image to this factor\n"
" -format binary|unicode|gs1: input data format. Default:unicode\n"
" -fullmultibyte: allow multibyte compaction for xQR, HanXin, Gridmatrix\n"
" -gssep bool: for gs1, use gs as separator instead fnc1 (Datamatrix only)\n"
" -eci number: ECI to use\n"
" -notext bool: no interpretation line\n"
" -square bool: force Data Matrix symbols to be square\n"
" -init bool: Create reader initialisation symbol (Code 128, Data Matrix)\n"
" -smalltext bool: tiny interpretation line font\n"
" -bold bool: use bold text\n"
" -to {x0 y0 ?width? ?height?}: place to put in photo image\n"
"\n"
"zint symbologies: List available symbologies\n"
"zint eci: List available eci tables\n"
" zint help\n"
" zint version\n"
;
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* Exported symbols */
#if defined(__WIN32__) || defined(_WIN32) || defined(WIN32)
EXPORT BOOL WINAPI DllEntryPoint (HINSTANCE hInstance,
DWORD seginfo, LPVOID lpCmdLine)
2015-08-18 14:23:28 +02:00
{
/* Don't do anything, so just return true */
return TRUE;
}
#endif
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* Initialisation Procedures */
2017-09-10 12:03:09 -03:00
EXPORT int Zint_Init (Tcl_Interp *interp)
2015-08-18 14:23:28 +02:00
{
/*------------------------------------------------------------------------*/
#ifdef USE_TCL_STUBS
2017-10-16 19:26:54 +02:00
if (Tcl_InitStubs(interp, "8.1", 0) == NULL)
#else
2017-10-16 19:26:54 +02:00
if (Tcl_PkgRequire(interp, "Tcl", "8.1", 0) == NULL)
#endif
{
return TCL_ERROR;
}
/*------------------------------------------------------------------------*/
#ifdef USE_TK_STUBS
2017-10-16 19:26:54 +02:00
if (Tk_InitStubs(interp, "8.1", 0) == NULL)
#else
2017-10-16 19:26:54 +02:00
if (Tcl_PkgRequire(interp, "Tk", "8.1", 0) == NULL)
#endif
{
return TCL_ERROR;
}
/*------------------------------------------------------------------------*/
2017-10-16 19:26:54 +02:00
Tcl_CreateObjCommand(interp, "zint", Zint, (ClientData)NULL,
(Tcl_CmdDeleteProc *)NULL);
2017-10-16 19:26:54 +02:00
Tcl_PkgProvide (interp, "zint", version_string);
/*------------------------------------------------------------------------*/
return TCL_OK;
2015-08-18 14:23:28 +02:00
}
2017-10-23 21:34:31 +02:00
//------------------------------------------------------------------------------
// >>>> Unload Procedures
//------------------------------------------------------------------------------
EXPORT int Zint_Unload (Tcl_Interp *Interp, int Flags)
{
// Allow unload
return TCL_OK;
2017-10-23 21:34:31 +02:00
}
2015-08-18 14:23:28 +02:00
/*----------------------------------------------------------------------------*/
/* >>>>> Called routine */
/*----------------------------------------------------------------------------*/
/* Decode tcl commands */
static int Zint(ClientData unused, Tcl_Interp *interp, int objc,
Tcl_Obj *CONST objv[])
2015-08-18 14:23:28 +02:00
{
/* Option list and indexes */
enum iCommand {iEncode, iSymbologies, iECI, iVersion, iHelp};
/* choice of option */
int Index;
/*------------------------------------------------------------------------*/
/* > Check if option argument is given and decode it */
if (objc > 1)
{
char *subCmds[] = {"encode", "symbologies", "eci", "version", "help", NULL};
if(Tcl_GetIndexFromObj(interp, objv[1], (const char **) subCmds,
"option", 0, &Index)
== TCL_ERROR)
{
return TCL_ERROR;
}
} else {
Tcl_WrongNumArgs(interp, 1, objv, "option");
return TCL_ERROR;
}
/*------------------------------------------------------------------------*/
/* > Call functions in dependency of Index */
/*------------------------------------------------------------------------*/
switch (Index)
{
case iEncode:
return Encode(interp, objc, objv);
case iSymbologies:
2015-08-18 14:23:28 +02:00
{
Tcl_Obj *oRes;
int posCur;
oRes = Tcl_NewObj();
for (posCur = 0 ; s_code_list[posCur] != NULL; posCur++) {
if( ZBarcode_ValidID(s_code_number[posCur]) != 0) {
if (TCL_OK != Tcl_ListObjAppendElement(interp,
oRes, Tcl_NewStringObj(s_code_list[posCur],-1)))
2015-08-18 14:23:28 +02:00
{
return TCL_ERROR;
}
}
}
Tcl_SetObjResult(interp,oRes);
return TCL_OK;
}
case iECI:
{
Tcl_Obj *oRes;
int posCur;
oRes = Tcl_NewObj();
for (posCur = 0 ; s_eci_list[posCur] != NULL; posCur++) {
if (TCL_OK != Tcl_ListObjAppendElement(interp,
oRes, Tcl_NewStringObj(s_eci_list[posCur],-1)))
{
return TCL_ERROR;
}
}
Tcl_SetObjResult(interp,oRes);
return TCL_OK;
}
case iVersion:
Tcl_SetObjResult(interp,
Tcl_NewStringObj(version_string, -1));
return TCL_OK;
case iHelp:
default:
Tcl_SetObjResult(interp,
Tcl_NewStringObj(help_message, -1));
return TCL_OK;
}
2015-08-18 14:23:28 +02:00
}
/*----------------------------------------------------------------------------*/
/* >>>>> Encode */
/*----------------------------------------------------------------------------*/
/* Encode image */
static int Encode(Tcl_Interp *interp, int objc,
Tcl_Obj *CONST objv[])
2015-08-18 14:23:28 +02:00
{
struct zint_symbol *hSymbol;
Tcl_DString dsInput;
char *pStr = NULL;
int lStr;
Tcl_Encoding hZINTEncoding;
int rotate_angle=0;
int fError = 0;
Tcl_DString dString;
int optionPos;
int destX0 = 0;
int destY0 = 0;
int destWidth = 0;
int destHeight = 0;
int ECIIndex = 0;
int fFullMultiByte = 0;
int addon_gap = 0;
int Separator = 1;
/*------------------------------------------------------------------------*/
/* >> Check if at least data and object is given and a pair number of */
/* >> options */
if ( objc < 4 || (objc % 2) != 0 )
{
Tcl_WrongNumArgs(interp, 2, objv, "data photo ?-switch value?...");
return TCL_ERROR;
}
/*------------------------------------------------------------------------*/
/* >>> Prepare encoding */
hZINTEncoding = Tcl_GetEncoding(interp, "utf-8");
if (NULL == hZINTEncoding) {
return TCL_ERROR;
}
/*------------------------------------------------------------------------*/
/* >>> Prepare zint object */
hSymbol = ZBarcode_Create();
hSymbol->input_mode = UNICODE_MODE;
hSymbol->option_3 = 0;
/*------------------------------------------------------------------------*/
/* >> Decode options */
for (optionPos = 4; optionPos < objc; optionPos+=2) {
/*--------------------------------------------------------------------*/
/* Option list and indexes */
char *optionList[] = {
"-addongap", "-barcode", "-bg", "-bind", "-bold", "-border", "-box",
"-cols", "-dmre", "-dotsize", "-dotty", "-eci", "-fg", "-format",
"-gssep", "-height", "-init", "-mode", "-nobackground", "-notext",
"-primary", "-rotate", "-rows", "-scale", "-secure", "-smalltext",
"-square", "-to", "-vers", "-whitesp", "-fullmultibyte",
"-separator", NULL};
enum iOption {
iAddonGap, iBarcode, iBG, iBind, iBold, iBorder, iBox, iCols,
iDMRE, iDotSize, iDotty, iECI, iFG, iFormat, iGSSep, iHeight,
iInit, iMode, iNoBackground, iNoText, iPrimary, iRotate, iRows,
iScale, iSecure, iSmallText, iSquare, iTo, iVers,
iWhiteSp, iFullMultiByte, iSeparator
};
int optionIndex;
int intValue;
double doubleValue;
/*--------------------------------------------------------------------*/
if(Tcl_GetIndexFromObj(interp, objv[optionPos],
(const char **) optionList,
"zint option", optionPos-1, &optionIndex)
== TCL_ERROR)
{
fError = 1;
break;
}
/*--------------------------------------------------------------------*/
/* >> Decode object */
switch (optionIndex) {
case iBind:
case iBold:
case iBox:
case iDMRE:
case iDotty:
case iGSSep:
case iInit:
case iNoBackground:
case iNoText:
case iSmallText:
case iSquare:
case iFullMultiByte:
/* >> Binary options */
if (TCL_OK != Tcl_GetBooleanFromObj(interp, objv[optionPos+1],
&intValue))
{
fError = 1;
}
break;
case iFG:
case iBG:
/* >> Colors */
pStr = Tcl_GetStringFromObj(objv[optionPos+1],&lStr);
if (lStr != 6 && lStr != 8) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Color is not 6 or 8 hex",-1));
fError = 1;
}
break;
case iDotSize:
case iScale:
/* >> Float */
if (TCL_OK != Tcl_GetDoubleFromObj(interp, objv[optionPos+1],
&doubleValue))
{
fError = 1;
}
break;
case iAddonGap:
case iBorder:
case iCols:
case iHeight:
case iMode:
case iRotate:
case iRows:
case iSecure:
case iVers:
case iWhiteSp:
case iSeparator:
/* >> Int */
if (TCL_OK != Tcl_GetIntFromObj(interp, objv[optionPos+1],
&intValue))
{
fError = 1;
}
break;
case iPrimary:
/* > Primary String up to 90 characters */
/* > Output filename up to 250 characters */
Tcl_DStringInit(& dString);
pStr = Tcl_GetStringFromObj(objv[optionPos+1], &lStr);
Tcl_UtfToExternalDString( hZINTEncoding, pStr, lStr, &dString);
if (Tcl_DStringLength(&dString) > (optionIndex==iPrimary?90:250)) {
Tcl_DStringFree(&dString);
Tcl_SetObjResult(interp,Tcl_NewStringObj("String to long", -1));
fError = 1;
}
break;
}
if (fError) {
break;
}
/*--------------------------------------------------------------------*/
switch (optionIndex) {
case iAddonGap:
if (intValue < 7 || intValue > 12) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Invalid add-on gap value not within 7 to 12", -1));
fError = 1;
} else {
addon_gap = intValue;
}
break;
case iBind:
if (intValue) {
hSymbol->output_options |= BARCODE_BIND;
} else {
hSymbol->output_options &= ~BARCODE_BIND;
}
break;
case iBold:
if (intValue) {
hSymbol->output_options |= BOLD_TEXT;
} else {
hSymbol->output_options &= ~BOLD_TEXT;
}
break;
case iBox:
if (intValue) {
hSymbol->output_options |= BARCODE_BOX;
} else {
hSymbol->output_options &= ~BARCODE_BOX;
}
break;
case iDotSize:
if (doubleValue < 0.01) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Dot size below 0.01", -1));
fError = 1;
} else {
hSymbol->dot_size = (float)doubleValue;
}
break;
case iDotty:
if (intValue) {
hSymbol->output_options |= BARCODE_DOTTY_MODE;
} else {
hSymbol->output_options &= ~BARCODE_DOTTY_MODE;
}
break;
case iGSSep:
if (intValue) {
hSymbol->output_options |= GS1_GS_SEPARATOR;
} else {
hSymbol->output_options &= ~GS1_GS_SEPARATOR;
}
break;
case iFullMultiByte:
fFullMultiByte = intValue;
break;
case iECI:
if(Tcl_GetIndexFromObj(interp, objv[optionPos+1],
(const char **) s_eci_list,"-eci", optionPos, &ECIIndex)
== TCL_ERROR)
{
fError = 1;
} else {
hSymbol->eci = s_eci_number[ECIIndex];
}
break;
case iInit:
if (intValue) {
hSymbol->output_options |= READER_INIT;
} else {
hSymbol->output_options &= ~READER_INIT;
}
break;
case iSmallText:
if (intValue) {
hSymbol->output_options |= SMALL_TEXT;
} else {
hSymbol->output_options &= ~SMALL_TEXT;
}
break;
case iFG:
strncpy(hSymbol->fgcolour, pStr, lStr);
hSymbol->fgcolour[lStr]='\0';
break;
case iBG:
strncpy(hSymbol->bgcolour, pStr, lStr);
hSymbol->bgcolour[lStr]='\0';
break;
case iNoBackground:
if (intValue) {
strcpy(hSymbol->bgcolour, "ffffff00");
}
break;
case iNoText:
hSymbol->show_hrt = (intValue?0:1);
break;
case iSquare:
/* DM_SQUARE overwrites DM_DMRE */
if (intValue)
hSymbol->option_3 = DM_SQUARE;
break;
case iDMRE:
/* DM_DMRE overwrites DM_SQUARE */
if (intValue)
hSymbol->option_3 = DM_DMRE;
break;
case iScale:
2015-08-18 14:23:28 +02:00
if (doubleValue < 0.01) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Scale below 0.01", -1));
fError = 1;
2015-08-18 14:23:28 +02:00
} else {
hSymbol->scale = (float)doubleValue;
}
break;
case iBorder:
2015-08-18 14:23:28 +02:00
if (intValue < 0 || intValue > 1000) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Border out of range", -1));
fError = 1;
2015-08-18 14:23:28 +02:00
} else {
hSymbol->border_width = intValue;
2015-08-18 14:23:28 +02:00
}
break;
case iHeight:
2015-08-18 14:23:28 +02:00
if (intValue < 1 || intValue > 1000) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Height out of range", -1));
fError = 1;
2015-08-18 14:23:28 +02:00
} else {
hSymbol->height = intValue;
2015-08-18 14:23:28 +02:00
}
break;
case iSeparator:
if (intValue < 0 || intValue > 4) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Separator out of range", -1));
fError = 1;
} else {
Separator = intValue;
}
break;
case iCols:
case iVers:
/* >> Int in Option 2 */
2015-08-18 14:23:28 +02:00
if (intValue < 1
|| (optionIndex==iCols && intValue > 67)
|| (optionIndex==iVers && intValue > 47))
2015-08-18 14:23:28 +02:00
{
Tcl_SetObjResult(interp,
Tcl_NewStringObj("cols/vers out of range", -1));
fError = 1;
2015-08-18 14:23:28 +02:00
} else {
hSymbol->option_2 = intValue;
2015-08-18 14:23:28 +02:00
}
break;
case iSecure:
case iMode:
case iRows:
/* >> Int in Option 1 */
2015-08-18 14:23:28 +02:00
if ( (optionIndex==iSecure && (intValue < 1 || intValue > 8))
|| (optionIndex==iMode && (intValue < 0 || intValue > 6))
|| (optionIndex==iRows && (intValue < 0 || intValue > 44)))
2015-08-18 14:23:28 +02:00
{
Tcl_SetObjResult(interp,
Tcl_NewStringObj("secure/mode/rows out of range", -1));
fError = 1;
2015-08-18 14:23:28 +02:00
} else {
hSymbol->option_1 = intValue;
2015-08-18 14:23:28 +02:00
}
break;
case iPrimary:
strcpy(hSymbol->primary, Tcl_DStringValue( &dString ) );
Tcl_DStringFree(&dString);
break;
case iRotate:
2015-08-18 14:23:28 +02:00
/* >> Rotate angle */
/*----------------------------------------------------------------*/
{
char *rotateList[] = {"0", "90", "180", "270", NULL};
2015-08-18 14:23:28 +02:00
enum iRotate { iRotate0, iRotate90, iRotate180, iRotate270 };
/*------------------------------------------------------------*/
if(Tcl_GetIndexFromObj(interp, objv[optionPos+1],
(const char **) rotateList,
"rotate", optionPos, &intValue)
2015-08-18 14:23:28 +02:00
== TCL_ERROR)
{
fError = 1;
break;
}
switch (intValue) {
case iRotate90: rotate_angle = 90; break;
case iRotate180: rotate_angle = 180; break;
case iRotate270: rotate_angle = 270; break;
default: rotate_angle = 0; break;
}
}
break;
case iBarcode:
2015-08-18 14:23:28 +02:00
if(Tcl_GetIndexFromObj(interp, objv[optionPos+1],
(const char **) s_code_list,"-barcode", optionPos, &intValue)
2015-08-18 14:23:28 +02:00
== TCL_ERROR)
{
fError = 1;
} else {
hSymbol->symbology = s_code_number[intValue];
}
break;
case iWhiteSp:
hSymbol->whitespace_width = intValue;
break;
case iTo:
2015-08-18 14:23:28 +02:00
/* >> Decode the -to parameter as list of X0 Y0 ?Width Height? */
{
Tcl_Obj *poParam;
if (TCL_OK != Tcl_ListObjLength(interp,
objv[optionPos+1], &lStr))
2015-08-18 14:23:28 +02:00
{
fError = 1;
} else if ( ! ( lStr == 2 || lStr == 4 ) ) {
Tcl_SetObjResult(interp,
Tcl_NewStringObj(
"option -to not a list of 2 or 4", -1));
2015-08-18 14:23:28 +02:00
fError = 1;
} else if ((
TCL_OK != Tcl_ListObjIndex(interp, objv[optionPos+1],
0, &poParam)
|| TCL_OK != Tcl_GetIntFromObj(interp,poParam,&destX0)
|| TCL_OK != Tcl_ListObjIndex(interp, objv[optionPos+1],
1, &poParam)
|| TCL_OK != Tcl_GetIntFromObj(interp,poParam,&destY0)
|| lStr == 4) && (
TCL_OK != Tcl_ListObjIndex(interp, objv[optionPos+1],
2, &poParam)
|| TCL_OK != Tcl_GetIntFromObj(interp,poParam,
&destWidth)
|| TCL_OK != Tcl_ListObjIndex(interp, objv[optionPos+1],
3, &poParam)
|| TCL_OK != Tcl_GetIntFromObj(interp,poParam,
&destHeight)
))
2015-08-18 14:23:28 +02:00
{
fError = 1;
}
2015-08-18 14:23:28 +02:00
}
break;
2015-08-18 14:23:28 +02:00
case iFormat:
/* >> Format of the input data */
/*----------------------------------------------------------------*/
{
char *formatList[] = {"binary", "gs1", "unicode",NULL};
enum iFormat { iBinary, iGS1, iUnicode };
/*------------------------------------------------------------*/
if(Tcl_GetIndexFromObj(interp, objv[optionPos+1],
(const char **) formatList,
"format", optionPos, &intValue)
2015-08-18 14:23:28 +02:00
== TCL_ERROR)
{
fError = 1;
break;
}
switch (intValue) {
case iBinary: hSymbol->input_mode = DATA_MODE; break;
case iGS1: hSymbol->input_mode = GS1_MODE; break;
default: hSymbol->input_mode = UNICODE_MODE; break;
}
}
}
}
/*------------------------------------------------------------------------*/
/* >>> option_3 is set by two values depending on the symbology */
/* On wrong symbology, the option is ignored(as does the zint program)*/
if (fFullMultiByte && is_fullmultibyte(hSymbol)) {
hSymbol->option_3 = ZINT_FULL_MULTIBYTE;
} else if (Separator && is_stackable(hSymbol->symbology)) {
hSymbol->option_3 = Separator;
}
2015-08-18 14:23:28 +02:00
/*------------------------------------------------------------------------*/
/* >>> option_2 is set by two values depending on the symbology */
/* On wrong symbology, the option is ignored(as does the zint program)*/
if (addon_gap && is_extendable(hSymbol->symbology)) {
hSymbol->option_2 = addon_gap;
}
/*------------------------------------------------------------------------*/
/* >>> Prepare input dstring and encode it to ECI encoding*/
Tcl_DStringInit(& dsInput);
/*------------------------------------------------------------------------*/
if (!fError) {
/*--------------------------------------------------------------------*/
/* >>> Get input mode */
2015-08-18 14:23:28 +02:00
if (hSymbol->input_mode == DATA_MODE) {
/* Binary data */
pStr = (char *) Tcl_GetByteArrayFromObj(objv[2], &lStr);
2015-08-18 14:23:28 +02:00
} else {
/* UTF8 Data (or ECI encoding) */
if ( ! s_eci_unicode_input[ECIIndex]) {
/* For this ECI, the Data must be encoded in the ECI encoding */
Tcl_FreeEncoding(hZINTEncoding);
hZINTEncoding = Tcl_GetEncoding(interp, s_eci_list[ECIIndex]);
if (NULL == hZINTEncoding) {
/* Interpreter has error message */
fError = 1;
}
/* we must indicate binary data */
hSymbol->input_mode = DATA_MODE;
}
if (! fError ) {
pStr = Tcl_GetStringFromObj(objv[2], &lStr);
Tcl_UtfToExternalDString( hZINTEncoding, pStr, lStr, &dsInput);
pStr = Tcl_DStringValue( &dsInput );
lStr = Tcl_DStringLength( &dsInput );
}
2015-08-18 14:23:28 +02:00
}
}
/*------------------------------------------------------------------------*/
/* >>> Build symbol graphic */
if (! fError ) {
int ErrorNumber;
Tk_PhotoHandle hPhoto;
/*--------------------------------------------------------------------*/
/* call zint graphic creation to buffer */
ErrorNumber = ZBarcode_Encode_and_Buffer(hSymbol,
(unsigned char *) pStr, lStr, rotate_angle);
2015-08-18 14:23:28 +02:00
/*--------------------------------------------------------------------*/
/* >> Show a message */
if( 0 != ErrorNumber )
{
Tcl_SetObjResult(interp, Tcl_NewStringObj(hSymbol->errtxt, -1));
}
if( 5 <= ErrorNumber )
2015-08-18 14:23:28 +02:00
{
/* >> Encode error */
fError = 1;
} else if (
NULL == (hPhoto = Tk_FindPhoto(interp, Tcl_GetString(objv[3]))))
2015-08-18 14:23:28 +02:00
{
Tcl_SetObjResult(interp,
Tcl_NewStringObj("Unknown photo image", -1));
2015-08-18 14:23:28 +02:00
fError = 1;
} else {
Tk_PhotoImageBlock sImageBlock;
char * pImageRGBA = NULL;
if (hSymbol->alphamap == NULL) {
sImageBlock.pixelPtr = (unsigned char *) hSymbol->bitmap;
sImageBlock.width = hSymbol->bitmap_width;
sImageBlock.height = hSymbol->bitmap_height;
sImageBlock.pitch = 3*hSymbol->bitmap_width;
sImageBlock.pixelSize = 3;
sImageBlock.offset[0] = 0;
sImageBlock.offset[1] = 1;
sImageBlock.offset[2] = 2;
sImageBlock.offset[3] = 0;
} else {
int index;
/* Alpha channel present - prepare the image data in rgba order */
pImageRGBA = ckalloc(hSymbol->bitmap_width*hSymbol->bitmap_height*4);
for (index = 0; index < hSymbol->bitmap_width*hSymbol->bitmap_height; index++) {
pImageRGBA[index*4] = hSymbol->bitmap[index*3];
pImageRGBA[index*4+1] = hSymbol->bitmap[index*3+1];
pImageRGBA[index*4+2] = hSymbol->bitmap[index*3+2];
pImageRGBA[index*4+3] = hSymbol->alphamap[index];
}
sImageBlock.pixelPtr = (unsigned char *) pImageRGBA;
sImageBlock.width = hSymbol->bitmap_width;
sImageBlock.height = hSymbol->bitmap_height;
sImageBlock.pitch = 4*hSymbol->bitmap_width;
sImageBlock.pixelSize = 4;
sImageBlock.offset[0] = 0;
sImageBlock.offset[1] = 1;
sImageBlock.offset[2] = 2;
sImageBlock.offset[3] = 3;
}
2015-08-18 14:23:28 +02:00
if (0 == destWidth) {
destWidth = hSymbol->bitmap_width;
}
if (0 == destHeight) {
destHeight = hSymbol->bitmap_height;
}
if (TCL_OK != Tk_PhotoPutBlock(interp, hPhoto, &sImageBlock,
destX0, destY0, destWidth, destHeight,
TK_PHOTO_COMPOSITE_OVERLAY))
2015-08-18 14:23:28 +02:00
{
fError = 1;
}
if (pImageRGBA != NULL) {
ckfree(pImageRGBA);
}
2015-08-18 14:23:28 +02:00
}
}
/*------------------------------------------------------------------------*/
Tcl_FreeEncoding(hZINTEncoding);
Tcl_DStringFree(& dsInput);
ZBarcode_Delete(hSymbol);
/*------------------------------------------------------------------------*/
if (fError) {
return TCL_ERROR;
}
return TCL_OK;
2015-08-18 14:23:28 +02:00
}
2017-10-23 21:34:31 +02:00
static int is_fullmultibyte(struct zint_symbol* symbol) {
switch (symbol->symbology) {
case BARCODE_QRCODE:
case BARCODE_MICROQR:
//case BARCODE_HIBC_QR: Note character set restricted to ASCII subset
//case BARCODE_UPNQR: Note does not use Kanji mode
case BARCODE_RMQR:
case BARCODE_HANXIN:
case BARCODE_GRIDMATRIX:
return 1;
}
return 0;
}
/* Indicates which symbologies can have row binding
* Note: if change this must also change version in backend/common.c */
static int is_stackable(const int symbology) {
if (symbology < BARCODE_PDF417) {
return 1;
}
switch (symbology) {
case BARCODE_CODE128B:
case BARCODE_ISBNX:
case BARCODE_EAN14:
case BARCODE_NVE18:
case BARCODE_KOREAPOST:
case BARCODE_PLESSEY:
case BARCODE_TELEPEN_NUM:
case BARCODE_ITF14:
case BARCODE_CODE32:
case BARCODE_CODABLOCKF:
case BARCODE_HIBC_BLOCKF:
return 1;
}
return 0;
}
2017-10-23 21:37:52 +02:00
/* Indicates which symbols can have addon (EAN-2 and EAN-5)
* Note: if change this must also change version in backend/common.c */
static int is_extendable(const int symbology) {
switch (symbology) {
case BARCODE_EANX:
case BARCODE_EANX_CHK:
case BARCODE_UPCA:
case BARCODE_UPCA_CHK:
case BARCODE_UPCE:
case BARCODE_UPCE_CHK:
case BARCODE_ISBNX:
case BARCODE_EANX_CC:
case BARCODE_UPCA_CC:
case BARCODE_UPCE_CC:
return 1;
}
return 0;
}