433 lines
11 KiB
C++
433 lines
11 KiB
C++
/*
|
|
* Copyright 2023 siiky
|
|
* Copyright 2023 Axel Waggershauser
|
|
*/
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
#include "ZXingC.h"
|
|
|
|
#include "ZXingCpp.h"
|
|
|
|
#include <cstdlib>
|
|
#include <exception>
|
|
#include <string>
|
|
#include <string_view>
|
|
#include <tuple>
|
|
#include <utility>
|
|
|
|
using namespace ZXing;
|
|
|
|
static thread_local std::string lastErrorMsg;
|
|
|
|
template<typename R, typename T> R transmute_cast(const T& v) noexcept
|
|
{
|
|
static_assert(sizeof(T) == sizeof(R));
|
|
return *(const R*)(&v);
|
|
}
|
|
|
|
template<typename C, typename P = typename C::pointer>
|
|
P copy(const C& c) noexcept
|
|
{
|
|
auto ret = (P)malloc(c.size() + 1);
|
|
if (ret) {
|
|
memcpy(ret, c.data(), c.size());
|
|
ret[c.size()] = 0;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static uint8_t* copy(const ByteArray& ba, int* len) noexcept
|
|
{
|
|
// for convencience and as a safety measure, we NULL terminate even byte arrays
|
|
auto ret = copy(ba);
|
|
if (len)
|
|
*len = ret ? Size(ba) : 0;
|
|
return ret;
|
|
}
|
|
|
|
#define ZX_CHECK( GOOD, MSG ) \
|
|
if (!(GOOD)) { \
|
|
lastErrorMsg = MSG; \
|
|
return {}; \
|
|
}
|
|
|
|
#define ZX_CATCH(...) \
|
|
catch (std::exception & e) { \
|
|
lastErrorMsg = e.what(); \
|
|
} catch (...) { \
|
|
lastErrorMsg = "Unknown error"; \
|
|
} \
|
|
return __VA_ARGS__;
|
|
|
|
#define ZX_TRY(...) \
|
|
try { \
|
|
return __VA_ARGS__; \
|
|
} \
|
|
ZX_CATCH({})
|
|
|
|
|
|
static std::tuple<Barcodes, bool> ReadBarcodesAndSetLastError(const ZXing_ImageView* iv, const ZXing_ReaderOptions* opts,
|
|
int maxSymbols)
|
|
{
|
|
ZX_CHECK(iv, "ImageView param is NULL")
|
|
try {
|
|
auto o = opts ? *opts : ReaderOptions{};
|
|
if (maxSymbols)
|
|
o.setMaxNumberOfSymbols(maxSymbols);
|
|
return {ReadBarcodes(*iv, o), true};
|
|
}
|
|
ZX_CATCH({Barcodes{}, false})
|
|
}
|
|
|
|
extern "C" {
|
|
/*
|
|
* ZXing/ImageView.h
|
|
*/
|
|
|
|
ZXing_ImageView* ZXing_ImageView_new(const uint8_t* data, int width, int height, ZXing_ImageFormat format, int rowStride,
|
|
int pixStride)
|
|
{
|
|
ImageFormat cppformat = static_cast<ImageFormat>(format);
|
|
ZX_TRY(new ImageView(data, width, height, cppformat, rowStride, pixStride))
|
|
}
|
|
|
|
ZXing_ImageView* ZXing_ImageView_new_checked(const uint8_t* data, int size, int width, int height, ZXing_ImageFormat format,
|
|
int rowStride, int pixStride)
|
|
{
|
|
ImageFormat cppformat = static_cast<ImageFormat>(format);
|
|
ZX_TRY(new ImageView(data, size, width, height, cppformat, rowStride, pixStride))
|
|
}
|
|
|
|
void ZXing_ImageView_delete(ZXing_ImageView* iv)
|
|
{
|
|
delete iv;
|
|
}
|
|
|
|
void ZXing_ImageView_crop(ZXing_ImageView* iv, int left, int top, int width, int height)
|
|
{
|
|
*iv = iv->cropped(left, top, width, height);
|
|
}
|
|
|
|
void ZXing_ImageView_rotate(ZXing_ImageView* iv, int degree)
|
|
{
|
|
*iv = iv->rotated(degree);
|
|
}
|
|
|
|
void ZXing_Image_delete(ZXing_Image* img)
|
|
{
|
|
delete img;
|
|
}
|
|
|
|
const uint8_t* ZXing_Image_data(const ZXing_Image* img)
|
|
{
|
|
return img->data();
|
|
}
|
|
|
|
int ZXing_Image_width(const ZXing_Image* img)
|
|
{
|
|
return img->width();
|
|
}
|
|
|
|
int ZXing_Image_height(const ZXing_Image* img)
|
|
{
|
|
return img->height();
|
|
}
|
|
|
|
ZXing_ImageFormat ZXing_Image_format(const ZXing_Image* img)
|
|
{
|
|
return static_cast<ZXing_ImageFormat>(img->format());
|
|
}
|
|
|
|
/*
|
|
* ZXing/BarcodeFormat.h
|
|
*/
|
|
|
|
ZXing_BarcodeFormats ZXing_BarcodeFormatsFromString(const char* str)
|
|
{
|
|
if (!str)
|
|
return {};
|
|
try {
|
|
return transmute_cast<ZXing_BarcodeFormats>(BarcodeFormatsFromString(str));
|
|
}
|
|
ZX_CATCH(ZXing_BarcodeFormat_Invalid)
|
|
}
|
|
|
|
ZXing_BarcodeFormat ZXing_BarcodeFormatFromString(const char* str)
|
|
{
|
|
ZXing_BarcodeFormat res = ZXing_BarcodeFormatsFromString(str);
|
|
return BitHacks::CountBitsSet(res) == 1 ? res : ZXing_BarcodeFormat_Invalid;
|
|
}
|
|
|
|
char* ZXing_BarcodeFormatToString(ZXing_BarcodeFormat format)
|
|
{
|
|
return copy(ToString(static_cast<BarcodeFormat>(format)));
|
|
}
|
|
|
|
/*
|
|
* ZXing/ZXingCpp.h
|
|
*/
|
|
|
|
|
|
#ifdef ZXING_EXPERIMENTAL_API
|
|
ZXing_BarcodeFormats ZXing_SupportedBarcodeFormats(ZXing_Operation op)
|
|
{
|
|
return transmute_cast<ZXing_BarcodeFormats>(SupportedBarcodeFormats(static_cast<Operation>(op)));
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* ZXing/Barcode.h
|
|
*/
|
|
|
|
char* ZXing_ContentTypeToString(ZXing_ContentType type)
|
|
{
|
|
return copy(ToString(static_cast<ContentType>(type)));
|
|
}
|
|
|
|
char* ZXing_PositionToString(ZXing_Position position)
|
|
{
|
|
return copy(ToString(transmute_cast<Position>(position)));
|
|
}
|
|
|
|
|
|
bool ZXing_Barcode_isValid(const ZXing_Barcode* barcode)
|
|
{
|
|
return barcode != NULL && barcode->isValid();
|
|
}
|
|
|
|
ZXing_ErrorType ZXing_Barcode_errorType(const ZXing_Barcode* barcode)
|
|
{
|
|
return static_cast<ZXing_ErrorType>(barcode->error().type());
|
|
}
|
|
|
|
char* ZXing_Barcode_errorMsg(const ZXing_Barcode* barcode)
|
|
{
|
|
return copy(ToString(barcode->error()));
|
|
}
|
|
|
|
uint8_t* ZXing_Barcode_bytes(const ZXing_Barcode* barcode, int* len)
|
|
{
|
|
return copy(barcode->bytes(), len);
|
|
}
|
|
|
|
uint8_t* ZXing_Barcode_bytesECI(const ZXing_Barcode* barcode, int* len)
|
|
{
|
|
return copy(barcode->bytesECI(), len);
|
|
}
|
|
|
|
#define ZX_GETTER(TYPE, GETTER, TRANS) \
|
|
TYPE ZXing_Barcode_##GETTER(const ZXing_Barcode* barcode) { return TRANS(barcode->GETTER()); }
|
|
|
|
ZX_GETTER(ZXing_BarcodeFormat, format, static_cast<ZXing_BarcodeFormat>)
|
|
ZX_GETTER(ZXing_ContentType, contentType, static_cast<ZXing_ContentType>)
|
|
ZX_GETTER(char*, text, copy)
|
|
ZX_GETTER(char*, ecLevel, copy)
|
|
ZX_GETTER(char*, symbologyIdentifier, copy)
|
|
ZX_GETTER(ZXing_Position, position, transmute_cast<ZXing_Position>)
|
|
|
|
ZX_GETTER(int, orientation,)
|
|
ZX_GETTER(bool, hasECI,)
|
|
ZX_GETTER(bool, isInverted,)
|
|
ZX_GETTER(bool, isMirrored,)
|
|
ZX_GETTER(int, lineCount,)
|
|
|
|
void ZXing_Barcode_delete(ZXing_Barcode* barcode)
|
|
{
|
|
delete barcode;
|
|
}
|
|
|
|
void ZXing_Barcodes_delete(ZXing_Barcodes* barcodes)
|
|
{
|
|
delete barcodes;
|
|
}
|
|
|
|
int ZXing_Barcodes_size(const ZXing_Barcodes* barcodes)
|
|
{
|
|
return barcodes ? Size(*barcodes) : 0;
|
|
}
|
|
|
|
const ZXing_Barcode* ZXing_Barcodes_at(const ZXing_Barcodes* barcodes, int i)
|
|
{
|
|
if (!barcodes || i < 0 || i >= Size(*barcodes))
|
|
return NULL;
|
|
return &(*barcodes)[i];
|
|
}
|
|
|
|
ZXing_Barcode* ZXing_Barcodes_move(ZXing_Barcodes* barcodes, int i)
|
|
{
|
|
if (!barcodes || i < 0 || i >= Size(*barcodes))
|
|
return NULL;
|
|
|
|
ZX_TRY(new Barcode(std::move((*barcodes)[i])));
|
|
}
|
|
|
|
/*
|
|
* ZXing/ReaderOptions.h
|
|
*/
|
|
|
|
ZXing_ReaderOptions* ZXing_ReaderOptions_new()
|
|
{
|
|
ZX_TRY(new ReaderOptions());
|
|
}
|
|
|
|
void ZXing_ReaderOptions_delete(ZXing_ReaderOptions* opts)
|
|
{
|
|
delete opts;
|
|
}
|
|
|
|
#define ZX_PROPERTY(TYPE, GETTER, SETTER) \
|
|
TYPE ZXing_ReaderOptions_get##SETTER(const ZXing_ReaderOptions* opts) { return opts->GETTER(); } \
|
|
void ZXing_ReaderOptions_set##SETTER(ZXing_ReaderOptions* opts, TYPE val) { opts->set##SETTER(val); }
|
|
|
|
ZX_PROPERTY(bool, tryHarder, TryHarder)
|
|
ZX_PROPERTY(bool, tryRotate, TryRotate)
|
|
ZX_PROPERTY(bool, tryInvert, TryInvert)
|
|
ZX_PROPERTY(bool, tryDownscale, TryDownscale)
|
|
ZX_PROPERTY(bool, isPure, IsPure)
|
|
ZX_PROPERTY(bool, returnErrors, ReturnErrors)
|
|
ZX_PROPERTY(int, minLineCount, MinLineCount)
|
|
ZX_PROPERTY(int, maxNumberOfSymbols, MaxNumberOfSymbols)
|
|
|
|
#undef ZX_PROPERTY
|
|
|
|
void ZXing_ReaderOptions_setFormats(ZXing_ReaderOptions* opts, ZXing_BarcodeFormats formats)
|
|
{
|
|
opts->setFormats(static_cast<BarcodeFormat>(formats));
|
|
}
|
|
|
|
ZXing_BarcodeFormats ZXing_ReaderOptions_getFormats(const ZXing_ReaderOptions* opts)
|
|
{
|
|
auto v = opts->formats();
|
|
return transmute_cast<ZXing_BarcodeFormats>(v);
|
|
}
|
|
|
|
#define ZX_ENUM_PROPERTY(TYPE, GETTER, SETTER) \
|
|
ZXing_##TYPE ZXing_ReaderOptions_get##SETTER(const ZXing_ReaderOptions* opts) { return static_cast<ZXing_##TYPE>(opts->GETTER()); } \
|
|
void ZXing_ReaderOptions_set##SETTER(ZXing_ReaderOptions* opts, ZXing_##TYPE val) { opts->set##SETTER(static_cast<TYPE>(val)); }
|
|
|
|
ZX_ENUM_PROPERTY(Binarizer, binarizer, Binarizer)
|
|
ZX_ENUM_PROPERTY(EanAddOnSymbol, eanAddOnSymbol, EanAddOnSymbol)
|
|
ZX_ENUM_PROPERTY(TextMode, textMode, TextMode)
|
|
|
|
|
|
/*
|
|
* ZXing/ReadBarcode.h
|
|
*/
|
|
|
|
ZXing_Barcode* ZXing_ReadBarcode(const ZXing_ImageView* iv, const ZXing_ReaderOptions* opts)
|
|
{
|
|
auto [res, ok] = ReadBarcodesAndSetLastError(iv, opts, 1);
|
|
return !res.empty() ? new Barcode(std::move(res.front())) : NULL;
|
|
}
|
|
|
|
ZXing_Barcodes* ZXing_ReadBarcodes(const ZXing_ImageView* iv, const ZXing_ReaderOptions* opts)
|
|
{
|
|
auto [res, ok] = ReadBarcodesAndSetLastError(iv, opts, 0);
|
|
return !res.empty() || ok ? new Barcodes(std::move(res)) : NULL;
|
|
}
|
|
|
|
|
|
#ifdef ZXING_EXPERIMENTAL_API
|
|
/*
|
|
* ZXing/WriteBarcode.h
|
|
*/
|
|
|
|
ZXing_CreatorOptions* ZXing_CreatorOptions_new(ZXing_BarcodeFormat format)
|
|
{
|
|
ZX_TRY(new CreatorOptions(static_cast<BarcodeFormat>(format)));
|
|
}
|
|
|
|
void ZXing_CreatorOptions_delete(ZXing_CreatorOptions* opts)
|
|
{
|
|
delete opts;
|
|
}
|
|
|
|
#define ZX_PROPERTY(TYPE, GETTER, SETTER) \
|
|
TYPE ZXing_CreatorOptions_get##SETTER(const ZXing_CreatorOptions* opts) { return opts->GETTER(); } \
|
|
void ZXing_CreatorOptions_set##SETTER(ZXing_CreatorOptions* opts, TYPE val) { opts->GETTER(val); }
|
|
|
|
ZX_PROPERTY(bool, readerInit, ReaderInit)
|
|
ZX_PROPERTY(bool, forceSquareDataMatrix, ForceSquareDataMatrix)
|
|
|
|
#undef ZX_PROPERTY
|
|
|
|
//ZX_PROPERTY(BarcodeFormat, format, Format)
|
|
|
|
char* ZXing_CreatorOptions_getEcLevel(const ZXing_CreatorOptions* opts)
|
|
{
|
|
return copy(opts->ecLevel());
|
|
}
|
|
|
|
void ZXing_CreatorOptions_setEcLevel(ZXing_CreatorOptions* opts, const char* val)
|
|
{
|
|
opts->ecLevel(val);
|
|
}
|
|
|
|
|
|
ZXing_WriterOptions* ZXing_WriterOptions_new()
|
|
{
|
|
ZX_TRY(new ZXing_WriterOptions());
|
|
}
|
|
|
|
void ZXing_WriterOptions_delete(ZXing_WriterOptions* opts)
|
|
{
|
|
delete opts;
|
|
}
|
|
|
|
#define ZX_PROPERTY(TYPE, GETTER, SETTER) \
|
|
TYPE ZXing_WriterOptions_get##SETTER(const ZXing_WriterOptions* opts) { return opts->GETTER(); } \
|
|
void ZXing_WriterOptions_set##SETTER(ZXing_WriterOptions* opts, TYPE val) { opts->GETTER(val); }
|
|
|
|
ZX_PROPERTY(int, scale, Scale)
|
|
ZX_PROPERTY(int, sizeHint, SizeHint)
|
|
ZX_PROPERTY(int, rotate, Rotate)
|
|
ZX_PROPERTY(bool, withHRT, WithHRT)
|
|
ZX_PROPERTY(bool, withQuietZones, WithQuietZones)
|
|
|
|
#undef ZX_PROPERTY
|
|
|
|
ZXing_Barcode* ZXing_CreateBarcodeFromText(const char* data, int size, const ZXing_CreatorOptions* opts)
|
|
{
|
|
ZX_CHECK(data && opts, "Data and/or options param in CreateBarcodeFromText is NULL")
|
|
ZX_TRY(new Barcode(CreateBarcodeFromText({data, size ? static_cast<size_t>(size) : strlen(data)}, *opts));)
|
|
}
|
|
|
|
ZXing_Barcode* ZXing_CreateBarcodeFromBytes(const void* data, int size, const ZXing_CreatorOptions* opts)
|
|
{
|
|
ZX_CHECK(data && size && opts, "Data and/or options param in CreateBarcodeFromBytes is NULL")
|
|
ZX_TRY(new Barcode(CreateBarcodeFromBytes(data, size, *opts)))
|
|
}
|
|
|
|
char* ZXing_WriteBarcodeToSVG(const ZXing_Barcode* barcode, const ZXing_WriterOptions* opts)
|
|
{
|
|
ZX_CHECK(barcode, "Barcode param in WriteBarcodeToSVG is NULL")
|
|
ZX_TRY(copy(opts ? WriteBarcodeToSVG(*barcode, *opts) : WriteBarcodeToSVG(*barcode)))
|
|
}
|
|
|
|
ZXing_Image* ZXing_WriteBarcodeToImage(const ZXing_Barcode* barcode, const ZXing_WriterOptions* opts)
|
|
{
|
|
ZX_CHECK(barcode, "Barcode param in WriteBarcodeToSVG is NULL")
|
|
ZX_TRY(new Image(opts ? WriteBarcodeToImage(*barcode, *opts) : WriteBarcodeToImage(*barcode)))
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
* ZXingC.h
|
|
*/
|
|
|
|
char* ZXing_LastErrorMsg()
|
|
{
|
|
if (lastErrorMsg.empty())
|
|
return NULL;
|
|
|
|
return copy(std::exchange(lastErrorMsg, {}));
|
|
}
|
|
|
|
void ZXing_free(void* ptr)
|
|
{
|
|
free(ptr);
|
|
}
|
|
|
|
} // extern "C"
|