#include <zxing/qrcode/decoder/DecodedBitStreamParser.h>
#include <iostream>
+#ifndef NO_ICONV
#include <iconv.h>
+#endif
// Required for compatibility. TODO: test on Symbian
#ifdef ZXING_ICONV_CONST
const char *DecodedBitStreamParser::SHIFT_JIS = "SHIFT_JIS";
const char *DecodedBitStreamParser::EUC_JP = "EUC-JP";
-void DecodedBitStreamParser::append(ostream &ost, const unsigned char *bufIn, size_t nIn, const char *src) {
+string DecodedBitStreamParser::convert(const unsigned char *bufIn, size_t nIn, const char *src) {
+#ifndef NO_ICONV
if (nIn == 0) {
- return;
+ return string();
}
iconv_t cd = iconv_open(UTF8, src);
int nResult = maxOut - nTo;
bufOut[nResult] = '\0';
-
- ost << bufOut;
+ string result((const char *)bufOut);
delete[] bufOut;
+ return result;
+ #else
+ return string((const char *)bufIn, nIn);
+ #endif
}
-void DecodedBitStreamParser::decodeKanjiSegment(Ref<BitSource> bits, ostringstream &result, int count) {
+string DecodedBitStreamParser::decodeKanjiSegment(Ref<BitSource> bits, int count) {
// Each character will require 2 bytes. Read the characters as 2-byte pairs
// and decode as Shift_JIS afterwards
size_t nBytes = 2 * count;
count--;
}
- append(result, buffer, nBytes, SHIFT_JIS);
+ string result = convert(buffer, nBytes, SHIFT_JIS);
delete[] buffer;
+ return result;
}
-void DecodedBitStreamParser::decodeByteSegment(Ref<BitSource> bits, ostringstream &result, int count) {
+string DecodedBitStreamParser::decodeByteSegment(Ref<BitSource> bits, int count) {
int nBytes = count;
unsigned char* readBytes = new unsigned char[nBytes];
if (count << 3 > bits->available()) {
// Shift_JIS -- without anything like an ECI designator to
// give a hint.
const char *encoding = guessEncoding(readBytes, nBytes);
- append(result, readBytes, nBytes, encoding);
+ string result = convert(readBytes, nBytes, encoding);
delete[] readBytes;
+ return result;
}
-void DecodedBitStreamParser::decodeNumericSegment(Ref<BitSource> bits, ostringstream &result, int count) {
+string DecodedBitStreamParser::decodeNumericSegment(Ref<BitSource> bits, int count) {
int nBytes = count;
unsigned char* bytes = new unsigned char[nBytes];
int i = 0;
}
bytes[i++] = ALPHANUMERIC_CHARS[digitBits];
}
- append(result, bytes, nBytes, ASCII);
+ string result = convert(bytes, nBytes, ASCII);
delete[] bytes;
+ return result;
}
-void DecodedBitStreamParser::decodeAlphanumericSegment(Ref<BitSource> bits, ostringstream &result, int count) {
+string DecodedBitStreamParser::decodeAlphanumericSegment(Ref<BitSource> bits, int count) {
int nBytes = count;
unsigned char* bytes = new unsigned char[nBytes];
int i = 0;
if (count == 1) {
bytes[i++] = ALPHANUMERIC_CHARS[bits->readBits(6)];
}
- append(result, bytes, nBytes, ASCII);
+ string result = convert(bytes, nBytes, ASCII);
delete[] bytes;
+ return result;
}
const char *
}
string DecodedBitStreamParser::decode(ArrayRef<unsigned char> bytes, Version *version) {
- ostringstream result;
+ string result;
Ref<BitSource> bits(new BitSource(bytes));
Mode *mode = &Mode::TERMINATOR;
do {
// How many characters will follow, encoded in this mode?
int count = bits->readBits(mode->getCharacterCountBits(version));
if (mode == &Mode::NUMERIC) {
- decodeNumericSegment(bits, result, count);
+ result = decodeNumericSegment(bits, count);
} else if (mode == &Mode::ALPHANUMERIC) {
- decodeAlphanumericSegment(bits, result, count);
+ result = decodeAlphanumericSegment(bits, count);
} else if (mode == &Mode::BYTE) {
- decodeByteSegment(bits, result, count);
+ result = decodeByteSegment(bits, count);
} else if (mode == &Mode::KANJI) {
- decodeKanjiSegment(bits, result, count);
+ result = decodeKanjiSegment(bits, count);
} else {
throw ReaderException("Unsupported mode indicator");
}
}
} while (mode != &Mode::TERMINATOR);
- return result.str();
+ return result;
}
}