1
0
mirror of https://github.com/tumic0/GPXSee.git synced 2024-10-07 23:33:21 +02:00
GPXSee/src/map/IMG/lblfile.cpp

207 lines
5.2 KiB
C++
Raw Normal View History

2019-05-10 18:56:19 +02:00
#include <QTextCodec>
#include "lblfile.h"
enum Charset {Normal, Symbol, Special};
static quint8 NORMAL_CHARS[] = {
' ', 'A', 'B', 'C', 'D', 'E', 'F', 'G',
'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',
2019-06-30 20:39:22 +02:00
'X', 'Y', 'Z', '~', '~', '~', ' ', ' ',
2019-05-10 18:56:19 +02:00
'0', '1', '2', '3', '4', '5', '6', '7',
'8', '9', '~', '~', '~', '~', '~', '~'
};
static quint8 SYMBOL_CHARS[] = {
'@', '!', '"', '#', '$', '%', '&', '\'',
'(', ')', '*', '+', ',', '-', '.', '/',
'~', '~', '~', '~', '~', '~', '~', '~',
'~', '~', ':', ';', '<', '=', '>', '?',
'~', '~', '~', '~', '~', '~', '~', '~',
'~', '~', '~', '[', '\\', ']', '^', '_'
};
static quint8 SPECIAL_CHARS[] = {
'`', 'a', 'b', 'c', 'd', 'e', 'f', 'g',
'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',
'p', 'q', 'r', 's', 't', 'u', 'v', 'w',
'x', 'y', 'z', '~', '~', '~', '~', '~',
'0', '1', '2', '3', '4', '5', '6', '7',
'8', '9', '~', '~', '~', '~', '~', '~'
};
2020-02-16 12:57:40 +01:00
static bool isAllUpperCase(const QString &str)
2019-05-10 18:56:19 +02:00
{
if (str.isEmpty())
2020-02-16 12:57:40 +01:00
return false;
2019-05-10 18:56:19 +02:00
for (int i = 0; i < str.size(); i++)
if (str.at(i).isLetter() && !(str.at(i).isUpper()
|| str.at(i) == QChar(0x00DF)))
2020-02-16 12:57:40 +01:00
return false;
return true;
}
2019-05-10 18:56:19 +02:00
2020-02-16 12:57:40 +01:00
static QString capitalized(const QString &str)
{
2019-05-10 18:56:19 +02:00
QString ret(str);
for (int i = 0; i < str.size(); i++)
if (i && !str.at(i-1).isSpace())
ret[i] = str.at(i).toLower();
else
ret[i] = str.at(i);
return ret;
}
2019-06-30 20:39:22 +02:00
2020-01-11 23:41:04 +01:00
bool LBLFile::init(Handle &hdl)
2019-05-10 18:56:19 +02:00
{
quint16 codepage;
2019-08-12 22:20:12 +02:00
quint8 multiplier, poiMultiplier;
2019-05-10 18:56:19 +02:00
if (!(seek(hdl, _gmpOffset + 0x15) && readUInt32(hdl, _offset)
&& readUInt32(hdl, _size) && readUInt8(hdl, multiplier)
&& readUInt8(hdl, _encoding) && seek(hdl, _gmpOffset + 0x57)
2019-05-10 18:56:19 +02:00
&& readUInt32(hdl, _poiOffset) && readUInt32(hdl, _poiSize)
&& readUInt8(hdl, poiMultiplier) && seek(hdl, _gmpOffset + 0xAA)
2019-08-12 22:20:12 +02:00
&& readUInt16(hdl, codepage)))
2019-05-10 18:56:19 +02:00
return false;
2019-08-12 22:20:12 +02:00
_multiplier = 1<<multiplier;
_poiMultiplier = 1<<poiMultiplier;
2019-05-10 18:56:19 +02:00
if (codepage == 65001)
_codec = QTextCodec::codecForName("UTF-8");
else if (codepage == 0)
_codec = 0;
else
_codec = QTextCodec::codecForName(QString("CP%1").arg(codepage)
.toLatin1());
return true;
}
2020-02-16 12:57:40 +01:00
Label LBLFile::label6b(Handle &hdl, quint32 offset, bool capitalize) const
2019-05-10 18:56:19 +02:00
{
2019-06-30 20:39:22 +02:00
Label::Shield::Type shieldType = Label::Shield::None;
QByteArray label, shieldLabel;
QByteArray *bap = &label;
2019-05-10 18:56:19 +02:00
Charset curCharSet = Normal;
quint8 b1, b2, b3;
if (!seek(hdl, offset))
2019-06-30 20:39:22 +02:00
return Label();
2019-05-10 18:56:19 +02:00
while (true) {
if (!(readUInt8(hdl, b1) && readUInt8(hdl, b2) && readUInt8(hdl, b3)))
2019-06-30 20:39:22 +02:00
return Label();
2019-05-10 18:56:19 +02:00
int c[]= {b1>>2, (b1&0x3)<<4|b2>>4, (b2&0xF)<<2|b3>>6, b3&0x3F};
for (int cpt = 0; cpt < 4; cpt++) {
2020-02-16 12:57:40 +01:00
if (c[cpt] > 0x2f || (curCharSet == Normal && c[cpt] == 0x1d)) {
QString text(QString::fromLatin1(label));
2020-02-16 13:59:19 +01:00
return Label(capitalize && isAllUpperCase(text)
2020-02-16 12:57:40 +01:00
? capitalized(text) : text, Label::Shield(shieldType,
shieldLabel));
}
2019-05-10 18:56:19 +02:00
switch (curCharSet) {
case Normal:
if (c[cpt] == 0x1c)
curCharSet = Symbol;
else if (c[cpt] == 0x1b)
curCharSet = Special;
2019-06-30 20:39:22 +02:00
else if (c[cpt] >= 0x2a && c[cpt] <= 0x2f) {
2019-07-10 21:37:03 +02:00
shieldType = static_cast<Label::Shield::Type>
(c[cpt] - 0x29);
2019-06-30 20:39:22 +02:00
bap = &shieldLabel;
} else if (bap == &shieldLabel
&& NORMAL_CHARS[c[cpt]] == ' ')
bap = &label;
2019-05-10 18:56:19 +02:00
else
2019-06-30 20:39:22 +02:00
bap->append(NORMAL_CHARS[c[cpt]]);
2019-05-10 18:56:19 +02:00
break;
case Symbol:
2019-06-30 20:39:22 +02:00
bap->append(SYMBOL_CHARS[c[cpt]]);
2019-05-10 18:56:19 +02:00
curCharSet = Normal;
break;
case Special:
2019-06-30 20:39:22 +02:00
bap->append(SPECIAL_CHARS[c[cpt]]);
2019-05-10 18:56:19 +02:00
curCharSet = Normal;
break;
}
}
}
}
2020-02-16 12:57:40 +01:00
Label LBLFile::label8b(Handle &hdl, quint32 offset, bool capitalize) const
2019-05-10 18:56:19 +02:00
{
2019-06-30 20:39:22 +02:00
Label::Shield::Type shieldType = Label::Shield::None;
QByteArray label, shieldLabel;
QByteArray *bap = &label;
2019-05-10 18:56:19 +02:00
quint8 c;
if (!seek(hdl, offset))
2019-06-30 20:39:22 +02:00
return Label();
2019-05-10 18:56:19 +02:00
while (true) {
if (!readUInt8(hdl, c))
2019-06-30 20:39:22 +02:00
return Label();
if (!c || c == 0x1d)
2019-05-10 18:56:19 +02:00
break;
2020-04-18 00:00:48 +02:00
if (c == 0x1c)
capitalize = false;
else if ((c >= 0x1e && c <= 0x1f)) {
2019-06-30 23:18:30 +02:00
if (bap == &shieldLabel)
bap = &label;
else
bap->append(' ');
} else if (c <= 0x07) {
2019-07-10 21:37:03 +02:00
shieldType = static_cast<Label::Shield::Type>(c);
2019-06-30 20:39:22 +02:00
bap = &shieldLabel;
} else if (bap == &shieldLabel && QChar(c).isSpace()) {
bap = &label;
} else
bap->append(c);
2019-05-10 18:56:19 +02:00
}
2020-02-16 12:57:40 +01:00
QString text(_codec ? _codec->toUnicode(label) : QString::fromLatin1(label));
QString shieldText(_codec ? _codec->toUnicode(shieldLabel)
: QString::fromLatin1(shieldLabel));
return Label(capitalize && isAllUpperCase(text) ? capitalized(text) : text,
Label::Shield(shieldType, shieldText));
2019-05-10 18:56:19 +02:00
}
2020-02-16 12:57:40 +01:00
Label LBLFile::label(Handle &hdl, quint32 offset, bool poi, bool capitalize)
2019-05-10 18:56:19 +02:00
{
2020-01-11 23:41:04 +01:00
if (!_multiplier && !init(hdl))
return QString();
2019-05-10 18:56:19 +02:00
quint32 labelOffset;
if (poi) {
quint32 poiOffset;
2019-08-12 22:20:12 +02:00
if (!(_poiSize >= offset * _poiMultiplier
&& seek(hdl, _poiOffset + offset * _poiMultiplier)
&& readUInt24(hdl, poiOffset) && (poiOffset & 0x3FFFFF)))
2019-05-10 18:56:19 +02:00
return QString();
labelOffset = _offset + (poiOffset & 0x3FFFFF) * _multiplier;
} else
labelOffset = _offset + offset * _multiplier;
if (labelOffset > _offset + _size)
return QString();
switch (_encoding) {
case 6:
2020-02-16 12:57:40 +01:00
return label6b(hdl, labelOffset, capitalize);
2019-05-10 18:56:19 +02:00
case 9:
case 10:
2020-02-16 12:57:40 +01:00
return label8b(hdl, labelOffset, capitalize);
2019-05-10 18:56:19 +02:00
default:
2019-06-30 20:39:22 +02:00
return Label();
2019-05-10 18:56:19 +02:00
}
}