QGIS/external/libdxfrw/libdxfrw.cpp
2019-02-19 19:56:40 +01:00

3428 lines
89 KiB
C++

/******************************************************************************
** libDXFrw - Library to read/write DXF files (ascii & binary) **
** **
** Copyright (C) 2011-2015 José F. Soriano, rallazz@gmail.com **
** **
** This library is free software, licensed under the terms of the GNU **
** General Public License as published by the Free Software Foundation, **
** either version 2 of the License, or (at your option) any later version. **
** You should have received a copy of the GNU General Public License **
** along with this program. If not, see <http://www.gnu.org/licenses/>. **
******************************************************************************/
// uncomment to get detailed debug output on DWG read. Caution: this option makes DWG import super-slow!
// #define DWGDEBUG 1
#include "libdxfrw.h"
#include <fstream>
#include <algorithm>
#include <sstream>
#include <cassert>
#include "intern/drw_textcodec.h"
#include "intern/dxfreader.h"
#include "intern/dxfwriter.h"
#include "intern/drw_dbg.h"
#include "qgslogger.h"
#ifndef DWGDEBUG
#undef QgsDebugCall
#undef QgsDebugMsg
#undef QgsDebugMsgLevel
#define QgsDebugCall
#define QgsDebugMsg(str)
#define QgsDebugMsgLevel(str, level)
#endif
#if __cplusplus >= 201500
#define FALLTHROUGH [[fallthrough]];
#elif defined(__clang__)
#define FALLTHROUGH [[clang::fallthrough]];
#elif defined(__GNUC__) && __GNUC__ >= 7
#define FALLTHROUGH [[gnu::fallthrough]];
#else
#define FALLTHROUGH
#endif
#define FIRSTHANDLE 48
#if 0
enum sections
{
secUnknown,
secHeader,
secTables,
secBlocks,
secEntities,
secObjects
};
#endif
dxfRW::dxfRW( const char *name )
: fileName( name )
, binFile( false )
, reader( nullptr )
, writer( nullptr )
, iface( nullptr )
, entCount( 0 )
, wlayer0( false )
, dimstyleStd( false )
, applyExt( false )
, writingBlock( false )
, elParts( 128 ) //parts number when convert ellipse to polyline
, currHandle( 0 )
{
DRW_DBGSL( DRW_dbg::none );
}
dxfRW::~dxfRW()
{
delete reader;
delete writer;
for ( std::vector<DRW_ImageDef *>::iterator it = imageDef.begin(); it != imageDef.end(); ++it )
delete *it;
imageDef.clear();
}
void dxfRW::setDebug( DRW::DBG_LEVEL lvl )
{
switch ( lvl )
{
case DRW::debug:
DRW_DBGSL( DRW_dbg::debug );
break;
default:
DRW_DBGSL( DRW_dbg::none );
}
}
bool dxfRW::read( DRW_Interface *interface_, bool ext )
{
drw_assert( fileName.empty() == false );
bool isOk = false;
applyExt = ext;
std::ifstream filestr;
if ( !interface_ )
return isOk;
filestr.open( fileName.c_str(), std::ios_base::in | std::ios::binary );
if ( !filestr.is_open() )
return isOk;
if ( !filestr.good() )
return isOk;
char line[22];
char line2[22] = "AutoCAD Binary DXF\r\n";
line2[20] = ( char )26;
line2[21] = '\0';
filestr.read( line, 22 );
filestr.close();
iface = interface_;
if ( strcmp( line, line2 ) == 0 )
{
filestr.open( fileName.c_str(), std::ios_base::in | std::ios::binary );
binFile = true;
//skip sentinel
filestr.seekg( 22, std::ios::beg );
reader = new dxfReaderBinary( &filestr );
QgsDebugMsg( "binary file" );
}
else
{
binFile = false;
filestr.open( fileName.c_str(), std::ios_base::in );
reader = new dxfReaderAscii( &filestr );
}
isOk = processDxf();
filestr.close();
delete reader;
reader = nullptr;
return isOk;
}
bool dxfRW::write( DRW_Interface *interface_, DRW::Version ver, bool bin )
{
bool isOk = false;
std::ofstream filestr;
version = ver;
binFile = bin;
iface = interface_;
if ( binFile )
{
filestr.open( fileName.c_str(), std::ios_base::out | std::ios::binary | std::ios::trunc );
//write sentinel
filestr << "AutoCAD Binary DXF\r\n" << ( char )26 << '\0';
writer = new dxfWriterBinary( &filestr );
QgsDebugMsg( "binary file" );
}
else
{
filestr.open( fileName.c_str(), std::ios_base::out | std::ios::trunc );
writer = new dxfWriterAscii( &filestr );
std::string comm = std::string( "dxfrw " ) + std::string( DRW_VERSION );
writer->writeString( 999, comm );
}
DRW_Header header;
iface->writeHeader( header );
writer->writeString( 0, "SECTION" );
entCount = FIRSTHANDLE;
header.write( writer, version );
writer->writeString( 0, "ENDSEC" );
if ( ver > DRW::AC1009 )
{
writer->writeString( 0, "SECTION" );
writer->writeString( 2, "CLASSES" );
writer->writeString( 0, "ENDSEC" );
}
writer->writeString( 0, "SECTION" );
writer->writeString( 2, "TABLES" );
writeTables();
writer->writeString( 0, "ENDSEC" );
writer->writeString( 0, "SECTION" );
writer->writeString( 2, "BLOCKS" );
writeBlocks();
writer->writeString( 0, "ENDSEC" );
writer->writeString( 0, "SECTION" );
writer->writeString( 2, "ENTITIES" );
iface->writeEntities();
writer->writeString( 0, "ENDSEC" );
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "SECTION" );
writer->writeString( 2, "OBJECTS" );
writeObjects();
writer->writeString( 0, "ENDSEC" );
}
writer->writeString( 0, "EOF" );
filestr.flush();
filestr.close();
isOk = true;
delete writer;
writer = nullptr;
return isOk;
}
bool dxfRW::writeEntity( DRW_Entity *ent )
{
ent->handle = ++entCount;
writer->writeString( 5, toHexStr( ent->handle ) );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbEntity" );
}
if ( ent->space == 1 )
writer->writeInt16( 67, 1 );
if ( version > DRW::AC1009 )
{
writer->writeUtf8String( 8, ent->layer );
writer->writeUtf8String( 6, ent->lineType );
}
else
{
writer->writeUtf8Caps( 8, ent->layer );
writer->writeUtf8Caps( 6, ent->lineType );
}
writer->writeInt16( 62, ent->color );
if ( version > DRW::AC1015 && ent->color24 >= 0 )
{
writer->writeInt32( 420, ent->color24 );
}
if ( version > DRW::AC1014 )
{
writer->writeInt16( 370, DRW_LW_Conv::lineWidth2dxfInt( ent->lWeight ) );
}
return true;
}
bool dxfRW::writeLineType( DRW_LType *ent )
{
std::string strname = ent->name;
transform( strname.begin(), strname.end(), strname.begin(), ::toupper );
//do not write linetypes handled by library
if ( strname == "BYLAYER" || strname == "BYBLOCK" || strname == "CONTINUOUS" )
{
return true;
}
writer->writeString( 0, "LTYPE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( ++entCount ) );
if ( version > DRW::AC1012 )
{
writer->writeString( 330, "5" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbLinetypeTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
writer->writeUtf8Caps( 2, ent->name );
writer->writeInt16( 70, ent->flags );
writer->writeUtf8String( 3, ent->desc );
ent->update();
writer->writeInt16( 72, 65 );
writer->writeInt16( 73, static_cast<int>( ent->size ) );
writer->writeDouble( 40, ent->length );
for ( unsigned int i = 0; i < ent->path.size(); i++ )
{
writer->writeDouble( 49, ent->path.at( i ) );
if ( version > DRW::AC1009 )
{
writer->writeInt16( 74, 0 );
}
}
return true;
}
bool dxfRW::writeLayer( DRW_Layer *ent )
{
writer->writeString( 0, "LAYER" );
if ( !wlayer0 && ent->name == "0" )
{
wlayer0 = true;
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "10" );
}
}
else
{
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( ++entCount ) );
}
}
if ( version > DRW::AC1012 )
{
writer->writeString( 330, "2" );
}
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbLayerTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
{
writer->writeUtf8Caps( 2, ent->name );
}
writer->writeInt16( 70, ent->flags );
writer->writeInt16( 62, ent->color );
if ( version > DRW::AC1015 && ent->color24 >= 0 )
{
writer->writeInt32( 420, ent->color24 );
}
if ( version > DRW::AC1009 )
{
writer->writeUtf8String( 6, ent->lineType );
if ( ! ent->plotF )
writer->writeBool( 290, ent->plotF );
writer->writeInt16( 370, DRW_LW_Conv::lineWidth2dxfInt( ent->lWeight ) );
writer->writeString( 390, "F" );
}
else
writer->writeUtf8Caps( 6, ent->lineType );
if ( !ent->extData.empty() )
{
writeExtData( ent->extData );
}
// writer->writeString(347, "10012");
return true;
}
bool dxfRW::writeTextstyle( DRW_Textstyle *ent )
{
writer->writeString( 0, "STYLE" );
if ( !dimstyleStd )
{
//stringstream cause crash in OS/X, bug#3597944
std::string name = ent->name;
transform( name.begin(), name.end(), name.begin(), toupper );
if ( name == "STANDARD" )
dimstyleStd = true;
}
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( ++entCount ) );
}
if ( version > DRW::AC1012 )
{
writer->writeString( 330, "2" );
}
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbTextStyleTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
{
writer->writeUtf8Caps( 2, ent->name );
}
writer->writeInt16( 70, ent->flags );
writer->writeDouble( 40, ent->height );
writer->writeDouble( 41, ent->width );
writer->writeDouble( 50, ent->oblique );
writer->writeInt16( 71, ent->genFlag );
writer->writeDouble( 42, ent->lastHeight );
if ( version > DRW::AC1009 )
{
writer->writeUtf8String( 3, ent->font );
writer->writeUtf8String( 4, ent->bigFont );
if ( ent->fontFamily != 0 )
writer->writeInt32( 1071, ent->fontFamily );
}
else
{
writer->writeUtf8Caps( 3, ent->font );
writer->writeUtf8Caps( 4, ent->bigFont );
}
return true;
}
bool dxfRW::writeVport( DRW_Vport *ent )
{
if ( !dimstyleStd )
{
ent->name = "*ACTIVE";
dimstyleStd = true;
}
writer->writeString( 0, "VPORT" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( ++entCount ) );
if ( version > DRW::AC1012 )
writer->writeString( 330, "2" );
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbViewportTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
writer->writeUtf8Caps( 2, ent->name );
writer->writeInt16( 70, ent->flags );
writer->writeDouble( 10, ent->lowerLeft.x );
writer->writeDouble( 20, ent->lowerLeft.y );
writer->writeDouble( 11, ent->UpperRight.x );
writer->writeDouble( 21, ent->UpperRight.y );
writer->writeDouble( 12, ent->center.x );
writer->writeDouble( 22, ent->center.y );
writer->writeDouble( 13, ent->snapBase.x );
writer->writeDouble( 23, ent->snapBase.y );
writer->writeDouble( 14, ent->snapSpacing.x );
writer->writeDouble( 24, ent->snapSpacing.y );
writer->writeDouble( 15, ent->gridSpacing.x );
writer->writeDouble( 25, ent->gridSpacing.y );
writer->writeDouble( 16, ent->viewDir.x );
writer->writeDouble( 26, ent->viewDir.y );
writer->writeDouble( 36, ent->viewDir.z );
writer->writeDouble( 17, ent->viewTarget.z );
writer->writeDouble( 27, ent->viewTarget.z );
writer->writeDouble( 37, ent->viewTarget.z );
writer->writeDouble( 40, ent->height );
writer->writeDouble( 41, ent->ratio );
writer->writeDouble( 42, ent->lensHeight );
writer->writeDouble( 43, ent->frontClip );
writer->writeDouble( 44, ent->backClip );
writer->writeDouble( 50, ent->snapAngle );
writer->writeDouble( 51, ent->twistAngle );
writer->writeInt16( 71, ent->viewMode );
writer->writeInt16( 72, ent->circleZoom );
writer->writeInt16( 73, ent->fastZoom );
writer->writeInt16( 74, ent->ucsIcon );
writer->writeInt16( 75, ent->snap );
writer->writeInt16( 76, ent->grid );
writer->writeInt16( 77, ent->snapStyle );
writer->writeInt16( 78, ent->snapIsopair );
if ( version > DRW::AC1014 )
{
writer->writeInt16( 281, 0 );
writer->writeInt16( 65, 1 );
writer->writeDouble( 110, 0.0 );
writer->writeDouble( 120, 0.0 );
writer->writeDouble( 130, 0.0 );
writer->writeDouble( 111, 1.0 );
writer->writeDouble( 121, 0.0 );
writer->writeDouble( 131, 0.0 );
writer->writeDouble( 112, 0.0 );
writer->writeDouble( 122, 1.0 );
writer->writeDouble( 132, 0.0 );
writer->writeInt16( 79, 0 );
writer->writeDouble( 146, 0.0 );
if ( version > DRW::AC1018 )
{
writer->writeString( 348, "10020" );
writer->writeInt16( 60, ent->gridBehavior ); // v2007 undocumented see DRW_Vport class
writer->writeInt16( 61, 5 );
writer->writeBool( 292, 1 );
writer->writeInt16( 282, 1 );
writer->writeDouble( 141, 0.0 );
writer->writeDouble( 142, 0.0 );
writer->writeInt16( 63, 250 );
writer->writeInt32( 421, 3358443 );
}
}
return true;
}
bool dxfRW::writeDimstyle( DRW_Dimstyle *ent )
{
writer->writeString( 0, "DIMSTYLE" );
if ( !dimstyleStd )
{
std::string name = ent->name;
std::transform( name.begin(), name.end(), name.begin(), ::toupper );
if ( name == "STANDARD" )
dimstyleStd = true;
}
if ( version > DRW::AC1009 )
{
writer->writeString( 105, toHexStr( ++entCount ) );
}
if ( version > DRW::AC1012 )
{
writer->writeString( 330, "A" );
}
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbDimStyleTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
writer->writeUtf8Caps( 2, ent->name );
writer->writeInt16( 70, ent->flags );
if ( version == DRW::AC1009 || !( ent->dimpost.empty() ) )
writer->writeUtf8String( 3, ent->dimpost );
if ( version == DRW::AC1009 || !( ent->dimapost.empty() ) )
writer->writeUtf8String( 4, ent->dimapost );
if ( version == DRW::AC1009 || !( ent->dimblk.empty() ) )
writer->writeUtf8String( 5, ent->dimblk );
if ( version == DRW::AC1009 || !( ent->dimblk1.empty() ) )
writer->writeUtf8String( 6, ent->dimblk1 );
if ( version == DRW::AC1009 || !( ent->dimblk2.empty() ) )
writer->writeUtf8String( 7, ent->dimblk2 );
writer->writeDouble( 40, ent->dimscale );
writer->writeDouble( 41, ent->dimasz );
writer->writeDouble( 42, ent->dimexo );
writer->writeDouble( 43, ent->dimdli );
writer->writeDouble( 44, ent->dimexe );
writer->writeDouble( 45, ent->dimrnd );
writer->writeDouble( 46, ent->dimdle );
writer->writeDouble( 47, ent->dimtp );
writer->writeDouble( 48, ent->dimtm );
if ( version > DRW::AC1018 || ent->dimfxl != 0 )
writer->writeDouble( 49, ent->dimfxl );
writer->writeDouble( 140, ent->dimtxt );
writer->writeDouble( 141, ent->dimcen );
writer->writeDouble( 142, ent->dimtsz );
writer->writeDouble( 143, ent->dimaltf );
writer->writeDouble( 144, ent->dimlfac );
writer->writeDouble( 145, ent->dimtvp );
writer->writeDouble( 146, ent->dimtfac );
writer->writeDouble( 147, ent->dimgap );
if ( version > DRW::AC1014 )
{
writer->writeDouble( 148, ent->dimaltrnd );
}
writer->writeInt16( 71, ent->dimtol );
writer->writeInt16( 72, ent->dimlim );
writer->writeInt16( 73, ent->dimtih );
writer->writeInt16( 74, ent->dimtoh );
writer->writeInt16( 75, ent->dimse1 );
writer->writeInt16( 76, ent->dimse2 );
writer->writeInt16( 77, ent->dimtad );
writer->writeInt16( 78, ent->dimzin );
if ( version > DRW::AC1014 )
{
writer->writeInt16( 79, ent->dimazin );
}
writer->writeInt16( 170, ent->dimalt );
writer->writeInt16( 171, ent->dimaltd );
writer->writeInt16( 172, ent->dimtofl );
writer->writeInt16( 173, ent->dimsah );
writer->writeInt16( 174, ent->dimtix );
writer->writeInt16( 175, ent->dimsoxd );
writer->writeInt16( 176, ent->dimclrd );
writer->writeInt16( 177, ent->dimclre );
writer->writeInt16( 178, ent->dimclrt );
if ( version > DRW::AC1014 )
{
writer->writeInt16( 179, ent->dimadec );
}
if ( version > DRW::AC1009 )
{
if ( version < DRW::AC1015 )
writer->writeInt16( 270, ent->dimunit );
writer->writeInt16( 271, ent->dimdec );
writer->writeInt16( 272, ent->dimtdec );
writer->writeInt16( 273, ent->dimaltu );
writer->writeInt16( 274, ent->dimalttd );
writer->writeInt16( 275, ent->dimaunit );
}
if ( version > DRW::AC1014 )
{
writer->writeInt16( 276, ent->dimfrac );
writer->writeInt16( 277, ent->dimlunit );
writer->writeInt16( 278, ent->dimdsep );
writer->writeInt16( 279, ent->dimtmove );
}
if ( version > DRW::AC1009 )
{
writer->writeInt16( 280, ent->dimjust );
writer->writeInt16( 281, ent->dimsd1 );
writer->writeInt16( 282, ent->dimsd2 );
writer->writeInt16( 283, ent->dimtolj );
writer->writeInt16( 284, ent->dimtzin );
writer->writeInt16( 285, ent->dimaltz );
writer->writeInt16( 286, ent->dimaltttz );
if ( version < DRW::AC1015 )
writer->writeInt16( 287, ent->dimfit );
writer->writeInt16( 288, ent->dimupt );
}
if ( version > DRW::AC1014 )
{
writer->writeInt16( 289, ent->dimatfit );
}
if ( version > DRW::AC1018 && ent->dimfxlon != 0 )
writer->writeInt16( 290, ent->dimfxlon );
if ( version > DRW::AC1009 )
{
writer->writeUtf8String( 340, ent->dimtxsty );
}
if ( version > DRW::AC1014 )
{
writer->writeUtf8String( 341, ent->dimldrblk );
writer->writeInt16( 371, ent->dimlwd );
writer->writeInt16( 372, ent->dimlwe );
}
return true;
}
bool dxfRW::writeAppId( DRW_AppId *ent )
{
std::string strname = ent->name;
transform( strname.begin(), strname.end(), strname.begin(), ::toupper );
//do not write mandatory ACAD appId, handled by library
if ( strname == "ACAD" )
return true;
writer->writeString( 0, "APPID" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( ++entCount ) );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "9" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbRegAppTableRecord" );
writer->writeUtf8String( 2, ent->name );
}
else
{
writer->writeUtf8Caps( 2, ent->name );
}
writer->writeInt16( 70, ent->flags );
return true;
}
bool dxfRW::writePoint( DRW_Point *ent )
{
writer->writeString( 0, "POINT" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbPoint" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
}
return true;
}
bool dxfRW::writeLine( DRW_Line *ent )
{
writer->writeString( 0, "LINE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbLine" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 || ent->secPoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
}
else
{
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
}
return true;
}
bool dxfRW::writeRay( DRW_Ray *ent )
{
writer->writeString( 0, "RAY" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbRay" );
}
DRW_Coord crd = ent->secPoint;
crd.unitize();
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 || ent->secPoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, crd.x );
writer->writeDouble( 21, crd.y );
writer->writeDouble( 31, crd.z );
}
else
{
writer->writeDouble( 11, crd.x );
writer->writeDouble( 21, crd.y );
}
return true;
}
bool dxfRW::writeXline( DRW_Xline *ent )
{
writer->writeString( 0, "XLINE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbXline" );
}
DRW_Coord crd = ent->secPoint;
crd.unitize();
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 || ent->secPoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, crd.x );
writer->writeDouble( 21, crd.y );
writer->writeDouble( 31, crd.z );
}
else
{
writer->writeDouble( 11, crd.x );
writer->writeDouble( 21, crd.y );
}
return true;
}
bool dxfRW::writeCircle( DRW_Circle *ent )
{
writer->writeString( 0, "CIRCLE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbCircle" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
}
writer->writeDouble( 40, ent->mRadius );
return true;
}
bool dxfRW::writeArc( DRW_Arc *ent )
{
writer->writeString( 0, "ARC" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbCircle" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 )
{
writer->writeDouble( 30, ent->basePoint.z );
}
writer->writeDouble( 40, ent->mRadius );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbArc" );
}
writer->writeDouble( 50, ent->staangle * ARAD );
writer->writeDouble( 51, ent->endangle * ARAD );
return true;
}
bool dxfRW::writeEllipse( DRW_Ellipse *ent )
{
//verify axis/ratio and params for full ellipse
ent->correctAxis();
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "ELLIPSE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbEllipse" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
writer->writeDouble( 40, ent->ratio );
writer->writeDouble( 41, ent->staparam );
writer->writeDouble( 42, ent->endparam );
}
else
{
DRW_Polyline pol;
//RLZ: copy properties
ent->toPolyline( &pol, elParts );
writePolyline( &pol );
}
return true;
}
bool dxfRW::writeTrace( DRW_Trace *ent )
{
writer->writeString( 0, "TRACE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbTrace" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
writer->writeDouble( 12, ent->thirdPoint.x );
writer->writeDouble( 22, ent->thirdPoint.y );
writer->writeDouble( 32, ent->thirdPoint.z );
writer->writeDouble( 13, ent->fourthPoint.x );
writer->writeDouble( 23, ent->fourthPoint.y );
writer->writeDouble( 33, ent->fourthPoint.z );
return true;
}
bool dxfRW::writeSolid( DRW_Solid *ent )
{
writer->writeString( 0, "SOLID" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbTrace" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
writer->writeDouble( 12, ent->thirdPoint.x );
writer->writeDouble( 22, ent->thirdPoint.y );
writer->writeDouble( 32, ent->thirdPoint.z );
writer->writeDouble( 13, ent->fourthPoint.x );
writer->writeDouble( 23, ent->fourthPoint.y );
writer->writeDouble( 33, ent->fourthPoint.z );
return true;
}
bool dxfRW::write3dface( DRW_3Dface *ent )
{
writer->writeString( 0, "3DFACE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbFace" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
writer->writeDouble( 12, ent->thirdPoint.x );
writer->writeDouble( 22, ent->thirdPoint.y );
writer->writeDouble( 32, ent->thirdPoint.z );
writer->writeDouble( 13, ent->fourthPoint.x );
writer->writeDouble( 23, ent->fourthPoint.y );
writer->writeDouble( 33, ent->fourthPoint.z );
writer->writeInt16( 70, ent->invisibleflag );
return true;
}
bool dxfRW::writeLWPolyline( DRW_LWPolyline *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "LWPOLYLINE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbPolyline" );
}
ent->vertexnum = ent->vertlist.size();
writer->writeInt32( 90, static_cast<int>( ent->vertexnum ) );
writer->writeInt16( 70, ent->flags );
writer->writeDouble( 43, ent->width );
if ( ent->elevation != 0 )
writer->writeDouble( 38, ent->elevation );
if ( ent->thickness != 0 )
writer->writeDouble( 39, ent->thickness );
for ( std::vector<DRW_Vertex2D *>::size_type i = 0; i < ent->vertexnum; i++ )
{
DRW_Vertex2D *v = ent->vertlist.at( i );
writer->writeDouble( 10, v->x );
writer->writeDouble( 20, v->y );
if ( v->stawidth != 0 )
writer->writeDouble( 40, v->stawidth );
if ( v->endwidth != 0 )
writer->writeDouble( 41, v->endwidth );
if ( v->bulge != 0 )
writer->writeDouble( 42, v->bulge );
}
}
else
{
//RLZ: TODO convert lwpolyline in polyline (not exist in acad 12)
}
return true;
}
bool dxfRW::writePolyline( DRW_Polyline *ent )
{
writer->writeString( 0, "POLYLINE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
if ( ent->flags & 8 || ent->flags & 16 )
writer->writeString( 100, "AcDb2dPolyline" );
else
writer->writeString( 100, "AcDb3dPolyline" );
}
else
writer->writeInt16( 66, 1 );
writer->writeDouble( 10, 0.0 );
writer->writeDouble( 20, 0.0 );
writer->writeDouble( 30, ent->basePoint.z );
if ( ent->thickness != 0 )
{
writer->writeDouble( 39, ent->thickness );
}
writer->writeInt16( 70, ent->flags );
if ( ent->defstawidth != 0 )
{
writer->writeDouble( 40, ent->defstawidth );
}
if ( ent->defendwidth != 0 )
{
writer->writeDouble( 41, ent->defendwidth );
}
if ( ent->flags & 16 || ent->flags & 32 )
{
writer->writeInt16( 71, ent->vertexcount );
writer->writeInt16( 72, ent->facecount );
}
if ( ent->smoothM != 0 )
{
writer->writeInt16( 73, ent->smoothM );
}
if ( ent->smoothN != 0 )
{
writer->writeInt16( 74, ent->smoothN );
}
if ( ent->curvetype != 0 )
{
writer->writeInt16( 75, ent->curvetype );
}
DRW_Coord crd = ent->extPoint;
if ( crd.x != 0 || crd.y != 0 || crd.z != 1 )
{
writer->writeDouble( 210, crd.x );
writer->writeDouble( 220, crd.y );
writer->writeDouble( 230, crd.z );
}
std::vector<DRW_Vertex2D *>::size_type vertexnum = ent->vertlist.size();
for ( std::vector<DRW_Vertex2D *>::size_type i = 0; i < vertexnum; i++ )
{
DRW_Vertex *v = ent->vertlist.at( i );
writer->writeString( 0, "VERTEX" );
writeEntity( ent );
if ( version > DRW::AC1009 )
writer->writeString( 100, "AcDbVertex" );
if ( ( v->flags & 128 ) && !( v->flags & 64 ) )
{
writer->writeDouble( 10, 0 );
writer->writeDouble( 20, 0 );
writer->writeDouble( 30, 0 );
}
else
{
writer->writeDouble( 10, v->basePoint.x );
writer->writeDouble( 20, v->basePoint.y );
writer->writeDouble( 30, v->basePoint.z );
}
if ( v->stawidth != 0 )
writer->writeDouble( 40, v->stawidth );
if ( v->endwidth != 0 )
writer->writeDouble( 41, v->endwidth );
if ( v->bulge != 0 )
writer->writeDouble( 42, v->bulge );
if ( v->flags != 0 )
{
writer->writeInt16( 70, ent->flags );
}
if ( v->flags & 2 )
{
writer->writeDouble( 50, v->tgdir );
}
if ( v->flags & 128 )
{
if ( v->vindex1 != 0 )
{
writer->writeInt16( 71, v->vindex1 );
}
if ( v->vindex2 != 0 )
{
writer->writeInt16( 72, v->vindex2 );
}
if ( v->vindex3 != 0 )
{
writer->writeInt16( 73, v->vindex3 );
}
if ( v->vindex4 != 0 )
{
writer->writeInt16( 74, v->vindex4 );
}
if ( !( v->flags & 64 ) )
{
writer->writeInt32( 91, v->identifier );
}
}
}
writer->writeString( 0, "SEQEND" );
writeEntity( ent );
return true;
}
bool dxfRW::writeSpline( DRW_Spline *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "SPLINE" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbSpline" );
}
writer->writeDouble( 210, ent->normalVec.x );
writer->writeDouble( 220, ent->normalVec.y );
writer->writeDouble( 230, ent->normalVec.z );
writer->writeInt16( 70, ent->flags );
writer->writeInt16( 71, ent->degree );
writer->writeInt16( 72, ent->nknots );
writer->writeInt16( 73, ent->ncontrol );
writer->writeInt16( 74, ent->nfit );
writer->writeDouble( 42, ent->tolknot );
writer->writeDouble( 43, ent->tolcontrol );
//RLZ: warning check if nknots are correct and ncontrol
for ( int i = 0; i < ent->nknots; i++ )
{
writer->writeDouble( 40, ent->knotslist.at( i ) );
}
for ( int i = 0; i < ent->ncontrol; i++ )
{
DRW_Coord *crd = ent->controllist.at( i );
writer->writeDouble( 10, crd->x );
writer->writeDouble( 20, crd->y );
writer->writeDouble( 30, crd->z );
}
}
else
{
//RLZ: TODO convert spline in polyline (not exist in acad 12)
}
return true;
}
bool dxfRW::writeHatch( DRW_Hatch *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "HATCH" );
writeEntity( ent );
writer->writeString( 100, "AcDbHatch" );
writer->writeDouble( 10, 0.0 );
writer->writeDouble( 20, 0.0 );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 210, ent->extPoint.x );
writer->writeDouble( 220, ent->extPoint.y );
writer->writeDouble( 230, ent->extPoint.z );
writer->writeString( 2, ent->name );
writer->writeInt16( 70, ent->solid );
writer->writeInt16( 71, ent->associative );
ent->loopsnum = ent->looplist.size();
writer->writeInt16( 91, static_cast<int>( ent->loopsnum ) );
//write paths data
for ( std::vector<DRW_HatchLoop *>::size_type i = 0; i < ent->loopsnum; i++ )
{
DRW_HatchLoop *loop = ent->looplist.at( i );
writer->writeInt16( 92, loop->type );
if ( ( loop->type & 2 ) == 2 )
{
//RLZ: polyline boundary writeme
}
else
{
//boundary path
loop->update();
writer->writeInt16( 93, static_cast<int>( loop->numedges ) );
for ( std::vector<DRW_Entity *>::size_type j = 0; j < loop->numedges; ++j )
{
switch ( ( loop->objlist.at( j ) )->eType )
{
case DRW::LINE:
{
writer->writeInt16( 72, 1 );
DRW_Line *l = ( DRW_Line * )loop->objlist.at( j );
writer->writeDouble( 10, l->basePoint.x );
writer->writeDouble( 20, l->basePoint.y );
writer->writeDouble( 11, l->secPoint.x );
writer->writeDouble( 21, l->secPoint.y );
break;
}
case DRW::ARC:
{
writer->writeInt16( 72, 2 );
DRW_Arc *a = ( DRW_Arc * )loop->objlist.at( j );
writer->writeDouble( 10, a->basePoint.x );
writer->writeDouble( 20, a->basePoint.y );
writer->writeDouble( 40, a->mRadius );
writer->writeDouble( 50, a->staangle * ARAD );
writer->writeDouble( 51, a->endangle * ARAD );
writer->writeInt16( 73, a->isccw );
break;
}
case DRW::ELLIPSE:
{
writer->writeInt16( 72, 3 );
DRW_Ellipse *a = ( DRW_Ellipse * )loop->objlist.at( j );
a->correctAxis();
writer->writeDouble( 10, a->basePoint.x );
writer->writeDouble( 20, a->basePoint.y );
writer->writeDouble( 11, a->secPoint.x );
writer->writeDouble( 21, a->secPoint.y );
writer->writeDouble( 40, a->ratio );
writer->writeDouble( 50, a->staparam * ARAD );
writer->writeDouble( 51, a->endparam * ARAD );
writer->writeInt16( 73, a->isccw );
break;
}
case DRW::SPLINE:
//RLZ: spline boundary writeme
// writer->writeInt16(72, 4);
break;
default:
break;
}
}
writer->writeInt16( 97, 0 );
}
}
writer->writeInt16( 75, ent->hstyle );
writer->writeInt16( 76, ent->hpattern );
if ( !ent->solid )
{
writer->writeDouble( 52, ent->angle );
writer->writeDouble( 41, ent->scale );
writer->writeInt16( 77, ent->doubleflag );
writer->writeInt16( 78, ent->deflines );
}
#if 0
if ( ent->deflines > 0 )
{
writer->writeInt16( 78, ent->deflines );
}
#endif
writer->writeInt32( 98, 0 );
}
else
{
//RLZ: TODO verify in acad12
}
return true;
}
bool dxfRW::writeLeader( DRW_Leader *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "LEADER" );
writeEntity( ent );
writer->writeString( 100, "AcDbLeader" );
writer->writeUtf8String( 3, ent->style );
writer->writeInt16( 71, ent->arrow );
writer->writeInt16( 72, ent->leadertype );
writer->writeInt16( 73, ent->flag );
writer->writeInt16( 74, ent->hookline );
writer->writeInt16( 75, ent->hookflag );
writer->writeDouble( 40, ent->textheight );
writer->writeDouble( 41, ent->textwidth );
writer->writeDouble( 76, ent->vertnum );
writer->writeDouble( 76, ent->vertexlist.size() );
for ( unsigned int i = 0; i < ent->vertexlist.size(); i++ )
{
DRW_Coord *vert = ent->vertexlist.at( i );
writer->writeDouble( 10, vert->x );
writer->writeDouble( 20, vert->y );
writer->writeDouble( 30, vert->z );
}
}
else
{
//RLZ: todo not supported by acad 12 saved as unnamed block
}
return true;
}
bool dxfRW::writeDimension( DRW_Dimension *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "DIMENSION" );
writeEntity( ent );
writer->writeString( 100, "AcDbDimension" );
if ( !ent->getName().empty() )
{
writer->writeString( 2, ent->getName() );
}
writer->writeDouble( 10, ent->getDefPoint().x );
writer->writeDouble( 20, ent->getDefPoint().y );
writer->writeDouble( 30, ent->getDefPoint().z );
writer->writeDouble( 11, ent->getTextPoint().x );
writer->writeDouble( 21, ent->getTextPoint().y );
writer->writeDouble( 31, ent->getTextPoint().z );
if ( !( ent->type & 32 ) )
ent->type = ent->type + 32;
writer->writeInt16( 70, ent->type );
if ( !( ent->getText().empty() ) )
writer->writeUtf8String( 1, ent->getText() );
writer->writeInt16( 71, ent->getAlign() );
if ( ent->getTextLineStyle() != 1 )
writer->writeInt16( 72, ent->getTextLineStyle() );
if ( ent->getTextLineFactor() != 1 )
writer->writeDouble( 41, ent->getTextLineFactor() );
writer->writeUtf8String( 3, ent->getStyle() );
if ( ent->getTextLineFactor() != 0 )
writer->writeDouble( 53, ent->getDir() );
writer->writeDouble( 210, ent->getExtrusion().x );
writer->writeDouble( 220, ent->getExtrusion().y );
writer->writeDouble( 230, ent->getExtrusion().z );
switch ( ent->eType )
{
case DRW::DIMALIGNED:
case DRW::DIMLINEAR:
{
DRW_DimAligned *dd = ( DRW_DimAligned * )ent;
writer->writeString( 100, "AcDbAlignedDimension" );
DRW_Coord crd = dd->getClonepoint();
if ( crd.x != 0 || crd.y != 0 || crd.z != 0 )
{
writer->writeDouble( 12, crd.x );
writer->writeDouble( 22, crd.y );
writer->writeDouble( 32, crd.z );
}
writer->writeDouble( 13, dd->getDef1Point().x );
writer->writeDouble( 23, dd->getDef1Point().y );
writer->writeDouble( 33, dd->getDef1Point().z );
writer->writeDouble( 14, dd->getDef2Point().x );
writer->writeDouble( 24, dd->getDef2Point().y );
writer->writeDouble( 34, dd->getDef2Point().z );
if ( ent->eType == DRW::DIMLINEAR )
{
DRW_DimLinear *dl = ( DRW_DimLinear * )ent;
if ( dl->getAngle() != 0 )
writer->writeDouble( 50, dl->getAngle() );
if ( dl->getOblique() != 0 )
writer->writeDouble( 52, dl->getOblique() );
writer->writeString( 100, "AcDbRotatedDimension" );
}
break;
}
case DRW::DIMRADIAL:
{
DRW_DimRadial *dd = ( DRW_DimRadial * )ent;
writer->writeString( 100, "AcDbRadialDimension" );
writer->writeDouble( 15, dd->getDiameterPoint().x );
writer->writeDouble( 25, dd->getDiameterPoint().y );
writer->writeDouble( 35, dd->getDiameterPoint().z );
writer->writeDouble( 40, dd->getLeaderLength() );
break;
}
case DRW::DIMDIAMETRIC:
{
DRW_DimDiametric *dd = ( DRW_DimDiametric * )ent;
writer->writeString( 100, "AcDbDiametricDimension" );
writer->writeDouble( 15, dd->getDiameter1Point().x );
writer->writeDouble( 25, dd->getDiameter1Point().y );
writer->writeDouble( 35, dd->getDiameter1Point().z );
writer->writeDouble( 40, dd->getLeaderLength() );
break;
}
case DRW::DIMANGULAR:
{
DRW_DimAngular *dd = ( DRW_DimAngular * )ent;
writer->writeString( 100, "AcDb2LineAngularDimension" );
writer->writeDouble( 13, dd->getFirstLine1().x );
writer->writeDouble( 23, dd->getFirstLine1().y );
writer->writeDouble( 33, dd->getFirstLine1().z );
writer->writeDouble( 14, dd->getFirstLine2().x );
writer->writeDouble( 24, dd->getFirstLine2().y );
writer->writeDouble( 34, dd->getFirstLine2().z );
writer->writeDouble( 15, dd->getSecondLine1().x );
writer->writeDouble( 25, dd->getSecondLine1().y );
writer->writeDouble( 35, dd->getSecondLine1().z );
writer->writeDouble( 16, dd->getDimPoint().x );
writer->writeDouble( 26, dd->getDimPoint().y );
writer->writeDouble( 36, dd->getDimPoint().z );
break;
}
case DRW::DIMANGULAR3P:
{
DRW_DimAngular3p *dd = ( DRW_DimAngular3p * )ent;
writer->writeDouble( 13, dd->getFirstLine().x );
writer->writeDouble( 23, dd->getFirstLine().y );
writer->writeDouble( 33, dd->getFirstLine().z );
writer->writeDouble( 14, dd->getSecondLine().x );
writer->writeDouble( 24, dd->getSecondLine().y );
writer->writeDouble( 34, dd->getSecondLine().z );
writer->writeDouble( 15, dd->getVertexPoint().x );
writer->writeDouble( 25, dd->getVertexPoint().y );
writer->writeDouble( 35, dd->getVertexPoint().z );
break;
}
case DRW::DIMORDINATE:
{
DRW_DimOrdinate *dd = ( DRW_DimOrdinate * )ent;
writer->writeString( 100, "AcDbOrdinateDimension" );
writer->writeDouble( 13, dd->getFirstLine().x );
writer->writeDouble( 23, dd->getFirstLine().y );
writer->writeDouble( 33, dd->getFirstLine().z );
writer->writeDouble( 14, dd->getSecondLine().x );
writer->writeDouble( 24, dd->getSecondLine().y );
writer->writeDouble( 34, dd->getSecondLine().z );
break;
}
default:
break;
}
}
else
{
//RLZ: todo not supported by acad 12 saved as unnamed block
}
return true;
}
bool dxfRW::writeInsert( DRW_Insert *ent )
{
writer->writeString( 0, "INSERT" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbBlockReference" );
writer->writeUtf8String( 2, ent->name );
}
else
writer->writeUtf8Caps( 2, ent->name );
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 41, ent->xscale );
writer->writeDouble( 42, ent->yscale );
writer->writeDouble( 43, ent->zscale );
writer->writeDouble( 50, ( ent->angle )*ARAD ); //in dxf angle is writed in degrees
writer->writeInt16( 70, ent->colcount );
writer->writeInt16( 71, ent->rowcount );
writer->writeDouble( 44, ent->colspace );
writer->writeDouble( 45, ent->rowspace );
return true;
}
bool dxfRW::writeText( DRW_Text *ent )
{
writer->writeString( 0, "TEXT" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbText" );
}
// writer->writeDouble(39, ent->thickness);
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 40, ent->height );
writer->writeUtf8String( 1, ent->text );
writer->writeDouble( 50, ent->angle );
writer->writeDouble( 41, ent->widthscale );
writer->writeDouble( 51, ent->oblique );
if ( version > DRW::AC1009 )
writer->writeUtf8String( 7, ent->style );
else
writer->writeUtf8Caps( 7, ent->style );
writer->writeInt16( 71, ent->textgen );
if ( ent->alignH != DRW_Text::HLeft )
{
writer->writeInt16( 72, ent->alignH );
}
if ( ent->alignH != DRW_Text::HLeft || ent->alignV != DRW_Text::VBaseLine )
{
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
}
writer->writeDouble( 210, ent->extPoint.x );
writer->writeDouble( 220, ent->extPoint.y );
writer->writeDouble( 230, ent->extPoint.z );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbText" );
}
if ( ent->alignV != DRW_Text::VBaseLine )
{
writer->writeInt16( 73, ent->alignV );
}
return true;
}
bool dxfRW::writeMText( DRW_MText *ent )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "MTEXT" );
writeEntity( ent );
writer->writeString( 100, "AcDbMText" );
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 40, ent->height );
writer->writeDouble( 41, ent->widthscale );
writer->writeInt16( 71, ent->textgen );
writer->writeInt16( 72, ent->alignH );
std::string text = writer->fromUtf8String( ent->text );
int i;
for ( i = 0; ( text.size() - i ) > 250; )
{
writer->writeString( 3, text.substr( i, 250 ) );
i += 250;
}
writer->writeString( 1, text.substr( i ) );
writer->writeString( 7, ent->style );
writer->writeDouble( 210, ent->extPoint.x );
writer->writeDouble( 220, ent->extPoint.y );
writer->writeDouble( 230, ent->extPoint.z );
writer->writeDouble( 50, ent->angle );
writer->writeInt16( 73, ent->alignV );
writer->writeDouble( 44, ent->interlin );
//RLZ ... 11, 21, 31 needed?
}
else
{
//RLZ: TODO convert mtext in text lines (not exist in acad 12)
}
return true;
}
bool dxfRW::writeViewport( DRW_Viewport *ent )
{
writer->writeString( 0, "VIEWPORT" );
writeEntity( ent );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbViewport" );
}
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
if ( ent->basePoint.z != 0.0 )
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 40, ent->pswidth );
writer->writeDouble( 41, ent->psheight );
writer->writeInt16( 68, ent->vpstatus );
writer->writeInt16( 69, ent->vpID );
writer->writeDouble( 12, ent->centerPX );//RLZ: verify if exist in V12
writer->writeDouble( 22, ent->centerPY );//RLZ: verify if exist in V12
return true;
}
DRW_ImageDef *dxfRW::writeImage( DRW_Image *ent, std::string name )
{
if ( version > DRW::AC1009 )
{
//search if exist imagedef with this mane (image inserted more than 1 time)
//RLZ: imagedef_reactor seem needed to read in acad
DRW_ImageDef *id = nullptr;
for ( unsigned int i = 0; i < imageDef.size(); i++ )
{
if ( imageDef.at( i )->name == name )
{
id = imageDef.at( i );
continue;
}
}
if ( !id )
{
id = new DRW_ImageDef();
imageDef.push_back( id );
id->handle = ++entCount;
}
id->name = name;
std::string idReactor = toHexStr( ++entCount );
writer->writeString( 0, "IMAGE" );
writeEntity( ent );
writer->writeString( 100, "AcDbRasterImage" );
writer->writeDouble( 10, ent->basePoint.x );
writer->writeDouble( 20, ent->basePoint.y );
writer->writeDouble( 30, ent->basePoint.z );
writer->writeDouble( 11, ent->secPoint.x );
writer->writeDouble( 21, ent->secPoint.y );
writer->writeDouble( 31, ent->secPoint.z );
writer->writeDouble( 12, ent->vVector.x );
writer->writeDouble( 22, ent->vVector.y );
writer->writeDouble( 32, ent->vVector.z );
writer->writeDouble( 13, ent->sizeu );
writer->writeDouble( 23, ent->sizev );
writer->writeString( 340, toHexStr( id->handle ) );
writer->writeInt16( 70, 1 );
writer->writeInt16( 280, ent->clip );
writer->writeInt16( 281, ent->brightness );
writer->writeInt16( 282, ent->contrast );
writer->writeInt16( 283, ent->fade );
writer->writeString( 360, idReactor );
id->reactors[idReactor] = toHexStr( ent->handle );
return id;
}
return nullptr; //not exist in acad 12
}
bool dxfRW::writeBlockRecord( std::string name )
{
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "BLOCK_RECORD" );
writer->writeString( 5, toHexStr( ++entCount ) );
blockMap[name] = entCount;
entCount = 2 + entCount;//reserve 2 for BLOCK & ENDBLOCK
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbBlockTableRecord" );
writer->writeUtf8String( 2, name );
if ( version > DRW::AC1018 )
{
// writer->writeInt16(340, 22);
writer->writeInt16( 70, 0 );
writer->writeInt16( 280, 1 );
writer->writeInt16( 281, 0 );
}
}
return true;
}
bool dxfRW::writeBlock( DRW_Block *bk )
{
if ( writingBlock )
{
writer->writeString( 0, "ENDBLK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( currHandle + 2 ) );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, toHexStr( currHandle ) );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbBlockEnd" );
}
}
writingBlock = true;
writer->writeString( 0, "BLOCK" );
if ( version > DRW::AC1009 )
{
currHandle = ( *( blockMap.find( bk->name ) ) ).second;
writer->writeString( 5, toHexStr( currHandle + 1 ) );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, toHexStr( currHandle ) );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbBlockBegin" );
writer->writeUtf8String( 2, bk->name );
}
else
writer->writeUtf8Caps( 2, bk->name );
writer->writeInt16( 70, bk->flags );
writer->writeDouble( 10, bk->basePoint.x );
writer->writeDouble( 20, bk->basePoint.y );
if ( bk->basePoint.z != 0.0 )
{
writer->writeDouble( 30, bk->basePoint.z );
}
if ( version > DRW::AC1009 )
writer->writeUtf8String( 3, bk->name );
else
writer->writeUtf8Caps( 3, bk->name );
writer->writeString( 1, "" );
return true;
}
bool dxfRW::writeTables()
{
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "VPORT" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "8" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 1 ); //end table def
/*** VPORT ***/
dimstyleStd = false;
iface->writeVports();
if ( !dimstyleStd )
{
DRW_Vport portact;
portact.name = "*ACTIVE";
writeVport( &portact );
}
writer->writeString( 0, "ENDTAB" );
/*** LTYPE ***/
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "LTYPE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "5" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 4 ); //end table def
//Mandatory linetypes
writer->writeString( 0, "LTYPE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "14" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "5" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbLinetypeTableRecord" );
writer->writeString( 2, "ByBlock" );
}
else
writer->writeString( 2, "BYBLOCK" );
writer->writeInt16( 70, 0 );
writer->writeString( 3, "" );
writer->writeInt16( 72, 65 );
writer->writeInt16( 73, 0 );
writer->writeDouble( 40, 0.0 );
writer->writeString( 0, "LTYPE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "15" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "5" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbLinetypeTableRecord" );
writer->writeString( 2, "ByLayer" );
}
else
writer->writeString( 2, "BYLAYER" );
writer->writeInt16( 70, 0 );
writer->writeString( 3, "" );
writer->writeInt16( 72, 65 );
writer->writeInt16( 73, 0 );
writer->writeDouble( 40, 0.0 );
writer->writeString( 0, "LTYPE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "16" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "5" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbLinetypeTableRecord" );
writer->writeString( 2, "Continuous" );
}
else
{
writer->writeString( 2, "CONTINUOUS" );
}
writer->writeInt16( 70, 0 );
writer->writeString( 3, "Solid line" );
writer->writeInt16( 72, 65 );
writer->writeInt16( 73, 0 );
writer->writeDouble( 40, 0.0 );
// Application linetypes
iface->writeLTypes();
writer->writeString( 0, "ENDTAB" );
/*** LAYER ***/
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "LAYER" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "2" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 1 ); //end table def
wlayer0 = false;
iface->writeLayers();
if ( !wlayer0 )
{
DRW_Layer lay0;
lay0.name = "0";
writeLayer( &lay0 );
}
writer->writeString( 0, "ENDTAB" );
/*** STYLE ***/
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "STYLE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "3" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 3 ); //end table def
dimstyleStd = false;
iface->writeTextstyles();
if ( !dimstyleStd )
{
DRW_Textstyle tsty;
tsty.name = "Standard";
writeTextstyle( &tsty );
}
writer->writeString( 0, "ENDTAB" );
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "VIEW" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "6" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 0 ); //end table def
writer->writeString( 0, "ENDTAB" );
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "UCS" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "7" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 0 ); //end table def
writer->writeString( 0, "ENDTAB" );
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "APPID" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "9" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 1 ); //end table def
writer->writeString( 0, "APPID" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "12" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "9" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbRegAppTableRecord" );
}
writer->writeString( 2, "ACAD" );
writer->writeInt16( 70, 0 );
iface->writeAppId();
writer->writeString( 0, "ENDTAB" );
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "DIMSTYLE" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "A" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
}
writer->writeInt16( 70, 1 ); //end table def
if ( version > DRW::AC1014 )
{
writer->writeString( 100, "AcDbDimStyleTable" );
writer->writeInt16( 71, 1 ); //end table def
}
dimstyleStd = false;
iface->writeDimstyles();
if ( !dimstyleStd )
{
DRW_Dimstyle dsty;
dsty.name = "Standard";
writeDimstyle( &dsty );
}
writer->writeString( 0, "ENDTAB" );
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "TABLE" );
writer->writeString( 2, "BLOCK_RECORD" );
writer->writeString( 5, "1" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbSymbolTable" );
writer->writeInt16( 70, 2 ); //end table def
writer->writeString( 0, "BLOCK_RECORD" );
writer->writeString( 5, "1F" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbBlockTableRecord" );
writer->writeString( 2, "*Model_Space" );
if ( version > DRW::AC1018 )
{
// writer->writeInt16(340, 22);
writer->writeInt16( 70, 0 );
writer->writeInt16( 280, 1 );
writer->writeInt16( 281, 0 );
}
writer->writeString( 0, "BLOCK_RECORD" );
writer->writeString( 5, "1E" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1" );
}
writer->writeString( 100, "AcDbSymbolTableRecord" );
writer->writeString( 100, "AcDbBlockTableRecord" );
writer->writeString( 2, "*Paper_Space" );
if ( version > DRW::AC1018 )
{
// writer->writeInt16(340, 22);
writer->writeInt16( 70, 0 );
writer->writeInt16( 280, 1 );
writer->writeInt16( 281, 0 );
}
}
/* always call writeBlockRecords to iface for prepare unnamed blocks */
iface->writeBlockRecords();
if ( version > DRW::AC1009 )
{
writer->writeString( 0, "ENDTAB" );
}
return true;
}
bool dxfRW::writeBlocks()
{
writer->writeString( 0, "BLOCK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "20" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1F" );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbBlockBegin" );
writer->writeString( 2, "*Model_Space" );
}
else
writer->writeString( 2, "$MODEL_SPACE" );
writer->writeInt16( 70, 0 );
writer->writeDouble( 10, 0.0 );
writer->writeDouble( 20, 0.0 );
writer->writeDouble( 30, 0.0 );
if ( version > DRW::AC1009 )
writer->writeString( 3, "*Model_Space" );
else
writer->writeString( 3, "$MODEL_SPACE" );
writer->writeString( 1, "" );
writer->writeString( 0, "ENDBLK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "21" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1F" );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
writer->writeString( 100, "AcDbBlockEnd" );
writer->writeString( 0, "BLOCK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "1C" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1B" );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
{
writer->writeString( 100, "AcDbBlockBegin" );
writer->writeString( 2, "*Paper_Space" );
}
else
writer->writeString( 2, "$PAPER_SPACE" );
writer->writeInt16( 70, 0 );
writer->writeDouble( 10, 0.0 );
writer->writeDouble( 20, 0.0 );
writer->writeDouble( 30, 0.0 );
if ( version > DRW::AC1009 )
writer->writeString( 3, "*Paper_Space" );
else
writer->writeString( 3, "$PAPER_SPACE" );
writer->writeString( 1, "" );
writer->writeString( 0, "ENDBLK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, "1D" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "1F" );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
writer->writeString( 100, "AcDbBlockEnd" );
writingBlock = false;
iface->writeBlocks();
if ( writingBlock )
{
writingBlock = false;
writer->writeString( 0, "ENDBLK" );
if ( version > DRW::AC1009 )
{
writer->writeString( 5, toHexStr( currHandle + 2 ) );
// writer->writeString(5, "1D");
if ( version > DRW::AC1014 )
{
writer->writeString( 330, toHexStr( currHandle ) );
}
writer->writeString( 100, "AcDbEntity" );
}
writer->writeString( 8, "0" );
if ( version > DRW::AC1009 )
writer->writeString( 100, "AcDbBlockEnd" );
}
return true;
}
bool dxfRW::writeObjects()
{
writer->writeString( 0, "DICTIONARY" );
std::string imgDictH;
writer->writeString( 5, "C" );
if ( version > DRW::AC1014 )
{
writer->writeString( 330, "0" );
}
writer->writeString( 100, "AcDbDictionary" );
writer->writeInt16( 281, 1 );
writer->writeString( 3, "ACAD_GROUP" );
writer->writeString( 350, "D" );
if ( imageDef.size() != 0 )
{
writer->writeString( 3, "ACAD_IMAGE_DICT" );
imgDictH = toHexStr( ++entCount );
writer->writeString( 350, imgDictH );
}
writer->writeString( 0, "DICTIONARY" );
writer->writeString( 5, "D" );
writer->writeString( 330, "C" );
writer->writeString( 100, "AcDbDictionary" );
writer->writeInt16( 281, 1 );
//write IMAGEDEF_REACTOR
for ( unsigned int i = 0; i < imageDef.size(); i++ )
{
DRW_ImageDef *id = imageDef.at( i );
std::map<std::string, std::string>::iterator it;
for ( it = id->reactors.begin() ; it != id->reactors.end(); ++it )
{
writer->writeString( 0, "IMAGEDEF_REACTOR" );
writer->writeString( 5, ( *it ).first );
writer->writeString( 330, ( *it ).second );
writer->writeString( 100, "AcDbRasterImageDefReactor" );
writer->writeInt16( 90, 2 ); //version 2=R14 to v2010
writer->writeString( 330, ( *it ).second );
}
}
if ( imageDef.size() != 0 )
{
writer->writeString( 0, "DICTIONARY" );
writer->writeString( 5, imgDictH );
writer->writeString( 330, "C" );
writer->writeString( 100, "AcDbDictionary" );
writer->writeInt16( 281, 1 );
for ( unsigned int i = 0; i < imageDef.size(); i++ )
{
size_t f1, f2;
f1 = imageDef.at( i )->name.find_last_of( "/\\" );
f2 = imageDef.at( i )->name.find_last_of( '.' );
++f1;
writer->writeString( 3, imageDef.at( i )->name.substr( f1, f2 - f1 ) );
writer->writeString( 350, toHexStr( imageDef.at( i )->handle ) );
}
}
for ( unsigned int i = 0; i < imageDef.size(); i++ )
{
DRW_ImageDef *id = imageDef.at( i );
writer->writeString( 0, "IMAGEDEF" );
writer->writeString( 5, toHexStr( id->handle ) );
if ( version > DRW::AC1014 )
{
// writer->writeString(330, "0"); handle to DICTIONARY
}
writer->writeString( 102, "{ACAD_REACTORS" );
std::map<std::string, std::string>::iterator it;
for ( it = id->reactors.begin() ; it != id->reactors.end(); ++it )
{
writer->writeString( 330, ( *it ).first );
}
writer->writeString( 102, "}" );
writer->writeString( 100, "AcDbRasterImageDef" );
writer->writeInt16( 90, 0 ); //version 0=R14 to v2010
writer->writeUtf8String( 1, id->name );
writer->writeDouble( 10, id->u );
writer->writeDouble( 20, id->v );
writer->writeDouble( 11, id->up );
writer->writeDouble( 21, id->vp );
writer->writeInt16( 280, id->loaded );
writer->writeInt16( 281, id->resolution );
}
//no more needed imageDef, delete it
while ( !imageDef.empty() )
{
imageDef.pop_back();
}
return true;
}
bool dxfRW::writeExtData( const std::vector<DRW_Variant *> &ed )
{
for ( std::vector<DRW_Variant *>::const_iterator it = ed.begin(); it != ed.end(); ++it )
{
switch ( ( *it )->code() )
{
case 1000:
case 1001:
case 1002:
case 1003:
case 1004:
case 1005:
{
int cc = ( *it )->code();
if ( ( *it )->type() == DRW_Variant::STRING )
writer->writeUtf8String( cc, *( *it )->content.s );
// writer->writeUtf8String((*it)->code, (*it)->content.s);
break;
}
case 1010:
case 1011:
case 1012:
case 1013:
if ( ( *it )->type() == DRW_Variant::COORD )
{
writer->writeDouble( ( *it )->code(), ( *it )->content.v->x );
writer->writeDouble( ( *it )->code() + 10, ( *it )->content.v->y );
writer->writeDouble( ( *it )->code() + 20, ( *it )->content.v->z );
}
break;
case 1040:
case 1041:
case 1042:
if ( ( *it )->type() == DRW_Variant::DOUBLE )
writer->writeDouble( ( *it )->code(), ( *it )->content.d );
break;
case 1070:
if ( ( *it )->type() == DRW_Variant::INTEGER )
writer->writeInt16( ( *it )->code(), ( *it )->content.i );
break;
case 1071:
if ( ( *it )->type() == DRW_Variant::INTEGER )
writer->writeInt32( ( *it )->code(), ( *it )->content.i );
break;
default:
break;
}
}
return true;
}
/********* Reader Process *********/
bool dxfRW::processDxf()
{
QgsDebugMsg( "Entering." );
int code;
bool more = true;
std::string sectionstr;
// section = secUnknown;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 999 )
{
header.addComment( reader->getString() );
}
else if ( code == 0 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "EOF" )
{
return true; //found EOF terminate
}
if ( sectionstr == "SECTION" )
{
more = reader->readRec( &code );
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( !more )
return false; //wrong dxf file
if ( code == 2 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
//found section, process it
if ( sectionstr == "HEADER" )
{
processHeader();
}
else if ( sectionstr == "CLASSES" )
{
// processClasses();
}
else if ( sectionstr == "TABLES" )
{
processTables();
}
else if ( sectionstr == "BLOCKS" )
{
processBlocks();
}
else if ( sectionstr == "ENTITIES" )
{
processEntities( false );
}
else if ( sectionstr == "OBJECTS" )
{
processObjects();
}
}
}
}
#if 0
if ( !more )
return true;
#endif
}
return true;
}
/********* Header Section *********/
bool dxfRW::processHeader()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "ENDSEC" )
{
iface->addHeader( &header );
return true; //found ENDSEC terminate
}
}
else header.parseCode( code, reader );
}
return true;
}
/********* Tables Section *********/
bool dxfRW::processTables()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool more = true;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "TABLE" )
{
more = reader->readRec( &code );
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( !more )
return false; //wrong dxf file
if ( code == 2 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
//found section, process it
if ( sectionstr == "LTYPE" )
{
processLType();
}
else if ( sectionstr == "LAYER" )
{
processLayer();
}
else if ( sectionstr == "STYLE" )
{
processTextStyle();
}
else if ( sectionstr == "VPORT" )
{
processVports();
}
else if ( sectionstr == "VIEW" )
{
// processView();
}
else if ( sectionstr == "UCS" )
{
// processUCS();
}
else if ( sectionstr == "APPID" )
{
processAppId();
}
else if ( sectionstr == "DIMSTYLE" )
{
processDimStyle();
}
else if ( sectionstr == "BLOCK_RECORD" )
{
// processBlockRecord();
}
}
}
else if ( sectionstr == "ENDSEC" )
{
return true; //found ENDSEC terminate
}
}
}
return true;
}
bool dxfRW::processLType()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_LType ltype;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
{
ltype.update();
iface->addLType( ltype );
}
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "LTYPE" )
{
reading = true;
ltype.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
ltype.parseCode( code, reader );
}
return true;
}
bool dxfRW::processLayer()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_Layer layer;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
iface->addLayer( layer );
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "LAYER" )
{
reading = true;
layer.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
layer.parseCode( code, reader );
}
return true;
}
bool dxfRW::processDimStyle()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_Dimstyle dimSty;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
iface->addDimStyle( dimSty );
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "DIMSTYLE" )
{
reading = true;
dimSty.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
dimSty.parseCode( code, reader );
}
return true;
}
bool dxfRW::processTextStyle()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_Textstyle TxtSty;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
iface->addTextStyle( TxtSty );
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "STYLE" )
{
reading = true;
TxtSty.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
TxtSty.parseCode( code, reader );
}
return true;
}
bool dxfRW::processVports()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_Vport vp;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
iface->addVport( vp );
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "VPORT" )
{
reading = true;
vp.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
vp.parseCode( code, reader );
}
return true;
}
bool dxfRW::processAppId()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
bool reading = false;
DRW_AppId vp;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
if ( reading )
iface->addAppId( vp );
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "APPID" )
{
reading = true;
vp.reset();
}
else if ( sectionstr == "ENDTAB" )
{
return true; //found ENDTAB terminate
}
}
else if ( reading )
vp.parseCode( code, reader );
}
return true;
}
/********* Block Section *********/
bool dxfRW::processBlocks()
{
QgsDebugMsg( "Entering." );
int code;
std::string sectionstr;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
if ( code == 0 )
{
sectionstr = reader->getString();
QgsDebugMsg( QString( "section=%1" ).arg( sectionstr.c_str() ) );
if ( sectionstr == "BLOCK" )
{
processBlock();
}
else if ( sectionstr == "ENDSEC" )
{
return true; //found ENDSEC terminate
}
}
}
return true;
}
bool dxfRW::processBlock()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Block block;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addBlock( block );
if ( nextentity == "ENDBLK" )
{
iface->endBlock();
return true; //found ENDBLK, terminate
}
else
{
processEntities( true );
iface->endBlock();
return true; //found ENDBLK, terminate
}
}
default:
block.parseCode( code, reader );
break;
}
}
return true;
}
/********* Entities Section *********/
bool dxfRW::processEntities( bool isblock )
{
QgsDebugMsg( "Entering." );
int code;
if ( !reader->readRec( &code ) )
{
return false;
}
bool next = true;
if ( code == 0 )
{
nextentity = reader->getString();
}
else if ( !isblock )
{
return false; //first record in entities is 0
}
do
{
if ( nextentity == "ENDSEC" || nextentity == "ENDBLK" )
{
return true; //found ENDSEC or ENDBLK terminate
}
else if ( nextentity == "POINT" )
{
processPoint();
}
else if ( nextentity == "LINE" )
{
processLine();
}
else if ( nextentity == "CIRCLE" )
{
processCircle();
}
else if ( nextentity == "ARC" )
{
processArc();
}
else if ( nextentity == "ELLIPSE" )
{
processEllipse();
}
else if ( nextentity == "TRACE" )
{
processTrace();
}
else if ( nextentity == "SOLID" )
{
processSolid();
}
else if ( nextentity == "INSERT" )
{
processInsert();
}
else if ( nextentity == "LWPOLYLINE" )
{
processLWPolyline();
}
else if ( nextentity == "POLYLINE" )
{
processPolyline();
}
else if ( nextentity == "TEXT" )
{
processText();
}
else if ( nextentity == "MTEXT" )
{
processMText();
}
else if ( nextentity == "HATCH" )
{
processHatch();
}
else if ( nextentity == "SPLINE" )
{
processSpline();
}
else if ( nextentity == "3DFACE" )
{
process3dface();
}
else if ( nextentity == "VIEWPORT" )
{
processViewport();
}
else if ( nextentity == "IMAGE" )
{
processImage();
}
else if ( nextentity == "DIMENSION" )
{
processDimension();
}
else if ( nextentity == "LEADER" )
{
processLeader();
}
else if ( nextentity == "RAY" )
{
processRay();
}
else if ( nextentity == "XLINE" )
{
processXline();
}
else
{
if ( reader->readRec( &code ) )
{
if ( code == 0 )
nextentity = reader->getString();
}
else
return false; //end of file without ENDSEC
}
}
while ( next );
return true;
}
bool dxfRW::processEllipse()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Ellipse ellipse;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
ellipse.applyExtrusion();
iface->addEllipse( ellipse );
return true; //found new entity or ENDSEC, terminate
}
default:
ellipse.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processTrace()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Trace trace;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
trace.applyExtrusion();
iface->addTrace( trace );
return true; //found new entity or ENDSEC, terminate
}
default:
trace.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processSolid()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Solid solid;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
solid.applyExtrusion();
iface->addSolid( solid );
return true; //found new entity or ENDSEC, terminate
}
default:
solid.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::process3dface()
{
QgsDebugMsg( "Entering." );
int code;
DRW_3Dface face;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->add3dFace( face );
return true; //found new entity or ENDSEC, terminate
}
default:
face.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processViewport()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Viewport vp;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addViewport( vp );
return true; //found new entity or ENDSEC, terminate
}
default:
vp.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processPoint()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Point point;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addPoint( point );
return true; //found new entity or ENDSEC, terminate
}
default:
point.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processLine()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Line line;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addLine( line );
return true; //found new entity or ENDSEC, terminate
}
default:
line.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processRay()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Ray line;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addRay( line );
return true; //found new entity or ENDSEC, terminate
}
default:
line.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processXline()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Xline line;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addXline( line );
return true; //found new entity or ENDSEC, terminate
}
default:
line.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processCircle()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Circle circle;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
circle.applyExtrusion();
iface->addCircle( circle );
return true; //found new entity or ENDSEC, terminate
}
default:
circle.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processArc()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Arc arc;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
arc.applyExtrusion();
iface->addArc( arc );
return true; //found new entity or ENDSEC, terminate
}
default:
arc.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processInsert()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Insert insert;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addInsert( insert );
return true; //found new entity or ENDSEC, terminate
}
default:
insert.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processLWPolyline()
{
QgsDebugMsg( "Entering." );
int code;
DRW_LWPolyline pl;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( applyExt )
pl.applyExtrusion();
iface->addLWPolyline( pl );
return true; //found new entity or ENDSEC, terminate
}
default:
pl.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processPolyline()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Polyline pl;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( nextentity != "VERTEX" )
{
iface->addPolyline( pl );
return true; //found new entity or ENDSEC, terminate
}
else
{
processVertex( &pl );
}
FALLTHROUGH
}
default:
pl.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processVertex( DRW_Polyline *pl )
{
QgsDebugMsg( "Entering." );
int code;
DRW_Vertex *v = new DRW_Vertex();
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
pl->appendVertex( v );
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
if ( nextentity == "SEQEND" )
{
return true; //found SEQEND no more vertex, terminate
}
else if ( nextentity == "VERTEX" )
{
v = new DRW_Vertex(); //another vertex
}
FALLTHROUGH
}
default:
v->parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processText()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Text txt;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addText( txt );
return true; //found new entity or ENDSEC, terminate
}
default:
txt.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processMText()
{
QgsDebugMsg( "Entering." );
int code;
DRW_MText txt;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
txt.updateAngle();
iface->addMText( txt );
return true; //found new entity or ENDSEC, terminate
}
default:
txt.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processHatch()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Hatch hatch;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addHatch( &hatch );
return true; //found new entity or ENDSEC, terminate
}
default:
hatch.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processSpline()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Spline sp;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addSpline( &sp );
return true; //found new entity or ENDSEC, terminate
}
default:
sp.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processImage()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Image img;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addImage( &img );
return true; //found new entity or ENDSEC, terminate
}
default:
img.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processDimension()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Dimension dim;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
int type = dim.type & 0x0F;
switch ( type )
{
case 0:
{
DRW_DimLinear d( dim );
iface->addDimLinear( &d );
break;
}
case 1:
{
DRW_DimAligned d( dim );
iface->addDimAlign( &d );
break;
}
case 2:
{
DRW_DimAngular d( dim );
iface->addDimAngular( &d );
break;
}
case 3:
{
DRW_DimDiametric d( dim );
iface->addDimDiametric( &d );
break;
}
case 4:
{
DRW_DimRadial d( dim );
iface->addDimRadial( &d );
break;
}
case 5:
{
DRW_DimAngular3p d( dim );
iface->addDimAngular3P( &d );
break;
}
case 6:
{
DRW_DimOrdinate d( dim );
iface->addDimOrdinate( &d );
break;
}
}
return true; //found new entity or ENDSEC, terminate
}
default:
dim.parseCode( code, reader );
break;
}
}
return true;
}
bool dxfRW::processLeader()
{
QgsDebugMsg( "Entering." );
int code;
DRW_Leader leader;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->addLeader( &leader );
return true; //found new entity or ENDSEC, terminate
}
default:
leader.parseCode( code, reader );
break;
}
}
return true;
}
/********* Objects Section *********/
bool dxfRW::processObjects()
{
QgsDebugMsg( "Entering." );
int code;
if ( !reader->readRec( &code ) )
{
return false;
}
bool next = true;
if ( code == 0 )
{
nextentity = reader->getString();
}
else
{
return false; //first record in objects is 0
}
do
{
if ( nextentity == "ENDSEC" )
{
return true; //found ENDSEC terminate
}
else if ( nextentity == "IMAGEDEF" )
{
processImageDef();
}
else
{
if ( reader->readRec( &code ) )
{
if ( code == 0 )
nextentity = reader->getString();
}
else
return false; //end of file without ENDSEC
}
}
while ( next );
return true;
}
bool dxfRW::processImageDef()
{
QgsDebugMsg( "Entering." );
int code;
DRW_ImageDef img;
while ( reader->readRec( &code ) )
{
QgsDebugMsg( QString( "code=%1" ).arg( code ) );
switch ( code )
{
case 0:
{
nextentity = reader->getString();
QgsDebugMsg( QString( "nextentity=%1" ).arg( nextentity.c_str() ) );
iface->linkImage( &img );
return true; //found new entity or ENDSEC, terminate
}
default:
img.parseCode( code, reader );
break;
}
}
return true;
}
/**
* Utility function
* convert a int to string in hex
**/
std::string dxfRW::toHexStr( int n )
{
#if defined(__APPLE__)
char buffer[9] = {'\0'};
snprintf( buffer, 9, "%X", n );
return std::string( buffer );
#else
std::ostringstream Convert;
Convert << std::uppercase << std::hex << n;
return Convert.str();
#endif
}