1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
|
#ifndef PARSER_H
#define PARSER_H
#include <qxml.h>
#include <qmap.h>
#include <qregexp.h>
struct BlockInfo {
int start_line;
int start_col;
int end_line;
int end_col;
// used to detect sub-messages
int offset;
BlockInfo() {
start_line = 0;
start_col = 0;
end_line = 0;
end_col = 0;
// used to detect sub-messages
offset = 0;
}
};
class MsgBlock {
public:
MsgBlock() { start = end = 0; do_not_split = false; }
MsgBlock(const MsgBlock &rhs ) {
*this = rhs;
}
QValueList<BlockInfo> lines;
QString tag;
QString comment;
QString msgid;
QString msgid_plural;
QString msgstr;
QStringList msgstr_plurals;
int start, end;
bool do_not_split;
void operator=(const MsgBlock& rhs) {
lines = rhs.lines;
tag = rhs.tag;
comment = rhs.comment;
msgid = rhs.msgid;
msgid_plural = rhs.msgid_plural;
msgstr = rhs.msgstr;
msgstr_plurals = rhs.msgstr_plurals;
start = rhs.start;
end = rhs.end;
do_not_split = rhs.do_not_split;
}
};
class ParaCounter
{
public:
ParaCounter() { current = 0; }
void addAnchor(QString anchor) { anchors.insert(anchor, current); }
void increasePara() { current++; }
QMap<QString, int> anchors;
int current;
};
class MsgList : public QValueList<MsgBlock>
{
public:
MsgList() {}
ParaCounter pc;
};
class StructureParser : public QXmlDefaultHandler
{
public:
bool startDocument();
bool startElement( const QString&, const QString&, const QString& ,
const QXmlAttributes& );
bool endElement( const QString&, const QString&, const QString& );
bool characters( const QString &ch);
static bool isCuttingTag(const QString &tag);
static bool isSingleTag(const QString &qName);
static bool isLiteralTag(const QString &qName);
void setDocumentLocator ( QXmlLocator * l ) { locator = l; }
bool skippedEntity ( const QString & name );
bool fatalError ( const QXmlParseException & );
bool comment ( const QString & );
bool error(const QXmlParseException &e ) { return fatalError(e); }
bool warning(const QXmlParseException &e ) { return fatalError(e); }
MsgList getList() const { return list; }
MsgList splitMessage(const MsgBlock &message);
virtual bool startCDATA();
virtual bool endCDATA();
static bool closureTag(const QString& message, const QString &tag);
static bool isClosure(const QString &message);
static void descape(QString &message);
static QString escapeLiterals( const QString &contents);
static QString descapeLiterals( const QString &contents);
static void cleanupTags( QString &contents );
static void removeEmptyTags( QString &contents);
static void stripWhiteSpace( QString &contents);
private:
bool formatMessage(MsgBlock &message) const;
QXmlLocator *locator;
QString message;
int inside, startline, startcol;
int line;
MsgList list;
mutable QRegExp infos_reg;
mutable QRegExp do_not_split_reg;
};
void outputMsg(const char *prefix, const QString &message);
MsgList parseXML(const char *filename);
QString escapePO(QString msgid);
#endif
|