Created initial XmlLabelCreator.
This commit is contained in:
@@ -0,0 +1,405 @@
|
||||
/* XmlLabelParser.cpp
|
||||
*
|
||||
* Copyright (C) 2014 Jim Evins <evins@snaught.com>
|
||||
*
|
||||
* This file is part of gLabels-qt.
|
||||
*
|
||||
* gLabels-qt is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* gLabels-qt is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with gLabels-qt. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "XmlLabelParser.h"
|
||||
|
||||
#include "LabelModel.h"
|
||||
#include "LabelModelObject.h"
|
||||
#include "LabelModelBoxObject.h"
|
||||
//#include "LabelObjectEllipse.h"
|
||||
//#include "LabelObjectLine.h"
|
||||
//#include "LabelObjectImage.h"
|
||||
//#include "LabelObjectBarcode.h"
|
||||
#include "libglabels/XmlTemplateParser.h"
|
||||
#include "libglabels/XmlUtil.h"
|
||||
|
||||
#include <QFile>
|
||||
#include <QByteArray>
|
||||
#include <zlib.h>
|
||||
#include <QtDebug>
|
||||
|
||||
|
||||
glabels::LabelModel*
|
||||
glabels::XmlLabelParser::readFile( const QString& fileName )
|
||||
{
|
||||
QFile file( fileName );
|
||||
|
||||
if ( !file.open( QFile::ReadOnly ) )
|
||||
{
|
||||
qWarning() << "Error: Cannot read file " << qPrintable(fileName)
|
||||
<< ": " << file.errorString();
|
||||
return 0;
|
||||
}
|
||||
|
||||
QDomDocument doc;
|
||||
bool success;
|
||||
QString errorString;
|
||||
int errorLine;
|
||||
int errorColumn;
|
||||
|
||||
QByteArray rawData = file.readAll();
|
||||
if ( ((rawData[0]&0xFF) == 0x1F) && ((rawData[1]&0xFF) == 0x8b) ) // gzip magic number 0x1F, 0x8B
|
||||
{
|
||||
// gzip compressed format
|
||||
QByteArray unzippedData;
|
||||
gunzip( rawData, unzippedData );
|
||||
success = doc.setContent( unzippedData, false, &errorString, &errorLine, &errorColumn );
|
||||
}
|
||||
else
|
||||
{
|
||||
// plain text
|
||||
success = doc.setContent( rawData, false, &errorString, &errorLine, &errorColumn );
|
||||
}
|
||||
|
||||
if ( !success )
|
||||
{
|
||||
qWarning() << "Error: Parse error at line " << errorLine
|
||||
<< "column " << errorColumn
|
||||
<< ": " << errorString;
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
QDomElement root = doc.documentElement();
|
||||
if ( root.tagName() != "Glabels-document" )
|
||||
{
|
||||
qWarning() << "Error: Not a Glabels-document file";
|
||||
return 0;
|
||||
}
|
||||
|
||||
return parseRootNode( root );
|
||||
}
|
||||
|
||||
|
||||
glabels::LabelModel*
|
||||
glabels::XmlLabelParser::readBuffer( const QString& buffer )
|
||||
{
|
||||
QDomDocument doc;
|
||||
QString errorString;
|
||||
int errorLine;
|
||||
int errorColumn;
|
||||
|
||||
if ( !doc.setContent( buffer, false, &errorString, &errorLine, &errorColumn ) )
|
||||
{
|
||||
qWarning() << "Error: Parse error at line " << errorLine
|
||||
<< "column " << errorColumn
|
||||
<< ": " << errorString;
|
||||
return 0;
|
||||
}
|
||||
|
||||
QDomElement root = doc.documentElement();
|
||||
if ( root.tagName() != "Glabels-document" )
|
||||
{
|
||||
qWarning() << "Error: Not a Glabels-document file";
|
||||
return 0;
|
||||
}
|
||||
|
||||
return parseRootNode( root );
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::gunzip( const QByteArray& data, QByteArray& result )
|
||||
{
|
||||
result.clear();
|
||||
|
||||
if (data.size() <= 4) {
|
||||
qWarning("XmlLabelParser::gunzip: Input data is truncated");
|
||||
return;
|
||||
}
|
||||
|
||||
// setup stream for inflate()
|
||||
z_stream strm;
|
||||
strm.zalloc = Z_NULL;
|
||||
strm.zfree = Z_NULL;
|
||||
strm.opaque = Z_NULL;
|
||||
strm.avail_in = data.size();
|
||||
strm.next_in = (Bytef*)(data.data());
|
||||
|
||||
int ret = inflateInit2(&strm, MAX_WBITS + 16); // gzip decoding
|
||||
if (ret != Z_OK)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
static const int CHUNK_SIZE = 1024;
|
||||
char out[CHUNK_SIZE];
|
||||
|
||||
// run inflate(), one chunk at a time
|
||||
do {
|
||||
strm.avail_out = CHUNK_SIZE;
|
||||
strm.next_out = (Bytef*)(out);
|
||||
|
||||
ret = inflate(&strm, Z_NO_FLUSH);
|
||||
Q_ASSERT(ret != Z_STREAM_ERROR); // state not clobbered
|
||||
|
||||
if ( (ret == Z_NEED_DICT) || (ret == Z_DATA_ERROR) || (ret == Z_MEM_ERROR) )
|
||||
{
|
||||
// clean up
|
||||
inflateEnd(&strm);
|
||||
return;
|
||||
}
|
||||
|
||||
result.append(out, CHUNK_SIZE - strm.avail_out);
|
||||
} while (strm.avail_out == 0);
|
||||
|
||||
// clean up
|
||||
inflateEnd(&strm);
|
||||
}
|
||||
|
||||
|
||||
glabels::LabelModel*
|
||||
glabels::XmlLabelParser::parseRootNode( const QDomElement &node )
|
||||
{
|
||||
using namespace libglabels;
|
||||
|
||||
LabelModel* label = new LabelModel();
|
||||
|
||||
/* Pass 1, extract data nodes to pre-load cache. */
|
||||
for ( QDomNode child = node.firstChild(); !child.isNull(); child = child.nextSibling() )
|
||||
{
|
||||
if ( child.toElement().tagName() == "Data" )
|
||||
{
|
||||
parseDataNode( child.toElement(), label );
|
||||
}
|
||||
}
|
||||
|
||||
/* Pass 2, now extract everything else. */
|
||||
for ( QDomNode child = node.firstChild(); !child.isNull(); child = child.nextSibling() )
|
||||
{
|
||||
QString tagName = child.toElement().tagName();
|
||||
|
||||
if ( tagName == "Template" )
|
||||
{
|
||||
Template* tmplate = XmlTemplateParser().parseTemplateNode( child.toElement() );
|
||||
if ( tmplate == 0 )
|
||||
{
|
||||
qWarning() << "Unable to parse template";
|
||||
return 0;
|
||||
}
|
||||
label->setTmplate( tmplate );
|
||||
}
|
||||
else if ( tagName == "Objects" )
|
||||
{
|
||||
parseObjectsNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Merge" )
|
||||
{
|
||||
parseMergeNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Data" )
|
||||
{
|
||||
/* Handled in pass 1. */
|
||||
}
|
||||
else if ( !child.isComment() )
|
||||
{
|
||||
qWarning() << "Unexpected" << node.tagName() << "child:" << tagName;
|
||||
}
|
||||
}
|
||||
|
||||
label->clearModified();
|
||||
return label;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectsNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
for ( QDomNode child = node.firstChild(); !child.isNull(); child = child.nextSibling() )
|
||||
{
|
||||
QString tagName = child.toElement().tagName();
|
||||
|
||||
if ( tagName == "Object-box" )
|
||||
{
|
||||
parseObjectBoxNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Object-ellipse" )
|
||||
{
|
||||
parseObjectEllipseNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Object-line" )
|
||||
{
|
||||
parseObjectLineNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Object-image" )
|
||||
{
|
||||
parseObjectImageNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Object-barcode" )
|
||||
{
|
||||
parseObjectBarcodeNode( child.toElement(), label );
|
||||
}
|
||||
else if ( tagName == "Object-text" )
|
||||
{
|
||||
parseObjectTextNode( child.toElement(), label );
|
||||
}
|
||||
else if ( !child.isComment() )
|
||||
{
|
||||
qWarning() << "Unexpected" << node.tagName() << "child:" << tagName;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectBoxNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
using namespace libglabels;
|
||||
|
||||
LabelModelBoxObject* object = new LabelModelBoxObject();
|
||||
label->addObject( object );
|
||||
|
||||
|
||||
/* position attrs */
|
||||
object->setX0( XmlUtil::getLengthAttr( node, "x", 0.0 ) );
|
||||
object->setY0( XmlUtil::getLengthAttr( node, "y", 0.0 ) );
|
||||
|
||||
/* size attrs */
|
||||
object->setW( XmlUtil::getLengthAttr( node, "w", 0 ) );
|
||||
object->setH( XmlUtil::getLengthAttr( node, "h", 0 ) );
|
||||
|
||||
/* line attrs */
|
||||
object->setLineWidth( XmlUtil::getLengthAttr( node, "line_width", 1.0 ) );
|
||||
{
|
||||
QString key = XmlUtil::getStringAttr( node, "line_color_field", "" );
|
||||
bool field_flag = !key.isEmpty();
|
||||
uint32_t color = XmlUtil::getUIntAttr( node, "line_color", 0 );
|
||||
|
||||
object->setLineColorNode( ColorNode( field_flag, color, key ) );
|
||||
}
|
||||
|
||||
/* fill attrs */
|
||||
{
|
||||
QString key = XmlUtil::getStringAttr( node, "line_color_field", "" );
|
||||
bool field_flag = !key.isEmpty();
|
||||
uint32_t color = XmlUtil::getUIntAttr( node, "fill_color", 0 );
|
||||
|
||||
object->setFillColorNode( ColorNode( field_flag, color, key ) );
|
||||
}
|
||||
|
||||
/* affine attrs */
|
||||
parseAffineAttrs( node, object );
|
||||
|
||||
/* shadow attrs */
|
||||
parseShadowAttrs( node, object );
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectEllipseNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectLineNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectImageNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectBarcodeNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseObjectTextNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseTopLevelSpanNode( const QDomElement &node, LabelModelTextObject* object )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseAffineAttrs( const QDomElement &node, LabelModelObject* object )
|
||||
{
|
||||
using namespace libglabels;
|
||||
|
||||
double a[6];
|
||||
|
||||
a[0] = XmlUtil::getDoubleAttr( node, "a0", 0.0 );
|
||||
a[1] = XmlUtil::getDoubleAttr( node, "a1", 0.0 );
|
||||
a[2] = XmlUtil::getDoubleAttr( node, "a2", 0.0 );
|
||||
a[3] = XmlUtil::getDoubleAttr( node, "a3", 0.0 );
|
||||
a[4] = XmlUtil::getDoubleAttr( node, "a4", 0.0 );
|
||||
a[5] = XmlUtil::getDoubleAttr( node, "a5", 0.0 );
|
||||
|
||||
object->setMatrix( QTransform( a[0], a[1], a[2], a[3], a[4], a[5] ) );
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseShadowAttrs( const QDomElement &node, LabelModelObject* object )
|
||||
{
|
||||
using namespace libglabels;
|
||||
|
||||
object->setShadow( XmlUtil::getBoolAttr( node, "shadow", false ) );
|
||||
|
||||
if ( object->shadow() )
|
||||
{
|
||||
object->setShadowX( XmlUtil::getLengthAttr( node, "shadow_x", 0.0 ) );
|
||||
object->setShadowY( XmlUtil::getLengthAttr( node, "shadow_y", 0.0 ) );
|
||||
|
||||
QString key = XmlUtil::getStringAttr( node, "shadow_color_field", "" );
|
||||
bool field_flag = !key.isEmpty();
|
||||
uint32_t color = XmlUtil::getUIntAttr( node, "shadow_color", 0 );
|
||||
|
||||
object->setShadowColorNode( ColorNode( field_flag, color, key ) );
|
||||
|
||||
object->setShadowOpacity( XmlUtil::getLengthAttr( node, "shadow_y", 1.0 ) );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseMergeNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseDataNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parsePixdataNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
glabels::XmlLabelParser::parseFileNode( const QDomElement &node, LabelModel* label )
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user