6435 lines
149 KiB
C++
6435 lines
149 KiB
C++
/****************************************************************************
|
|
** Copyright (C) 2001-2013 RibbonSoft, GmbH. All rights reserved.
|
|
**
|
|
** This file is part of the dxflib project.
|
|
**
|
|
** This file 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 2 of the License, or
|
|
** (at your option) any later version.
|
|
**
|
|
** Licensees holding valid dxflib Professional Edition licenses may use
|
|
** this file in accordance with the dxflib Commercial License
|
|
** Agreement provided with the Software.
|
|
**
|
|
** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
|
|
** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
**
|
|
** See http://www.ribbonsoft.com for further details.
|
|
**
|
|
** Contact info@ribbonsoft.com if any conditions of this licensing are
|
|
** not clear to you.
|
|
**
|
|
**********************************************************************/
|
|
|
|
#include "dl_dxf.h"
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
#include <cstdio>
|
|
#include <cassert>
|
|
#include <cmath>
|
|
|
|
#include "dl_attributes.h"
|
|
#include "dl_codes.h"
|
|
#include "dl_creationadapter.h"
|
|
#include "dl_writer_ascii.h"
|
|
|
|
#include "iostream"
|
|
|
|
/**
|
|
* Default constructor.
|
|
*/
|
|
DL_Dxf::DL_Dxf()
|
|
{
|
|
version = DL_VERSION_2000;
|
|
|
|
vertices = NULL;
|
|
maxVertices = 0;
|
|
vertexIndex = 0;
|
|
|
|
knots = NULL;
|
|
maxKnots = 0;
|
|
knotIndex = 0;
|
|
|
|
weights = NULL;
|
|
weightIndex = 0;
|
|
|
|
controlPoints = NULL;
|
|
maxControlPoints = 0;
|
|
controlPointIndex = 0;
|
|
|
|
fitPoints = NULL;
|
|
maxFitPoints = 0;
|
|
fitPointIndex = 0;
|
|
|
|
leaderVertices = NULL;
|
|
maxLeaderVertices = 0;
|
|
leaderVertexIndex = 0;
|
|
|
|
// Aux members, initialized here to avoid warnings in analysers
|
|
firstHatchLoop = true;
|
|
xRecordValues = false;
|
|
groupCode = 0;
|
|
currentObjectType = 0;
|
|
settingValue[0] = 0;
|
|
firstCall = true;
|
|
libVersion = 0;
|
|
appDictionaryHandle = 0;
|
|
styleHandleStd = 0;
|
|
}
|
|
|
|
|
|
/**
|
|
* Destructor.
|
|
*/
|
|
DL_Dxf::~DL_Dxf()
|
|
{
|
|
if( vertices!=NULL )
|
|
{
|
|
delete[] vertices;
|
|
}
|
|
|
|
if( knots!=NULL )
|
|
{
|
|
delete[] knots;
|
|
}
|
|
|
|
if( controlPoints!=NULL )
|
|
{
|
|
delete[] controlPoints;
|
|
}
|
|
|
|
if( fitPoints!=NULL )
|
|
{
|
|
delete[] fitPoints;
|
|
}
|
|
|
|
if( weights!=NULL )
|
|
{
|
|
delete[] weights;
|
|
}
|
|
|
|
if( leaderVertices!=NULL )
|
|
{
|
|
delete[] leaderVertices;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Reads the given file and calls the appropriate functions in
|
|
* the given creation interface for every entity found in the file.
|
|
*
|
|
* @param file Input the file pointer to read
|
|
* @param creationInterface
|
|
* Pointer to the class which takes care of the entities in the file.
|
|
*
|
|
* @retval true if fp is valid (i.e. not NULL), false otherwise.
|
|
*/
|
|
bool DL_Dxf::in( FILE* fp, DL_CreationInterface* creationInterface )
|
|
{
|
|
firstCall = true;
|
|
currentObjectType = DL_UNKNOWN;
|
|
|
|
if( fp )
|
|
{
|
|
std::locale oldLocale = std::locale::global( std::locale( "C" ) ); // use dot in numbers
|
|
|
|
while( readDxfGroups( fp, creationInterface ) )
|
|
{
|
|
}
|
|
|
|
std::locale::global( oldLocale );
|
|
fclose( fp );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @brief Reads the given file and calls the appropriate functions in
|
|
* the given creation interface for every entity found in the file.
|
|
*
|
|
* @param file Input
|
|
* Path and name of file to read
|
|
* Note: file is not very well utf8 compatible, depending on the platform.
|
|
* @param creationInterface
|
|
* Pointer to the class which takes care of the entities in the file.
|
|
*
|
|
* @retval true If \p file could be opened.
|
|
* @retval false If \p file could not be opened.
|
|
*/
|
|
bool DL_Dxf::in( const std::string& file, DL_CreationInterface* creationInterface )
|
|
{
|
|
FILE* fp;
|
|
|
|
firstCall = true;
|
|
currentObjectType = DL_UNKNOWN;
|
|
|
|
fp = fopen( file.c_str(), "rt" );
|
|
|
|
if( fp )
|
|
{
|
|
std::locale oldLocale = std::locale::global( std::locale( "C" ) ); // use dot in numbers
|
|
|
|
while( readDxfGroups( fp, creationInterface ) )
|
|
{
|
|
}
|
|
|
|
std::locale::global( oldLocale );
|
|
fclose( fp );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Reads a DXF file from an existing stream.
|
|
*
|
|
* @param stream The string stream.
|
|
* @param creationInterface
|
|
* Pointer to the class which takes care of the entities in the file.
|
|
*
|
|
* @retval true If \p file could be opened.
|
|
* @retval false If \p file could not be opened.
|
|
*/
|
|
bool DL_Dxf::in( std::stringstream& stream,
|
|
DL_CreationInterface* creationInterface )
|
|
{
|
|
if( stream.good() )
|
|
{
|
|
firstCall = true;
|
|
currentObjectType = DL_UNKNOWN;
|
|
|
|
while( readDxfGroups( stream, creationInterface ) )
|
|
{
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Reads a group couplet from a DXF file. Calls another function
|
|
* to process it.
|
|
*
|
|
* A group couplet consists of two lines that represent a single
|
|
* piece of data. An integer constant on the first line indicates
|
|
* the type of data. The value is on the next line.\n
|
|
*
|
|
* This function reads a couplet, determines the type of data, and
|
|
* passes the value to the the appropriate handler function of
|
|
* \p creationInterface.\n
|
|
*
|
|
* \p fp is advanced so that the next call to \p readDXFGroups() reads
|
|
* the next couplet in the file.
|
|
*
|
|
* @param fp Handle of input file
|
|
* @param creationInterface Handle of class which processes entities
|
|
* in the file
|
|
*
|
|
* @retval true If EOF not reached.
|
|
* @retval false If EOF reached.
|
|
*/
|
|
bool DL_Dxf::readDxfGroups( FILE* fp, DL_CreationInterface* creationInterface )
|
|
{
|
|
static int line = 1;
|
|
|
|
// Read one group of the DXF file and strip the lines:
|
|
if( DL_Dxf::getStrippedLine( groupCodeTmp, DL_DXF_MAXLINE, fp )
|
|
&& DL_Dxf::getStrippedLine( groupValue, DL_DXF_MAXLINE, fp, false ) )
|
|
{
|
|
groupCode = (unsigned int) toInt( groupCodeTmp );
|
|
|
|
creationInterface->processCodeValuePair( groupCode, groupValue );
|
|
line += 2;
|
|
processDXFGroup( creationInterface, groupCode, groupValue );
|
|
}
|
|
|
|
return !feof( fp );
|
|
}
|
|
|
|
|
|
/**
|
|
* Same as above but for stringstreams.
|
|
*/
|
|
bool DL_Dxf::readDxfGroups( std::stringstream& stream,
|
|
DL_CreationInterface* creationInterface )
|
|
{
|
|
static int line = 1;
|
|
|
|
// Read one group of the DXF file and chop the lines:
|
|
if( DL_Dxf::getStrippedLine( groupCodeTmp, DL_DXF_MAXLINE, stream )
|
|
&& DL_Dxf::getStrippedLine( groupValue, DL_DXF_MAXLINE, stream, false ) )
|
|
{
|
|
groupCode = (unsigned int) toInt( groupCodeTmp );
|
|
|
|
line += 2;
|
|
processDXFGroup( creationInterface, groupCode, groupValue );
|
|
}
|
|
|
|
return !stream.eof();
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Reads line from file & strips whitespace at start and newline
|
|
* at end.
|
|
*
|
|
* @param s Output\n
|
|
* Pointer to character array that chopped line will be returned in.
|
|
* @param size Size of \p s. (Including space for NULL.)
|
|
* @param fp Input\n
|
|
* Handle of input file.
|
|
*
|
|
* @retval true if line could be read
|
|
* @retval false if \p fp is already at end of file
|
|
*
|
|
* @todo Change function to use safer FreeBSD strl* functions
|
|
* @todo Is it a problem if line is blank (i.e., newline only)?
|
|
* Then, when function returns, (s==NULL).
|
|
*/
|
|
bool DL_Dxf::getStrippedLine( std::string& s, unsigned int size, FILE* fp, bool stripSpace )
|
|
{
|
|
if( !feof( fp ) )
|
|
{
|
|
// The whole line in the file. Includes space for NULL.
|
|
char* wholeLine = new char[size];
|
|
// Only the useful part of the line
|
|
char* line;
|
|
|
|
line = fgets( wholeLine, size, fp );
|
|
|
|
if( line!=NULL && line[0] != '\0' ) // Evaluates to fgets() retval
|
|
{
|
|
// line == wholeLine at this point.
|
|
// Both guaranteed to be NULL terminated.
|
|
|
|
// Strip leading whitespace and trailing CR/LF.
|
|
stripWhiteSpace( &line, stripSpace );
|
|
|
|
s = line;
|
|
assert( size > s.length() );
|
|
}
|
|
|
|
delete[] wholeLine; // Done with wholeLine
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
s = "";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Same as above but for stringstreams.
|
|
*/
|
|
bool DL_Dxf::getStrippedLine( std::string& s, unsigned int size,
|
|
std::stringstream& stream, bool stripSpace )
|
|
{
|
|
if( !stream.eof() )
|
|
{
|
|
// Only the useful part of the line
|
|
char* line = new char[size + 1];
|
|
char* oriLine = line;
|
|
stream.getline( line, size );
|
|
stripWhiteSpace( &line, stripSpace );
|
|
s = line;
|
|
assert( size > s.length() );
|
|
delete[] oriLine;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
s[0] = '\0';
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Strips leading whitespace and trailing Carriage Return (CR)
|
|
* and Line Feed (LF) from NULL terminated string.
|
|
*
|
|
* @param s Input and output.
|
|
* NULL terminates string.
|
|
*
|
|
* @retval true if \p s is non-NULL
|
|
* @retval false if \p s is NULL
|
|
*/
|
|
bool DL_Dxf::stripWhiteSpace( char** s, bool stripSpace )
|
|
{
|
|
// last non-NULL char:
|
|
int lastChar = strlen( *s ) - 1;
|
|
|
|
// Is last character CR or LF?
|
|
while( (lastChar >= 0)
|
|
&& ( ( (*s)[lastChar] == 10 ) || ( (*s)[lastChar] == 13 )
|
|
|| ( stripSpace && ( (*s)[lastChar] == ' ' || ( (*s)[lastChar] == '\t' ) ) ) ) )
|
|
{
|
|
(*s)[lastChar] = '\0';
|
|
lastChar--;
|
|
}
|
|
|
|
// Skip whitespace, excluding \n, at beginning of line
|
|
if( stripSpace )
|
|
{
|
|
while( (*s)[0]==' ' || (*s)[0]=='\t' )
|
|
{
|
|
++(*s);
|
|
}
|
|
}
|
|
|
|
return (*s) ? true : false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Processes a group (pair of group code and value).
|
|
*
|
|
* @param creationInterface Handle to class that creates entities and
|
|
* other CAD data from DXF group codes
|
|
*
|
|
* @param agroupCode Constant indicating the data type of the group.
|
|
* @param agroupValue The data value.
|
|
*
|
|
* @retval true if done processing current entity and new entity begun
|
|
* @retval false if not done processing current entity
|
|
*/
|
|
bool DL_Dxf::processDXFGroup( DL_CreationInterface* creationInterface,
|
|
int agroupCode, const std::string& agroupValue )
|
|
{
|
|
// printf("%d\n", agroupCode);
|
|
// printf("%s\n", agroupValue.c_str());
|
|
|
|
// Init values on first call
|
|
if( firstCall )
|
|
{
|
|
settingValue[0] = '\0';
|
|
firstCall = false;
|
|
}
|
|
|
|
// Indicates comment or dxflib version:
|
|
if( agroupCode==999 )
|
|
{
|
|
if( !agroupValue.empty() )
|
|
{
|
|
if( agroupValue.substr( 0, 6 )=="dxflib" )
|
|
{
|
|
libVersion = getLibVersion( agroupValue.substr( 7 ) );
|
|
}
|
|
|
|
addComment( creationInterface, agroupValue );
|
|
}
|
|
}
|
|
// Indicates start of new entity or variable:
|
|
else if( agroupCode==0 || agroupCode==9 )
|
|
{
|
|
// If new entity is encountered, the last one is complete.
|
|
// Prepare default attributes for next entity:
|
|
std::string layer = getStringValue( 8, "0" );
|
|
|
|
int width;
|
|
|
|
// Compatibility with qcad1:
|
|
if( hasValue( 39 ) && !hasValue( 370 ) )
|
|
{
|
|
width = getIntValue( 39, -1 );
|
|
}
|
|
// since autocad 2002:
|
|
else if( hasValue( 370 ) )
|
|
{
|
|
width = getIntValue( 370, -1 );
|
|
}
|
|
// default to BYLAYER:
|
|
else
|
|
{
|
|
width = -1;
|
|
}
|
|
|
|
int color;
|
|
color = getIntValue( 62, 256 );
|
|
int color24;
|
|
color24 = getIntValue( 420, -1 );
|
|
int handle;
|
|
handle = getInt16Value( 5, -1 );
|
|
|
|
std::string linetype = getStringValue( 6, "BYLAYER" );
|
|
|
|
attrib = DL_Attributes( layer, // layer
|
|
color, // color
|
|
color24, // 24 bit color
|
|
width, // width
|
|
linetype, // linetype
|
|
handle ); // handle
|
|
attrib.setInPaperSpace( (bool) getIntValue( 67, 0 ) );
|
|
attrib.setLinetypeScale( getRealValue( 48, 1.0 ) );
|
|
creationInterface->setAttributes( attrib );
|
|
|
|
int elevationGroupCode = 30;
|
|
|
|
if( currentObjectType==DL_ENTITY_LWPOLYLINE )
|
|
{
|
|
// see lwpolyline group codes reference
|
|
elevationGroupCode = 38;
|
|
}
|
|
else
|
|
{
|
|
// see polyline group codes reference
|
|
elevationGroupCode = 30;
|
|
}
|
|
|
|
creationInterface->setExtrusion( getRealValue( 210, 0.0 ),
|
|
getRealValue( 220, 0.0 ),
|
|
getRealValue( 230, 1.0 ),
|
|
getRealValue( elevationGroupCode, 0.0 ) );
|
|
|
|
// Add the previously parsed entity via creationInterface
|
|
switch( currentObjectType )
|
|
{
|
|
case DL_SETTING:
|
|
addSetting( creationInterface );
|
|
break;
|
|
|
|
case DL_LAYER:
|
|
addLayer( creationInterface );
|
|
break;
|
|
|
|
case DL_LINETYPE:
|
|
addLinetype( creationInterface );
|
|
break;
|
|
|
|
case DL_BLOCK:
|
|
addBlock( creationInterface );
|
|
break;
|
|
|
|
case DL_ENDBLK:
|
|
endBlock( creationInterface );
|
|
break;
|
|
|
|
case DL_STYLE:
|
|
addTextStyle( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_POINT:
|
|
addPoint( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_LINE:
|
|
addLine( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_XLINE:
|
|
addXLine( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_RAY:
|
|
addRay( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_POLYLINE:
|
|
case DL_ENTITY_LWPOLYLINE:
|
|
addPolyline( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_VERTEX:
|
|
addVertex( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_SPLINE:
|
|
addSpline( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_ARC:
|
|
addArc( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_CIRCLE:
|
|
addCircle( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_ELLIPSE:
|
|
addEllipse( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_INSERT:
|
|
addInsert( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_MTEXT:
|
|
addMText( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_TEXT:
|
|
addText( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_ARCALIGNEDTEXT:
|
|
addArcAlignedText( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_ATTRIB:
|
|
addAttribute( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_DIMENSION:
|
|
{
|
|
int type = (getIntValue( 70, 0 ) & 0x07);
|
|
|
|
switch( type )
|
|
{
|
|
case 0:
|
|
addDimLinear( creationInterface );
|
|
break;
|
|
|
|
case 1:
|
|
addDimAligned( creationInterface );
|
|
break;
|
|
|
|
case 2:
|
|
addDimAngular( creationInterface );
|
|
break;
|
|
|
|
case 3:
|
|
addDimDiametric( creationInterface );
|
|
break;
|
|
|
|
case 4:
|
|
addDimRadial( creationInterface );
|
|
break;
|
|
|
|
case 5:
|
|
addDimAngular3P( creationInterface );
|
|
break;
|
|
|
|
case 6:
|
|
addDimOrdinate( creationInterface );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case DL_ENTITY_LEADER:
|
|
addLeader( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_HATCH:
|
|
// addHatch(creationInterface);
|
|
handleHatchData( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_IMAGE:
|
|
addImage( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_IMAGEDEF:
|
|
addImageDef( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_TRACE:
|
|
addTrace( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_3DFACE:
|
|
add3dFace( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_SOLID:
|
|
addSolid( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_SEQEND:
|
|
endSequence( creationInterface );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
creationInterface->endSection();
|
|
|
|
// reset all values (they are not persistent and only this
|
|
// way we can set defaults for omitted values)
|
|
// for (int i=0; i<DL_DXF_MAXGROUPCODE; ++i) {
|
|
// values[i][0] = '\0';
|
|
// }
|
|
values.clear();
|
|
settingValue[0] = '\0';
|
|
settingKey = "";
|
|
firstHatchLoop = true;
|
|
// firstHatchEdge = true;
|
|
hatchEdge = DL_HatchEdgeData();
|
|
// xRecordHandle = "";
|
|
xRecordValues = false;
|
|
|
|
// Last DXF entity or setting has been handled
|
|
// Now determine what the next entity or setting type is
|
|
|
|
int prevEntity = currentObjectType;
|
|
|
|
// Read DXF variable:
|
|
if( agroupValue[0]=='$' )
|
|
{
|
|
currentObjectType = DL_SETTING;
|
|
settingKey = agroupValue;
|
|
}
|
|
// Read Layers:
|
|
else if( agroupValue=="LAYER" )
|
|
{
|
|
currentObjectType = DL_LAYER;
|
|
}
|
|
// Read Linetypes:
|
|
else if( agroupValue=="LTYPE" )
|
|
{
|
|
currentObjectType = DL_LINETYPE;
|
|
}
|
|
// Read Blocks:
|
|
else if( agroupValue=="BLOCK" )
|
|
{
|
|
currentObjectType = DL_BLOCK;
|
|
}
|
|
else if( agroupValue=="ENDBLK" )
|
|
{
|
|
currentObjectType = DL_ENDBLK;
|
|
}
|
|
// Read text styles:
|
|
else if( agroupValue=="STYLE" )
|
|
{
|
|
currentObjectType = DL_STYLE;
|
|
}
|
|
// Read entities:
|
|
else if( agroupValue=="POINT" )
|
|
{
|
|
currentObjectType = DL_ENTITY_POINT;
|
|
}
|
|
else if( agroupValue=="LINE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_LINE;
|
|
}
|
|
else if( agroupValue=="XLINE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_XLINE;
|
|
}
|
|
else if( agroupValue=="RAY" )
|
|
{
|
|
currentObjectType = DL_ENTITY_RAY;
|
|
}
|
|
else if( agroupValue=="POLYLINE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_POLYLINE;
|
|
}
|
|
else if( agroupValue=="LWPOLYLINE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_LWPOLYLINE;
|
|
}
|
|
else if( agroupValue=="VERTEX" )
|
|
{
|
|
currentObjectType = DL_ENTITY_VERTEX;
|
|
}
|
|
else if( agroupValue=="SPLINE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_SPLINE;
|
|
}
|
|
else if( agroupValue=="ARC" )
|
|
{
|
|
currentObjectType = DL_ENTITY_ARC;
|
|
}
|
|
else if( agroupValue=="ELLIPSE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_ELLIPSE;
|
|
}
|
|
else if( agroupValue=="CIRCLE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_CIRCLE;
|
|
}
|
|
else if( agroupValue=="INSERT" )
|
|
{
|
|
currentObjectType = DL_ENTITY_INSERT;
|
|
}
|
|
else if( agroupValue=="TEXT" )
|
|
{
|
|
currentObjectType = DL_ENTITY_TEXT;
|
|
}
|
|
else if( agroupValue=="MTEXT" )
|
|
{
|
|
currentObjectType = DL_ENTITY_MTEXT;
|
|
}
|
|
else if( agroupValue=="ARCALIGNEDTEXT" )
|
|
{
|
|
currentObjectType = DL_ENTITY_ARCALIGNEDTEXT;
|
|
}
|
|
else if( agroupValue=="ATTRIB" )
|
|
{
|
|
currentObjectType = DL_ENTITY_ATTRIB;
|
|
}
|
|
else if( agroupValue=="DIMENSION" )
|
|
{
|
|
currentObjectType = DL_ENTITY_DIMENSION;
|
|
}
|
|
else if( agroupValue=="LEADER" )
|
|
{
|
|
currentObjectType = DL_ENTITY_LEADER;
|
|
}
|
|
else if( agroupValue=="HATCH" )
|
|
{
|
|
currentObjectType = DL_ENTITY_HATCH;
|
|
}
|
|
else if( agroupValue=="IMAGE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_IMAGE;
|
|
}
|
|
else if( agroupValue=="IMAGEDEF" )
|
|
{
|
|
currentObjectType = DL_ENTITY_IMAGEDEF;
|
|
}
|
|
else if( agroupValue=="TRACE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_TRACE;
|
|
}
|
|
else if( agroupValue=="SOLID" )
|
|
{
|
|
currentObjectType = DL_ENTITY_SOLID;
|
|
}
|
|
else if( agroupValue=="3DFACE" )
|
|
{
|
|
currentObjectType = DL_ENTITY_3DFACE;
|
|
}
|
|
else if( agroupValue=="SEQEND" )
|
|
{
|
|
currentObjectType = DL_ENTITY_SEQEND;
|
|
}
|
|
else if( agroupValue=="XRECORD" )
|
|
{
|
|
currentObjectType = DL_XRECORD;
|
|
}
|
|
else if( agroupValue=="DICTIONARY" )
|
|
{
|
|
currentObjectType = DL_DICTIONARY;
|
|
}
|
|
else
|
|
{
|
|
currentObjectType = DL_UNKNOWN;
|
|
}
|
|
|
|
// end of old style POLYLINE entity
|
|
if( prevEntity==DL_ENTITY_VERTEX && currentObjectType!=DL_ENTITY_VERTEX )
|
|
{
|
|
endEntity( creationInterface );
|
|
}
|
|
|
|
// TODO: end of SPLINE entity
|
|
// if (prevEntity==DL_ENTITY_CONTROLPOINT && currentEntity!=DL_ENTITY_CONTROLPOINT) {
|
|
// endEntity(creationInterface);
|
|
// }
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
// Group code does not indicate start of new entity or setting,
|
|
// so this group must be continuation of data for the current
|
|
// one.
|
|
if( agroupCode<DL_DXF_MAXGROUPCODE )
|
|
{
|
|
bool handled = false;
|
|
|
|
switch( currentObjectType )
|
|
{
|
|
case DL_ENTITY_MTEXT:
|
|
handled = handleMTextData( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_LWPOLYLINE:
|
|
handled = handleLWPolylineData( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_SPLINE:
|
|
handled = handleSplineData( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_LEADER:
|
|
handled = handleLeaderData( creationInterface );
|
|
break;
|
|
|
|
case DL_ENTITY_HATCH:
|
|
handled = handleHatchData( creationInterface );
|
|
break;
|
|
|
|
case DL_XRECORD:
|
|
handled = handleXRecordData( creationInterface );
|
|
break;
|
|
|
|
case DL_DICTIONARY:
|
|
handled = handleDictionaryData( creationInterface );
|
|
break;
|
|
|
|
case DL_LINETYPE:
|
|
handled = handleLinetypeData( creationInterface );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// Always try to handle XData, unless we're in an XData record:
|
|
if( currentObjectType!=DL_XRECORD )
|
|
{
|
|
handled = handleXData( creationInterface );
|
|
}
|
|
|
|
if( !handled )
|
|
{
|
|
// Normal group / value pair:
|
|
values[agroupCode] = agroupValue;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a comment from the DXF file.
|
|
*/
|
|
void DL_Dxf::addComment( DL_CreationInterface* creationInterface, const std::string& comment )
|
|
{
|
|
creationInterface->addComment( comment );
|
|
}
|
|
|
|
|
|
void DL_Dxf::addDictionary( DL_CreationInterface* creationInterface )
|
|
{
|
|
creationInterface->addDictionary( DL_DictionaryData( getStringValue( 5, "" ) ) );
|
|
}
|
|
|
|
|
|
void DL_Dxf::addDictionaryEntry( DL_CreationInterface* creationInterface )
|
|
{
|
|
creationInterface->addDictionaryEntry( DL_DictionaryEntryData( getStringValue( 3, "" ),
|
|
getStringValue( 350, "" ) ) );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a variable from the DXF file.
|
|
*/
|
|
void DL_Dxf::addSetting( DL_CreationInterface* creationInterface )
|
|
{
|
|
int c = -1;
|
|
std::map<int, std::string>::iterator it = values.begin();
|
|
|
|
if( it!=values.end() )
|
|
{
|
|
c = it->first;
|
|
}
|
|
|
|
// for (int i=0; i<=380; ++i) {
|
|
// if (values[i][0]!='\0') {
|
|
// c = i;
|
|
// break;
|
|
// }
|
|
// }
|
|
|
|
// string
|
|
if( c>=0 && c<=9 )
|
|
{
|
|
creationInterface->setVariableString( settingKey, values[c], c );
|
|
#ifdef DL_COMPAT
|
|
// backwards compatibility:
|
|
creationInterface->setVariableString( settingKey.c_str(), values[c].c_str(), c );
|
|
#endif
|
|
}
|
|
// vector
|
|
else if( c>=10 && c<=39 )
|
|
{
|
|
if( c==10 )
|
|
{
|
|
creationInterface->setVariableVector(
|
|
settingKey,
|
|
getRealValue( c, 0.0 ),
|
|
getRealValue( c + 10, 0.0 ),
|
|
getRealValue( c + 20, 0.0 ),
|
|
c );
|
|
}
|
|
}
|
|
// double
|
|
else if( c>=40 && c<=59 )
|
|
{
|
|
creationInterface->setVariableDouble( settingKey, getRealValue( c, 0.0 ), c );
|
|
}
|
|
// int
|
|
else if( c>=60 && c<=99 )
|
|
{
|
|
creationInterface->setVariableInt( settingKey, getIntValue( c, 0 ), c );
|
|
}
|
|
// misc
|
|
else if( c>=0 )
|
|
{
|
|
creationInterface->setVariableString( settingKey, getStringValue( c, "" ), c );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a layer that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addLayer( DL_CreationInterface* creationInterface )
|
|
{
|
|
// correct some invalid attributes for layers:
|
|
attrib = creationInterface->getAttributes();
|
|
|
|
if( attrib.getColor()==256 || attrib.getColor()==0 )
|
|
{
|
|
attrib.setColor( 7 );
|
|
}
|
|
|
|
if( attrib.getWidth()<0 )
|
|
{
|
|
attrib.setWidth( 1 );
|
|
}
|
|
|
|
std::string linetype = attrib.getLinetype();
|
|
std::transform( linetype.begin(), linetype.end(), linetype.begin(), ::toupper );
|
|
|
|
if( linetype=="BYLAYER" || linetype=="BYBLOCK" )
|
|
{
|
|
attrib.setLinetype( "CONTINUOUS" );
|
|
}
|
|
|
|
// add layer
|
|
std::string name = getStringValue( 2, "" );
|
|
|
|
if( name.length()==0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
creationInterface->addLayer( DL_LayerData( name, getIntValue( 70, 0 ) ) );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a linetype that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addLinetype( DL_CreationInterface* creationInterface )
|
|
{
|
|
std::string name = getStringValue( 2, "" );
|
|
|
|
if( name.length()==0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
int numDashes = getIntValue( 73, 0 );
|
|
// double dashes[numDashes];
|
|
|
|
DL_LinetypeData d(
|
|
// name:
|
|
name,
|
|
// description:
|
|
getStringValue( 3, "" ),
|
|
// flags
|
|
getIntValue( 70, 0 ),
|
|
// number of dashes:
|
|
numDashes,
|
|
// pattern length:
|
|
getRealValue( 40, 0.0 )
|
|
// pattern:
|
|
// dashes
|
|
);
|
|
|
|
if( name != "By Layer" && name != "By Block" && name != "BYLAYER" && name != "BYBLOCK" )
|
|
{
|
|
creationInterface->addLinetype( d );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles all dashes in linetype pattern.
|
|
*/
|
|
bool DL_Dxf::handleLinetypeData( DL_CreationInterface* creationInterface )
|
|
{
|
|
if( groupCode == 49 )
|
|
{
|
|
creationInterface->addLinetypeDash( toReal( groupValue ) );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a block that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addBlock( DL_CreationInterface* creationInterface )
|
|
{
|
|
std::string name = getStringValue( 2, "" );
|
|
|
|
if( name.length()==0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DL_BlockData d(
|
|
// Name:
|
|
name,
|
|
// flags:
|
|
getIntValue( 70, 0 ),
|
|
// base point:
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ) );
|
|
|
|
creationInterface->addBlock( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Ends a block that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::endBlock( DL_CreationInterface* creationInterface )
|
|
{
|
|
creationInterface->endBlock();
|
|
}
|
|
|
|
|
|
void DL_Dxf::addTextStyle( DL_CreationInterface* creationInterface )
|
|
{
|
|
std::string name = getStringValue( 2, "" );
|
|
|
|
if( name.length()==0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DL_StyleData d(
|
|
// name:
|
|
name,
|
|
// flags
|
|
getIntValue( 70, 0 ),
|
|
// fixed text heigth:
|
|
getRealValue( 40, 0.0 ),
|
|
// width factor:
|
|
getRealValue( 41, 0.0 ),
|
|
// oblique angle:
|
|
getRealValue( 50, 0.0 ),
|
|
// text generation flags:
|
|
getIntValue( 71, 0 ),
|
|
// last height used:
|
|
getRealValue( 42, 0.0 ),
|
|
// primart font file:
|
|
getStringValue( 3, "" ),
|
|
// big font file:
|
|
getStringValue( 4, "" )
|
|
);
|
|
creationInterface->addTextStyle( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a point entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addPoint( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_PointData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ) );
|
|
|
|
creationInterface->addPoint( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a line entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addLine( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_LineData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ) );
|
|
|
|
creationInterface->addLine( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an xline entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addXLine( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_XLineData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ) );
|
|
|
|
creationInterface->addXLine( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a ray entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addRay( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_RayData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ) );
|
|
|
|
creationInterface->addRay( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a polyline entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addPolyline( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_PolylineData pd( maxVertices, getIntValue( 71, 0 ), getIntValue( 72, 0 ), getIntValue( 70,
|
|
0 ), getRealValue( 38, 0 ) );
|
|
|
|
creationInterface->addPolyline( pd );
|
|
|
|
maxVertices = std::min( maxVertices, vertexIndex + 1 );
|
|
|
|
if( currentObjectType==DL_ENTITY_LWPOLYLINE )
|
|
{
|
|
for( int i = 0; i<maxVertices; i++ )
|
|
{
|
|
DL_VertexData d( vertices[i * 4],
|
|
vertices[i * 4 + 1],
|
|
vertices[i * 4 + 2],
|
|
vertices[i * 4 + 3] );
|
|
|
|
creationInterface->addVertex( d );
|
|
}
|
|
|
|
creationInterface->endEntity();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a polyline vertex entity that was read from the file
|
|
* via the creation interface.
|
|
*/
|
|
void DL_Dxf::addVertex( DL_CreationInterface* creationInterface )
|
|
{
|
|
// vertex defines a face of the mesh if its vertex flags group has the
|
|
// 128 bit set but not the 64 bit. 10, 20, 30 are irrelevant and set to
|
|
// 0 in this case
|
|
if( (getIntValue( 70, 0 ) & 128) && !(getIntValue( 70, 0 ) & 64) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DL_VertexData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 42, 0.0 ) );
|
|
|
|
creationInterface->addVertex( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a spline entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addSpline( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_SplineData sd( getIntValue( 71, 3 ),
|
|
maxKnots,
|
|
maxControlPoints,
|
|
maxFitPoints,
|
|
getIntValue( 70, 4 ) );
|
|
|
|
sd.tangentStartX = getRealValue( 12, 0.0 );
|
|
sd.tangentStartY = getRealValue( 22, 0.0 );
|
|
sd.tangentStartZ = getRealValue( 32, 0.0 );
|
|
sd.tangentEndX = getRealValue( 13, 0.0 );
|
|
sd.tangentEndY = getRealValue( 23, 0.0 );
|
|
sd.tangentEndZ = getRealValue( 33, 0.0 );
|
|
|
|
creationInterface->addSpline( sd );
|
|
|
|
int i;
|
|
|
|
for( i = 0; i<maxControlPoints; i++ )
|
|
{
|
|
DL_ControlPointData d( controlPoints[i * 3],
|
|
controlPoints[i * 3 + 1],
|
|
controlPoints[i * 3 + 2],
|
|
weights[i] );
|
|
|
|
creationInterface->addControlPoint( d );
|
|
}
|
|
|
|
for( i = 0; i<maxFitPoints; i++ )
|
|
{
|
|
DL_FitPointData d( fitPoints[i * 3],
|
|
fitPoints[i * 3 + 1],
|
|
fitPoints[i * 3 + 2] );
|
|
|
|
creationInterface->addFitPoint( d );
|
|
}
|
|
|
|
for( i = 0; i<maxKnots; i++ )
|
|
{
|
|
DL_KnotData k( knots[i] );
|
|
|
|
creationInterface->addKnot( k );
|
|
}
|
|
|
|
creationInterface->endEntity();
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an arc entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addArc( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_ArcData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 40, 0.0 ),
|
|
getRealValue( 50, 0.0 ),
|
|
getRealValue( 51, 0.0 ) );
|
|
|
|
creationInterface->addArc( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a circle entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addCircle( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_CircleData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 40, 0.0 ) );
|
|
|
|
creationInterface->addCircle( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an ellipse entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addEllipse( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_EllipseData d( getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ),
|
|
getRealValue( 40, 1.0 ),
|
|
getRealValue( 41, 0.0 ),
|
|
getRealValue( 42, 2 * M_PI ) );
|
|
|
|
creationInterface->addEllipse( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an insert entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addInsert( DL_CreationInterface* creationInterface )
|
|
{
|
|
// printf("addInsert\n");
|
|
// printf("code 50: %s\n", values[50]);
|
|
// printf("code 50 length: %d\n", strlen(values[50]));
|
|
// printf("code 50:\n");
|
|
// getRealValue(50, 0.0);
|
|
|
|
std::string name = getStringValue( 2, "" );
|
|
|
|
if( name.length()==0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DL_InsertData d( name,
|
|
// insertion point
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// scale:
|
|
getRealValue( 41, 1.0 ),
|
|
getRealValue( 42, 1.0 ),
|
|
getRealValue( 43, 1.0 ),
|
|
// angle (deg):
|
|
getRealValue( 50, 0.0 ),
|
|
// cols / rows:
|
|
getIntValue( 70, 1 ),
|
|
getIntValue( 71, 1 ),
|
|
// spacing:
|
|
getRealValue( 44, 0.0 ),
|
|
getRealValue( 45, 0.0 ) );
|
|
|
|
creationInterface->addInsert( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a trace entity (4 edge closed polyline) that was read from the file via the creation interface.
|
|
*
|
|
* @author AHM
|
|
*/
|
|
void DL_Dxf::addTrace( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_TraceData td;
|
|
|
|
for( int k = 0; k < 4; k++ )
|
|
{
|
|
td.x[k] = getRealValue( 10 + k, 0.0 );
|
|
td.y[k] = getRealValue( 20 + k, 0.0 );
|
|
td.z[k] = getRealValue( 30 + k, 0.0 );
|
|
}
|
|
|
|
creationInterface->addTrace( td );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a 3dface entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::add3dFace( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_3dFaceData td;
|
|
|
|
for( int k = 0; k < 4; k++ )
|
|
{
|
|
td.x[k] = getRealValue( 10 + k, 0.0 );
|
|
td.y[k] = getRealValue( 20 + k, 0.0 );
|
|
td.z[k] = getRealValue( 30 + k, 0.0 );
|
|
}
|
|
|
|
creationInterface->add3dFace( td );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a solid entity (filled trace) that was read from the file via the creation interface.
|
|
*
|
|
* @author AHM
|
|
*/
|
|
void DL_Dxf::addSolid( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_SolidData sd;
|
|
|
|
for( int k = 0; k < 4; k++ )
|
|
{
|
|
sd.x[k] = getRealValue( 10 + k, 0.0 );
|
|
sd.y[k] = getRealValue( 20 + k, 0.0 );
|
|
sd.z[k] = getRealValue( 30 + k, 0.0 );
|
|
}
|
|
|
|
creationInterface->addSolid( sd );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an MText entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addMText( DL_CreationInterface* creationInterface )
|
|
{
|
|
double angle = 0.0;
|
|
|
|
if( hasValue( 50 ) )
|
|
{
|
|
if( libVersion<=0x02000200 )
|
|
{
|
|
// wrong but compatible with dxflib <=2.0.2.0 (angle stored in rad):
|
|
angle = getRealValue( 50, 0.0 );
|
|
}
|
|
else
|
|
{
|
|
angle = (getRealValue( 50, 0.0 ) * 2 * M_PI) / 360.0;
|
|
}
|
|
}
|
|
else if( hasValue( 11 ) && hasValue( 21 ) )
|
|
{
|
|
double x = getRealValue( 11, 0.0 );
|
|
double y = getRealValue( 21, 0.0 );
|
|
|
|
if( fabs( x )<1.0e-6 )
|
|
{
|
|
if( y>0.0 )
|
|
{
|
|
angle = M_PI / 2.0;
|
|
}
|
|
else
|
|
{
|
|
angle = M_PI / 2.0 * 3.0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
angle = atan( y / x );
|
|
}
|
|
}
|
|
|
|
DL_MTextData d(
|
|
// insertion point
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// X direction vector
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ),
|
|
// height
|
|
getRealValue( 40, 2.5 ),
|
|
// width
|
|
getRealValue( 41, 0.0 ),
|
|
// attachment point
|
|
getIntValue( 71, 1 ),
|
|
// drawing direction
|
|
getIntValue( 72, 1 ),
|
|
// line spacing style
|
|
getIntValue( 73, 1 ),
|
|
// line spacing factor
|
|
getRealValue( 44, 1.0 ),
|
|
// text
|
|
getStringValue( 1, "" ),
|
|
// style
|
|
getStringValue( 7, "" ),
|
|
// angle
|
|
angle );
|
|
creationInterface->addMText( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles all XRecord data.
|
|
*/
|
|
bool DL_Dxf::handleXRecordData( DL_CreationInterface* creationInterface )
|
|
{
|
|
if( groupCode==105 )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if( groupCode==5 )
|
|
{
|
|
creationInterface->addXRecord( groupValue );
|
|
return true;
|
|
}
|
|
|
|
if( groupCode==280 )
|
|
{
|
|
xRecordValues = true;
|
|
return true;
|
|
}
|
|
|
|
if( !xRecordValues )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// string:
|
|
if( groupCode<=9
|
|
|| groupCode==100 || groupCode==102 || groupCode==105
|
|
|| (groupCode>=300 && groupCode<=369)
|
|
|| (groupCode>=1000 && groupCode<=1009) )
|
|
{
|
|
creationInterface->addXRecordString( groupCode, groupValue );
|
|
return true;
|
|
}
|
|
// int:
|
|
else if( (groupCode>=60 && groupCode<=99) || (groupCode>=160 && groupCode<=179)
|
|
|| (groupCode>=270 && groupCode<=289) )
|
|
{
|
|
creationInterface->addXRecordInt( groupCode, toInt( groupValue ) );
|
|
return true;
|
|
}
|
|
// bool:
|
|
else if( groupCode>=290 && groupCode<=299 )
|
|
{
|
|
creationInterface->addXRecordBool( groupCode, toBool( groupValue ) );
|
|
return true;
|
|
}
|
|
// double:
|
|
else if( (groupCode>=10 && groupCode<=59) || (groupCode>=110 && groupCode<=149)
|
|
|| (groupCode>=210 && groupCode<=239) )
|
|
{
|
|
creationInterface->addXRecordReal( groupCode, toReal( groupValue ) );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles all dictionary data.
|
|
*/
|
|
bool DL_Dxf::handleDictionaryData( DL_CreationInterface* creationInterface )
|
|
{
|
|
if( groupCode==3 )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( groupCode==5 )
|
|
{
|
|
creationInterface->addDictionary( DL_DictionaryData( groupValue ) );
|
|
return true;
|
|
}
|
|
|
|
if( groupCode==350 )
|
|
{
|
|
creationInterface->addDictionaryEntry( DL_DictionaryEntryData( getStringValue( 3, "" ),
|
|
groupValue ) );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles XData for all object types.
|
|
*/
|
|
bool DL_Dxf::handleXData( DL_CreationInterface* creationInterface )
|
|
{
|
|
if( groupCode==1001 )
|
|
{
|
|
creationInterface->addXDataApp( groupValue );
|
|
return true;
|
|
}
|
|
else if( groupCode>=1000 && groupCode<=1009 )
|
|
{
|
|
creationInterface->addXDataString( groupCode, groupValue );
|
|
return true;
|
|
}
|
|
else if( groupCode>=1010 && groupCode<=1059 )
|
|
{
|
|
creationInterface->addXDataReal( groupCode, toReal( groupValue ) );
|
|
return true;
|
|
}
|
|
else if( groupCode>=1060 && groupCode<=1070 )
|
|
{
|
|
creationInterface->addXDataInt( groupCode, toInt( groupValue ) );
|
|
return true;
|
|
}
|
|
else if( groupCode==1071 )
|
|
{
|
|
creationInterface->addXDataInt( groupCode, toInt( groupValue ) );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles additional MText data.
|
|
*/
|
|
bool DL_Dxf::handleMTextData( DL_CreationInterface* creationInterface )
|
|
{
|
|
// Special handling of text chunks for MTEXT entities:
|
|
if( groupCode==3 )
|
|
{
|
|
creationInterface->addMTextChunk( groupValue );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles additional polyline data.
|
|
*/
|
|
bool DL_Dxf::handleLWPolylineData( DL_CreationInterface* /*creationInterface*/ )
|
|
{
|
|
// Allocate LWPolyline vertices (group code 90):
|
|
if( groupCode==90 )
|
|
{
|
|
maxVertices = toInt( groupValue );
|
|
|
|
if( maxVertices>0 )
|
|
{
|
|
if( vertices!=NULL )
|
|
{
|
|
delete[] vertices;
|
|
}
|
|
|
|
vertices = new double[4 * maxVertices];
|
|
|
|
for( int i = 0; i<maxVertices; ++i )
|
|
{
|
|
vertices[i * 4] = 0.0;
|
|
vertices[i * 4 + 1] = 0.0;
|
|
vertices[i * 4 + 2] = 0.0;
|
|
vertices[i * 4 + 3] = 0.0;
|
|
}
|
|
}
|
|
|
|
vertexIndex = -1;
|
|
return true;
|
|
}
|
|
// Process LWPolylines vertices (group codes 10/20/30/42):
|
|
else if( groupCode==10 || groupCode==20
|
|
|| groupCode==30 || groupCode==42 )
|
|
{
|
|
if( vertexIndex<maxVertices - 1 && groupCode==10 )
|
|
{
|
|
vertexIndex++;
|
|
}
|
|
|
|
if( groupCode<=30 )
|
|
{
|
|
if( vertexIndex>=0 && vertexIndex<maxVertices )
|
|
{
|
|
vertices[4 * vertexIndex + (groupCode / 10 - 1)] = toReal( groupValue );
|
|
}
|
|
}
|
|
else if( groupCode==42 && vertexIndex<maxVertices )
|
|
{
|
|
vertices[4 * vertexIndex + 3] = toReal( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles additional spline data.
|
|
*/
|
|
bool DL_Dxf::handleSplineData( DL_CreationInterface* /*creationInterface*/ )
|
|
{
|
|
// Allocate Spline knots (group code 72):
|
|
if( groupCode==72 )
|
|
{
|
|
maxKnots = toInt( groupValue );
|
|
|
|
if( maxKnots>0 )
|
|
{
|
|
if( knots!=NULL )
|
|
{
|
|
delete[] knots;
|
|
}
|
|
|
|
knots = new double[maxKnots];
|
|
|
|
for( int i = 0; i<maxKnots; ++i )
|
|
{
|
|
knots[i] = 0.0;
|
|
}
|
|
}
|
|
|
|
knotIndex = -1;
|
|
return true;
|
|
}
|
|
// Allocate Spline control points / weights (group code 73):
|
|
else if( groupCode==73 )
|
|
{
|
|
maxControlPoints = toInt( groupValue );
|
|
|
|
if( maxControlPoints>0 )
|
|
{
|
|
if( controlPoints!=NULL )
|
|
{
|
|
delete[] controlPoints;
|
|
}
|
|
|
|
if( weights!=NULL )
|
|
{
|
|
delete[] weights;
|
|
}
|
|
|
|
controlPoints = new double[3 * maxControlPoints];
|
|
weights = new double[maxControlPoints];
|
|
|
|
for( int i = 0; i<maxControlPoints; ++i )
|
|
{
|
|
controlPoints[i * 3] = 0.0;
|
|
controlPoints[i * 3 + 1] = 0.0;
|
|
controlPoints[i * 3 + 2] = 0.0;
|
|
weights[i] = 1.0;
|
|
}
|
|
}
|
|
|
|
controlPointIndex = -1;
|
|
weightIndex = -1;
|
|
return true;
|
|
}
|
|
// Allocate Spline fit points (group code 74):
|
|
else if( groupCode==74 )
|
|
{
|
|
maxFitPoints = toInt( groupValue );
|
|
|
|
if( maxFitPoints>0 )
|
|
{
|
|
if( fitPoints!=NULL )
|
|
{
|
|
delete[] fitPoints;
|
|
}
|
|
|
|
fitPoints = new double[3 * maxFitPoints];
|
|
|
|
for( int i = 0; i<maxFitPoints; ++i )
|
|
{
|
|
fitPoints[i * 3] = 0.0;
|
|
fitPoints[i * 3 + 1] = 0.0;
|
|
fitPoints[i * 3 + 2] = 0.0;
|
|
}
|
|
}
|
|
|
|
fitPointIndex = -1;
|
|
return true;
|
|
}
|
|
// Process spline knot vertices (group code 40):
|
|
else if( groupCode==40 )
|
|
{
|
|
if( knotIndex<maxKnots - 1 )
|
|
{
|
|
knotIndex++;
|
|
knots[knotIndex] = toReal( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
// Process spline control points (group codes 10/20/30):
|
|
else if( groupCode==10 || groupCode==20
|
|
|| groupCode==30 )
|
|
{
|
|
if( controlPointIndex<maxControlPoints - 1 && groupCode==10 )
|
|
{
|
|
controlPointIndex++;
|
|
}
|
|
|
|
if( controlPointIndex>=0 && controlPointIndex<maxControlPoints )
|
|
{
|
|
controlPoints[3 * controlPointIndex + (groupCode / 10 - 1)] = toReal( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
// Process spline fit points (group codes 11/21/31):
|
|
else if( groupCode==11 || groupCode==21 || groupCode==31 )
|
|
{
|
|
if( fitPointIndex<maxFitPoints - 1 && groupCode==11 )
|
|
{
|
|
fitPointIndex++;
|
|
}
|
|
|
|
if( fitPointIndex>=0 && fitPointIndex<maxFitPoints )
|
|
{
|
|
fitPoints[3 * fitPointIndex + ( (groupCode - 1) / 10 - 1 )] = toReal( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
// Process spline weights (group code 41)
|
|
else if( groupCode==41 )
|
|
{
|
|
if( weightIndex<maxControlPoints - 1 )
|
|
{
|
|
weightIndex++;
|
|
}
|
|
|
|
if( weightIndex>=0 && weightIndex<maxControlPoints )
|
|
{
|
|
weights[weightIndex] = toReal( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles additional leader data.
|
|
*/
|
|
bool DL_Dxf::handleLeaderData( DL_CreationInterface* /*creationInterface*/ )
|
|
{
|
|
// Allocate Leader vertices (group code 76):
|
|
if( groupCode==76 )
|
|
{
|
|
maxLeaderVertices = toInt( groupValue );
|
|
|
|
if( maxLeaderVertices>0 )
|
|
{
|
|
if( leaderVertices!=NULL )
|
|
{
|
|
delete[] leaderVertices;
|
|
}
|
|
|
|
leaderVertices = new double[3 * maxLeaderVertices];
|
|
|
|
for( int i = 0; i<maxLeaderVertices; ++i )
|
|
{
|
|
leaderVertices[i * 3] = 0.0;
|
|
leaderVertices[i * 3 + 1] = 0.0;
|
|
leaderVertices[i * 3 + 2] = 0.0;
|
|
}
|
|
}
|
|
|
|
leaderVertexIndex = -1;
|
|
return true;
|
|
}
|
|
// Process Leader vertices (group codes 10/20/30):
|
|
else if( groupCode==10 || groupCode==20 || groupCode==30 )
|
|
{
|
|
if( leaderVertexIndex<maxLeaderVertices - 1 && groupCode==10 )
|
|
{
|
|
leaderVertexIndex++;
|
|
}
|
|
|
|
if( groupCode<=30 )
|
|
{
|
|
if( leaderVertexIndex>=0
|
|
&& leaderVertexIndex<maxLeaderVertices )
|
|
{
|
|
leaderVertices[3 * leaderVertexIndex + (groupCode / 10 - 1)]
|
|
= toReal( groupValue );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an text entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addText( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_TextData d(
|
|
// insertion point
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// alignment point
|
|
getRealValue( 11, DL_NANDOUBLE ),
|
|
getRealValue( 21, DL_NANDOUBLE ),
|
|
getRealValue( 31, DL_NANDOUBLE ),
|
|
// height
|
|
getRealValue( 40, 2.5 ),
|
|
// x scale
|
|
getRealValue( 41, 1.0 ),
|
|
// generation flags
|
|
getIntValue( 71, 0 ),
|
|
// h just
|
|
getIntValue( 72, 0 ),
|
|
// v just
|
|
getIntValue( 73, 0 ),
|
|
// text
|
|
getStringValue( 1, "" ),
|
|
// style
|
|
getStringValue( 7, "" ),
|
|
// angle
|
|
(getRealValue( 50, 0.0 ) * 2 * M_PI) / 360.0 );
|
|
|
|
creationInterface->addText( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an arc aligned text entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addArcAlignedText( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_ArcAlignedTextData d;
|
|
|
|
d.text = getStringValue( 1, "" );
|
|
d.font = getStringValue( 2, "" );
|
|
d.style = getStringValue( 7, "" );
|
|
d.cx = getRealValue( 10, 0.0 );
|
|
d.cy = getRealValue( 20, 0.0 );
|
|
d.cz = getRealValue( 30, 0.0 );
|
|
d.radius = getRealValue( 40, 0.0 );
|
|
d.xScaleFactor = getRealValue( 41, 0.0 );
|
|
d.height = getRealValue( 42, 0.0 );
|
|
d.spacing = getRealValue( 43, 0.0 );
|
|
d.offset = getRealValue( 44, 0.0 );
|
|
d.rightOffset = getRealValue( 45, 0.0 );
|
|
d.leftOffset = getRealValue( 46, 0.0 );
|
|
d.startAngle = getRealValue( 50, 0.0 );
|
|
d.endAngle = getRealValue( 51, 0.0 );
|
|
d.reversedCharacterOrder = getIntValue( 70, 0 );
|
|
d.direction = getIntValue( 71, 0 );
|
|
d.alignment = getIntValue( 72, 0 );
|
|
d.side = getIntValue( 73, 0 );
|
|
d.bold = getIntValue( 74, 0 );
|
|
d.italic = getIntValue( 75, 0 );
|
|
d.underline = getIntValue( 76, 0 );
|
|
d.characerSet = getIntValue( 77, 0 );
|
|
d.pitch = getIntValue( 78, 0 );
|
|
d.shxFont = getIntValue( 79, 0 );
|
|
d.wizard = getIntValue( 280, 0 );
|
|
d.arcHandle = getIntValue( 330, 0 );
|
|
|
|
creationInterface->addArcAlignedText( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an attrib entity that was read from the file via the creation interface.
|
|
* @todo add attrib instead of normal text
|
|
*/
|
|
void DL_Dxf::addAttribute( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_AttributeData d(
|
|
// insertion point
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// alignment point
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ),
|
|
// height
|
|
getRealValue( 40, 2.5 ),
|
|
// x scale
|
|
getRealValue( 41, 1.0 ),
|
|
// generation flags
|
|
getIntValue( 71, 0 ),
|
|
// h just
|
|
getIntValue( 72, 0 ),
|
|
// v just
|
|
getIntValue( 74, 0 ),
|
|
// tag
|
|
getStringValue( 2, "" ),
|
|
// text
|
|
getStringValue( 1, "" ),
|
|
// style
|
|
getStringValue( 7, "" ),
|
|
// angle
|
|
(getRealValue( 50, 0.0 ) * 2 * M_PI) / 360.0 );
|
|
|
|
creationInterface->addAttribute( d );
|
|
}
|
|
|
|
|
|
/**
|
|
* @return dimension data from current values.
|
|
*/
|
|
DL_DimensionData DL_Dxf::getDimData()
|
|
{
|
|
// generic dimension data:
|
|
return DL_DimensionData(
|
|
// def point
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// text middle point
|
|
getRealValue( 11, 0.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ),
|
|
// type
|
|
getIntValue( 70, 0 ),
|
|
// attachment point
|
|
getIntValue( 71, 5 ),
|
|
// line sp. style
|
|
getIntValue( 72, 1 ),
|
|
// line sp. factor
|
|
getRealValue( 41, 1.0 ),
|
|
// text
|
|
getStringValue( 1, "" ),
|
|
// style
|
|
getStringValue( 3, "" ),
|
|
// angle
|
|
getRealValue( 53, 0.0 ) );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a linear dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimLinear( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// horizontal / vertical / rotated dimension:
|
|
DL_DimLinearData dl(
|
|
// definition point 1
|
|
getRealValue( 13, 0.0 ),
|
|
getRealValue( 23, 0.0 ),
|
|
getRealValue( 33, 0.0 ),
|
|
// definition point 2
|
|
getRealValue( 14, 0.0 ),
|
|
getRealValue( 24, 0.0 ),
|
|
getRealValue( 34, 0.0 ),
|
|
// angle
|
|
getRealValue( 50, 0.0 ),
|
|
// oblique
|
|
getRealValue( 52, 0.0 ) );
|
|
|
|
creationInterface->addDimLinear( d, dl );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an aligned dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimAligned( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// aligned dimension:
|
|
DL_DimAlignedData da(
|
|
// extension point 1
|
|
getRealValue( 13, 0.0 ),
|
|
getRealValue( 23, 0.0 ),
|
|
getRealValue( 33, 0.0 ),
|
|
// extension point 2
|
|
getRealValue( 14, 0.0 ),
|
|
getRealValue( 24, 0.0 ),
|
|
getRealValue( 34, 0.0 ) );
|
|
|
|
creationInterface->addDimAlign( d, da );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a radial dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimRadial( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
DL_DimRadialData dr(
|
|
// definition point
|
|
getRealValue( 15, 0.0 ),
|
|
getRealValue( 25, 0.0 ),
|
|
getRealValue( 35, 0.0 ),
|
|
// leader length:
|
|
getRealValue( 40, 0.0 ) );
|
|
|
|
creationInterface->addDimRadial( d, dr );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a diametric dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimDiametric( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// diametric dimension:
|
|
DL_DimDiametricData dr(
|
|
// definition point
|
|
getRealValue( 15, 0.0 ),
|
|
getRealValue( 25, 0.0 ),
|
|
getRealValue( 35, 0.0 ),
|
|
// leader length:
|
|
getRealValue( 40, 0.0 ) );
|
|
|
|
creationInterface->addDimDiametric( d, dr );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an angular dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimAngular( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// angular dimension:
|
|
DL_DimAngularData da(
|
|
// definition point 1
|
|
getRealValue( 13, 0.0 ),
|
|
getRealValue( 23, 0.0 ),
|
|
getRealValue( 33, 0.0 ),
|
|
// definition point 2
|
|
getRealValue( 14, 0.0 ),
|
|
getRealValue( 24, 0.0 ),
|
|
getRealValue( 34, 0.0 ),
|
|
// definition point 3
|
|
getRealValue( 15, 0.0 ),
|
|
getRealValue( 25, 0.0 ),
|
|
getRealValue( 35, 0.0 ),
|
|
// definition point 4
|
|
getRealValue( 16, 0.0 ),
|
|
getRealValue( 26, 0.0 ),
|
|
getRealValue( 36, 0.0 ) );
|
|
|
|
creationInterface->addDimAngular( d, da );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an angular dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimAngular3P( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// angular dimension (3P):
|
|
DL_DimAngular3PData da(
|
|
// definition point 1
|
|
getRealValue( 13, 0.0 ),
|
|
getRealValue( 23, 0.0 ),
|
|
getRealValue( 33, 0.0 ),
|
|
// definition point 2
|
|
getRealValue( 14, 0.0 ),
|
|
getRealValue( 24, 0.0 ),
|
|
getRealValue( 34, 0.0 ),
|
|
// definition point 3
|
|
getRealValue( 15, 0.0 ),
|
|
getRealValue( 25, 0.0 ),
|
|
getRealValue( 35, 0.0 ) );
|
|
|
|
creationInterface->addDimAngular3P( d, da );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an ordinate dimension entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addDimOrdinate( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_DimensionData d = getDimData();
|
|
|
|
// ordinate dimension:
|
|
DL_DimOrdinateData dl(
|
|
// definition point 1
|
|
getRealValue( 13, 0.0 ),
|
|
getRealValue( 23, 0.0 ),
|
|
getRealValue( 33, 0.0 ),
|
|
// definition point 2
|
|
getRealValue( 14, 0.0 ),
|
|
getRealValue( 24, 0.0 ),
|
|
getRealValue( 34, 0.0 ),
|
|
(getIntValue( 70, 0 ) & 64)==64 // true: X-type, false: Y-type
|
|
);
|
|
|
|
creationInterface->addDimOrdinate( d, dl );
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a leader entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addLeader( DL_CreationInterface* creationInterface )
|
|
{
|
|
// leader (arrow)
|
|
DL_LeaderData le(
|
|
// arrow head flag
|
|
getIntValue( 71, 1 ),
|
|
// leader path type
|
|
getIntValue( 72, 0 ),
|
|
// Leader creation flag
|
|
getIntValue( 73, 3 ),
|
|
// Hookline direction flag
|
|
getIntValue( 74, 1 ),
|
|
// Hookline flag
|
|
getIntValue( 75, 0 ),
|
|
// Text annotation height
|
|
getRealValue( 40, 1.0 ),
|
|
// Text annotation width
|
|
getRealValue( 41, 1.0 ),
|
|
// Number of vertices in leader
|
|
getIntValue( 76, 0 )
|
|
);
|
|
|
|
creationInterface->addLeader( le );
|
|
|
|
for( int i = 0; i<maxLeaderVertices; i++ )
|
|
{
|
|
DL_LeaderVertexData d( leaderVertices[i * 3],
|
|
leaderVertices[i * 3 + 1],
|
|
leaderVertices[i * 3 + 2] );
|
|
|
|
creationInterface->addLeaderVertex( d );
|
|
}
|
|
|
|
creationInterface->endEntity();
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds a hatch entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addHatch( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_HatchData hd( getIntValue( 91, 1 ),
|
|
getIntValue( 70, 0 ),
|
|
getRealValue( 41, 1.0 ),
|
|
getRealValue( 52, 0.0 ),
|
|
getStringValue( 2, "" ) );
|
|
|
|
creationInterface->addHatch( hd );
|
|
|
|
for( unsigned int i = 0; i<hatchEdges.size(); i++ )
|
|
{
|
|
creationInterface->addHatchLoop( DL_HatchLoopData( hatchEdges[i].size() ) );
|
|
|
|
for( unsigned int k = 0; k<hatchEdges[i].size(); k++ )
|
|
{
|
|
creationInterface->addHatchEdge( DL_HatchEdgeData( hatchEdges[i][k] ) );
|
|
}
|
|
}
|
|
|
|
creationInterface->endEntity();
|
|
}
|
|
|
|
|
|
void DL_Dxf::addHatchLoop()
|
|
{
|
|
addHatchEdge();
|
|
hatchEdges.push_back( std::vector<DL_HatchEdgeData>() );
|
|
}
|
|
|
|
|
|
void DL_Dxf::addHatchEdge()
|
|
{
|
|
if( hatchEdge.defined )
|
|
{
|
|
if( hatchEdges.size()>0 )
|
|
{
|
|
hatchEdges.back().push_back( hatchEdge );
|
|
}
|
|
|
|
hatchEdge = DL_HatchEdgeData();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Handles all hatch data.
|
|
*/
|
|
bool DL_Dxf::handleHatchData( DL_CreationInterface* creationInterface )
|
|
{
|
|
// New polyline loop, group code 92
|
|
// or new loop with individual edges, group code 93
|
|
if( groupCode==92 || groupCode==93 )
|
|
{
|
|
if( firstHatchLoop )
|
|
{
|
|
hatchEdges.clear();
|
|
firstHatchLoop = false;
|
|
}
|
|
|
|
if( groupCode==92 && (toInt( groupValue ) & 2)==2 )
|
|
{
|
|
addHatchLoop();
|
|
}
|
|
|
|
if( groupCode==93 )
|
|
{
|
|
addHatchLoop();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// New hatch edge or new section / entity: add last hatch edge:
|
|
if( groupCode==72 || groupCode==0 || groupCode==78 || groupCode==98 )
|
|
{
|
|
// polyline boundaries use code 72 for bulge flag:
|
|
if( groupCode!=72 || (getIntValue( 92, 0 ) & 2)==0 )
|
|
{
|
|
addHatchEdge();
|
|
}
|
|
|
|
if( groupCode==0 /*|| groupCode==78*/ )
|
|
{
|
|
addHatch( creationInterface );
|
|
}
|
|
else
|
|
{
|
|
hatchEdge.type = toInt( groupValue );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// polyline boundary:
|
|
if( (getIntValue( 92, 0 ) & 2)==2 )
|
|
{
|
|
switch( groupCode )
|
|
{
|
|
case 10:
|
|
hatchEdge.type = 0;
|
|
hatchEdge.vertices.push_back( std::vector<double>() );
|
|
hatchEdge.vertices.back().push_back( toReal( groupValue ) );
|
|
return true;
|
|
|
|
case 20:
|
|
|
|
if( !hatchEdge.vertices.empty() )
|
|
{
|
|
hatchEdge.vertices.back().push_back( toReal( groupValue ) );
|
|
hatchEdge.defined = true;
|
|
}
|
|
|
|
return true;
|
|
|
|
case 42:
|
|
|
|
if( !hatchEdge.vertices.empty() )
|
|
{
|
|
hatchEdge.vertices.back().push_back( toReal( groupValue ) );
|
|
hatchEdge.defined = true;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Line edge:
|
|
if( hatchEdge.type==1 )
|
|
{
|
|
switch( groupCode )
|
|
{
|
|
case 10:
|
|
hatchEdge.x1 = toReal( groupValue );
|
|
return true;
|
|
|
|
case 20:
|
|
hatchEdge.y1 = toReal( groupValue );
|
|
return true;
|
|
|
|
case 11:
|
|
hatchEdge.x2 = toReal( groupValue );
|
|
return true;
|
|
|
|
case 21:
|
|
hatchEdge.y2 = toReal( groupValue );
|
|
hatchEdge.defined = true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Arc edge:
|
|
if( hatchEdge.type==2 )
|
|
{
|
|
switch( groupCode )
|
|
{
|
|
case 10:
|
|
hatchEdge.cx = toReal( groupValue );
|
|
return true;
|
|
|
|
case 20:
|
|
hatchEdge.cy = toReal( groupValue );
|
|
return true;
|
|
|
|
case 40:
|
|
hatchEdge.radius = toReal( groupValue );
|
|
return true;
|
|
|
|
case 50:
|
|
hatchEdge.angle1 = toReal( groupValue ) / 360.0 * 2 * M_PI;
|
|
return true;
|
|
|
|
case 51:
|
|
hatchEdge.angle2 = toReal( groupValue ) / 360.0 * 2 * M_PI;
|
|
return true;
|
|
|
|
case 73:
|
|
hatchEdge.ccw = (bool) toInt( groupValue );
|
|
hatchEdge.defined = true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Ellipse arc edge:
|
|
if( hatchEdge.type==3 )
|
|
{
|
|
switch( groupCode )
|
|
{
|
|
case 10:
|
|
hatchEdge.cx = toReal( groupValue );
|
|
return true;
|
|
|
|
case 20:
|
|
hatchEdge.cy = toReal( groupValue );
|
|
return true;
|
|
|
|
case 11:
|
|
hatchEdge.mx = toReal( groupValue );
|
|
return true;
|
|
|
|
case 21:
|
|
hatchEdge.my = toReal( groupValue );
|
|
return true;
|
|
|
|
case 40:
|
|
hatchEdge.ratio = toReal( groupValue );
|
|
return true;
|
|
|
|
case 50:
|
|
hatchEdge.angle1 = toReal( groupValue ) / 360.0 * 2 * M_PI;
|
|
return true;
|
|
|
|
case 51:
|
|
hatchEdge.angle2 = toReal( groupValue ) / 360.0 * 2 * M_PI;
|
|
return true;
|
|
|
|
case 73:
|
|
hatchEdge.ccw = (bool) toInt( groupValue );
|
|
hatchEdge.defined = true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Spline edge:
|
|
if( hatchEdge.type==4 )
|
|
{
|
|
switch( groupCode )
|
|
{
|
|
case 94:
|
|
hatchEdge.degree = toInt( groupValue );
|
|
return true;
|
|
|
|
case 73:
|
|
hatchEdge.rational = toBool( groupValue );
|
|
return true;
|
|
|
|
case 74:
|
|
hatchEdge.periodic = toBool( groupValue );
|
|
return true;
|
|
|
|
case 95:
|
|
hatchEdge.nKnots = toInt( groupValue );
|
|
return true;
|
|
|
|
case 96:
|
|
hatchEdge.nControl = toInt( groupValue );
|
|
return true;
|
|
|
|
case 97:
|
|
hatchEdge.nFit = toInt( groupValue );
|
|
return true;
|
|
|
|
case 40:
|
|
|
|
if( hatchEdge.knots.size() < hatchEdge.nKnots )
|
|
{
|
|
hatchEdge.knots.push_back( toReal( groupValue ) );
|
|
}
|
|
|
|
return true;
|
|
|
|
case 10:
|
|
|
|
if( hatchEdge.controlPoints.size() < hatchEdge.nControl )
|
|
{
|
|
std::vector<double> v;
|
|
v.push_back( toReal( groupValue ) );
|
|
hatchEdge.controlPoints.push_back( v );
|
|
}
|
|
|
|
return true;
|
|
|
|
case 20:
|
|
|
|
if( !hatchEdge.controlPoints.empty() && hatchEdge.controlPoints.back().size()==1 )
|
|
{
|
|
hatchEdge.controlPoints.back().push_back( toReal( groupValue ) );
|
|
}
|
|
|
|
hatchEdge.defined = true;
|
|
return true;
|
|
|
|
case 42:
|
|
|
|
if( hatchEdge.weights.size() < hatchEdge.nControl )
|
|
{
|
|
hatchEdge.weights.push_back( toReal( groupValue ) );
|
|
}
|
|
|
|
return true;
|
|
|
|
case 11:
|
|
|
|
if( hatchEdge.fitPoints.size() < hatchEdge.nFit )
|
|
{
|
|
std::vector<double> v;
|
|
v.push_back( toReal( groupValue ) );
|
|
hatchEdge.fitPoints.push_back( v );
|
|
}
|
|
|
|
return true;
|
|
|
|
case 21:
|
|
|
|
if( !hatchEdge.fitPoints.empty() && hatchEdge.fitPoints.back().size()==1 )
|
|
{
|
|
hatchEdge.fitPoints.back().push_back( toReal( groupValue ) );
|
|
}
|
|
|
|
hatchEdge.defined = true;
|
|
return true;
|
|
|
|
case 12:
|
|
hatchEdge.startTangentX = toReal( groupValue );
|
|
return true;
|
|
|
|
case 22:
|
|
hatchEdge.startTangentY = toReal( groupValue );
|
|
return true;
|
|
|
|
case 13:
|
|
hatchEdge.endTangentX = toReal( groupValue );
|
|
return true;
|
|
|
|
case 23:
|
|
hatchEdge.endTangentY = toReal( groupValue );
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an image entity that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addImage( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_ImageData id( // pass ref insead of name we don't have yet
|
|
getStringValue( 340, "" ),
|
|
// ins point:
|
|
getRealValue( 10, 0.0 ),
|
|
getRealValue( 20, 0.0 ),
|
|
getRealValue( 30, 0.0 ),
|
|
// u vector:
|
|
getRealValue( 11, 1.0 ),
|
|
getRealValue( 21, 0.0 ),
|
|
getRealValue( 31, 0.0 ),
|
|
// v vector:
|
|
getRealValue( 12, 0.0 ),
|
|
getRealValue( 22, 1.0 ),
|
|
getRealValue( 32, 0.0 ),
|
|
// image size (pixel):
|
|
getIntValue( 13, 1 ),
|
|
getIntValue( 23, 1 ),
|
|
// brightness, contrast, fade
|
|
getIntValue( 281, 50 ),
|
|
getIntValue( 282, 50 ),
|
|
getIntValue( 283, 0 ) );
|
|
|
|
creationInterface->addImage( id );
|
|
creationInterface->endEntity();
|
|
currentObjectType = DL_UNKNOWN;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an image definition that was read from the file via the creation interface.
|
|
*/
|
|
void DL_Dxf::addImageDef( DL_CreationInterface* creationInterface )
|
|
{
|
|
DL_ImageDefData id( // handle
|
|
getStringValue( 5, "" ),
|
|
getStringValue( 1, "" ) );
|
|
|
|
creationInterface->linkImage( id );
|
|
creationInterface->endEntity();
|
|
currentObjectType = DL_UNKNOWN;
|
|
}
|
|
|
|
|
|
/**
|
|
* Ends some special entities like hatches or old style polylines.
|
|
*/
|
|
void DL_Dxf::endEntity( DL_CreationInterface* creationInterface )
|
|
{
|
|
creationInterface->endEntity();
|
|
}
|
|
|
|
|
|
/**
|
|
* Ends a sequence and notifies the creation interface.
|
|
*/
|
|
void DL_Dxf::endSequence( DL_CreationInterface* creationInterface )
|
|
{
|
|
creationInterface->endSequence();
|
|
}
|
|
|
|
|
|
/**
|
|
* Converts the given string into an int.
|
|
* ok is set to false if there was an error.
|
|
*/
|
|
// int DL_Dxf::stringToInt(const char* s, bool* ok) {
|
|
// if (ok!=NULL) {
|
|
//// check string:
|
|
// *ok = true;
|
|
// int i=0;
|
|
// bool dot = false;
|
|
// do {
|
|
// if (s[i]=='\0') {
|
|
// break;
|
|
// } else if (s[i]=='.') {
|
|
// if (dot==true) {
|
|
////std::cerr << "two dots\n";
|
|
// *ok = false;
|
|
// } else {
|
|
// dot = true;
|
|
// }
|
|
// } else if (s[i]<'0' || s[i]>'9') {
|
|
////std::cerr << "NaN: '" << s[i] << "'\n";
|
|
// *ok = false;
|
|
// }
|
|
// i++;
|
|
// } while(s[i]!='\0' && *ok==true);
|
|
// }
|
|
|
|
// return atoi(s);
|
|
// }
|
|
|
|
|
|
/**
|
|
* @brief Opens the given file for writing and returns a pointer
|
|
* to the dxf writer. This pointer needs to be passed on to other
|
|
* writing functions.
|
|
*
|
|
* @param file Full path of the file to open.
|
|
*
|
|
* @return Pointer to an ascii dxf writer object.
|
|
*/
|
|
DL_WriterA* DL_Dxf::out( const char* file, DL_Codes::version aVersion )
|
|
{
|
|
char* f = new char[strlen( file ) + 1];
|
|
|
|
strcpy( f, file );
|
|
this->version = aVersion;
|
|
|
|
DL_WriterA* dw = new DL_WriterA( f, aVersion );
|
|
|
|
if( dw->openFailed() )
|
|
{
|
|
delete dw;
|
|
delete[] f;
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
delete[] f;
|
|
return dw;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Writes a DXF header to the file currently opened
|
|
* by the given DXF writer object.
|
|
*/
|
|
void DL_Dxf::writeHeader( DL_WriterA& dw )
|
|
{
|
|
dw.comment( "dxflib " DL_VERSION );
|
|
dw.sectionHeader();
|
|
|
|
dw.dxfString( 9, "$ACADVER" );
|
|
|
|
switch( version )
|
|
{
|
|
case DL_Codes::AC1009:
|
|
dw.dxfString( 1, "AC1009" );
|
|
break;
|
|
|
|
case DL_Codes::AC1012:
|
|
dw.dxfString( 1, "AC1012" );
|
|
break;
|
|
|
|
case DL_Codes::AC1014:
|
|
dw.dxfString( 1, "AC1014" );
|
|
break;
|
|
|
|
case DL_Codes::AC1015:
|
|
dw.dxfString( 1, "AC1015" );
|
|
break;
|
|
|
|
case DL_Codes::AC1009_MIN:
|
|
// minimalistic DXF version is unidentified in file:
|
|
break;
|
|
}
|
|
|
|
// Newer version require that (otherwise a*cad crashes..)
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 9, "$HANDSEED" );
|
|
dw.dxfHex( 5, 0xFFFF );
|
|
}
|
|
|
|
// commented out: more variables can be added after that by caller:
|
|
// dw.sectionEnd();
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a point entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writePoint( DL_WriterA& dw,
|
|
const DL_PointData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "POINT" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbPoint" );
|
|
}
|
|
|
|
dw.coord( DL_POINT_COORD_CODE, data.x, data.y, data.z );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a line entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeLine( DL_WriterA& dw,
|
|
const DL_LineData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "LINE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbLine" );
|
|
}
|
|
|
|
dw.coord( DL_LINE_START_CODE, data.x1, data.y1, data.z1 );
|
|
dw.coord( DL_LINE_END_CODE, data.x2, data.y2, data.z2 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an x line entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeXLine( DL_WriterA& dw,
|
|
const DL_XLineData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "XLINE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbLine" );
|
|
}
|
|
|
|
dw.coord( DL_LINE_START_CODE, data.bx, data.by, data.bz );
|
|
dw.coord( DL_LINE_END_CODE, data.dx, data.dy, data.dz );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a ray entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeRay( DL_WriterA& dw,
|
|
const DL_RayData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "RAY" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbLine" );
|
|
}
|
|
|
|
dw.coord( DL_LINE_START_CODE, data.bx, data.by, data.bz );
|
|
dw.coord( DL_LINE_END_CODE, data.dx, data.dy, data.dz );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a polyline entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
* @see writeVertex
|
|
*/
|
|
void DL_Dxf::writePolyline( DL_WriterA& dw,
|
|
const DL_PolylineData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.entity( "LWPOLYLINE" );
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
dw.entityAttributes( attrib );
|
|
dw.dxfString( 100, "AcDbPolyline" );
|
|
dw.dxfInt( 90, (int) data.number );
|
|
dw.dxfInt( 70, data.flags );
|
|
}
|
|
else
|
|
{
|
|
dw.entity( "POLYLINE" );
|
|
dw.entityAttributes( attrib );
|
|
polylineLayer = attrib.getLayer();
|
|
dw.dxfInt( 66, 1 );
|
|
dw.dxfInt( 70, data.flags );
|
|
dw.coord( DL_VERTEX_COORD_CODE, 0.0, 0.0, 0.0 );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single vertex of a polyline to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeVertex( DL_WriterA& dw,
|
|
const DL_VertexData& data )
|
|
{
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfReal( 10, data.x );
|
|
dw.dxfReal( 20, data.y );
|
|
dw.dxfReal( 30, data.z );
|
|
|
|
if( fabs( data.bulge )>1.0e-10 )
|
|
{
|
|
dw.dxfReal( 42, data.bulge );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dw.entity( "VERTEX" );
|
|
// dw.entityAttributes(aAttrib);
|
|
dw.dxfString( 8, polylineLayer );
|
|
dw.coord( DL_VERTEX_COORD_CODE, data.x, data.y, data.z );
|
|
|
|
if( fabs( data.bulge )>1.0e-10 )
|
|
{
|
|
dw.dxfReal( 42, data.bulge );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the polyline end. Only needed for DXF R12.
|
|
*/
|
|
void DL_Dxf::writePolylineEnd( DL_WriterA& dw )
|
|
{
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
}
|
|
else
|
|
{
|
|
dw.entity( "SEQEND" );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a spline entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
* @see writeControlPoint
|
|
*/
|
|
void DL_Dxf::writeSpline( DL_WriterA& dw,
|
|
const DL_SplineData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "SPLINE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSpline" );
|
|
}
|
|
|
|
dw.dxfInt( 70, data.flags );
|
|
dw.dxfInt( 71, data.degree );
|
|
dw.dxfInt( 72, data.nKnots ); // number of knots
|
|
dw.dxfInt( 73, data.nControl ); // number of control points
|
|
dw.dxfInt( 74, data.nFit ); // number of fit points
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single control point of a spline to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeControlPoint( DL_WriterA& dw,
|
|
const DL_ControlPointData& data )
|
|
{
|
|
dw.dxfReal( 10, data.x );
|
|
dw.dxfReal( 20, data.y );
|
|
dw.dxfReal( 30, data.z );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single fit point of a spline to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
*/
|
|
void DL_Dxf::writeFitPoint( DL_WriterA& dw,
|
|
const DL_FitPointData& data )
|
|
{
|
|
dw.dxfReal( 11, data.x );
|
|
dw.dxfReal( 21, data.y );
|
|
dw.dxfReal( 31, data.z );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single knot of a spline to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
*/
|
|
void DL_Dxf::writeKnot( DL_WriterA& dw,
|
|
const DL_KnotData& data )
|
|
{
|
|
dw.dxfReal( 40, data.k );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a circle entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeCircle( DL_WriterA& dw,
|
|
const DL_CircleData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "CIRCLE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbCircle" );
|
|
}
|
|
|
|
dw.coord( 10, data.cx, data.cy, data.cz );
|
|
dw.dxfReal( 40, data.radius );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an arc entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeArc( DL_WriterA& dw,
|
|
const DL_ArcData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "ARC" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbCircle" );
|
|
}
|
|
|
|
dw.coord( 10, data.cx, data.cy, data.cz );
|
|
dw.dxfReal( 40, data.radius );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbArc" );
|
|
}
|
|
|
|
dw.dxfReal( 50, data.angle1 );
|
|
dw.dxfReal( 51, data.angle2 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an ellipse entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeEllipse( DL_WriterA& dw,
|
|
const DL_EllipseData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.entity( "ELLIPSE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEllipse" );
|
|
}
|
|
|
|
dw.coord( 10, data.cx, data.cy, data.cz );
|
|
dw.coord( 11, data.mx, data.my, data.mz );
|
|
dw.dxfReal( 40, data.ratio );
|
|
dw.dxfReal( 41, data.angle1 );
|
|
dw.dxfReal( 42, data.angle2 );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a solid entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeSolid( DL_WriterA& dw,
|
|
const DL_SolidData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "SOLID" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbTrace" );
|
|
}
|
|
|
|
dw.coord( 10, data.x[0], data.y[0], data.z[0] );
|
|
dw.coord( 11, data.x[1], data.y[1], data.z[1] );
|
|
dw.coord( 12, data.x[2], data.y[2], data.z[2] );
|
|
dw.coord( 13, data.x[3], data.y[3], data.z[3] );
|
|
dw.dxfReal( 39, data.thickness );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a trace entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeTrace( DL_WriterA& dw,
|
|
const DL_TraceData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "TRACE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbTrace" );
|
|
}
|
|
|
|
dw.coord( 10, data.x[0], data.y[0], data.z[0] );
|
|
dw.coord( 11, data.x[1], data.y[1], data.z[1] );
|
|
dw.coord( 12, data.x[2], data.y[2], data.z[2] );
|
|
dw.coord( 13, data.x[3], data.y[3], data.z[3] );
|
|
dw.dxfReal( 39, data.thickness );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a 3d face entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::write3dFace( DL_WriterA& dw,
|
|
const DL_3dFaceData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "3DFACE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbFace" );
|
|
}
|
|
|
|
dw.coord( 10, data.x[0], data.y[0], data.z[0] );
|
|
dw.coord( 11, data.x[1], data.y[1], data.z[1] );
|
|
dw.coord( 12, data.x[2], data.y[2], data.z[2] );
|
|
dw.coord( 13, data.x[3], data.y[3], data.z[3] );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an insert to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeInsert( DL_WriterA& dw,
|
|
const DL_InsertData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
if( data.name.empty() )
|
|
{
|
|
std::cerr << "DL_Dxf::writeInsert: "
|
|
<< "Block name must not be empty\n";
|
|
return;
|
|
}
|
|
|
|
dw.entity( "INSERT" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
if( data.cols!=1 || data.rows!=1 )
|
|
{
|
|
dw.dxfString( 100, "AcDbMInsertBlock" );
|
|
}
|
|
else
|
|
{
|
|
dw.dxfString( 100, "AcDbBlockReference" );
|
|
}
|
|
}
|
|
|
|
dw.dxfString( 2, data.name );
|
|
dw.dxfReal( 10, data.ipx );
|
|
dw.dxfReal( 20, data.ipy );
|
|
dw.dxfReal( 30, data.ipz );
|
|
|
|
if( data.sx!=1.0 || data.sy!=1.0 )
|
|
{
|
|
dw.dxfReal( 41, data.sx );
|
|
dw.dxfReal( 42, data.sy );
|
|
dw.dxfReal( 43, 1.0 );
|
|
}
|
|
|
|
if( data.angle!=0.0 )
|
|
{
|
|
dw.dxfReal( 50, data.angle );
|
|
}
|
|
|
|
if( data.cols!=1 || data.rows!=1 )
|
|
{
|
|
dw.dxfInt( 70, data.cols );
|
|
dw.dxfInt( 71, data.rows );
|
|
}
|
|
|
|
if( data.colSp!=0.0 || data.rowSp!=0.0 )
|
|
{
|
|
dw.dxfReal( 44, data.colSp );
|
|
dw.dxfReal( 45, data.rowSp );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a multi text entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeMText( DL_WriterA& dw,
|
|
const DL_MTextData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "MTEXT" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbMText" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.ipx );
|
|
dw.dxfReal( 20, data.ipy );
|
|
dw.dxfReal( 30, data.ipz );
|
|
dw.dxfReal( 40, data.height );
|
|
dw.dxfReal( 41, data.width );
|
|
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.drawingDirection );
|
|
|
|
// Creare text chunks of 250 characters each:
|
|
int length = data.text.length();
|
|
char chunk[251];
|
|
int i;
|
|
|
|
for( i = 250; i<length; i += 250 )
|
|
{
|
|
strncpy( chunk, &data.text.c_str()[i - 250], 250 );
|
|
chunk[250] = '\0';
|
|
dw.dxfString( 3, chunk );
|
|
}
|
|
|
|
strncpy( chunk, &data.text.c_str()[i - 250], 250 );
|
|
chunk[250] = '\0';
|
|
dw.dxfString( 1, chunk );
|
|
|
|
dw.dxfString( 7, data.style );
|
|
|
|
// since dxflib 2.0.2.1: degrees not rad (error in autodesk dxf doc)
|
|
dw.dxfReal( 50, data.angle / (2.0 * M_PI) * 360.0 );
|
|
|
|
dw.dxfInt( 73, data.lineSpacingStyle );
|
|
dw.dxfReal( 44, data.lineSpacingFactor );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a text entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeText( DL_WriterA& dw,
|
|
const DL_TextData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "TEXT" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbText" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.ipx );
|
|
dw.dxfReal( 20, data.ipy );
|
|
dw.dxfReal( 30, data.ipz );
|
|
dw.dxfReal( 40, data.height );
|
|
dw.dxfString( 1, data.text );
|
|
dw.dxfReal( 50, data.angle / (2 * M_PI) * 360.0 );
|
|
dw.dxfReal( 41, data.xScaleFactor );
|
|
dw.dxfString( 7, data.style );
|
|
|
|
dw.dxfInt( 71, data.textGenerationFlags );
|
|
dw.dxfInt( 72, data.hJustification );
|
|
|
|
dw.dxfReal( 11, data.apx );
|
|
dw.dxfReal( 21, data.apy );
|
|
dw.dxfReal( 31, data.apz );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
// required twice for some reason:
|
|
dw.dxfString( 100, "AcDbText" );
|
|
}
|
|
|
|
dw.dxfInt( 73, data.vJustification );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeAttribute( DL_WriterA& dw,
|
|
const DL_AttributeData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "ATTRIB" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbText" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.ipx );
|
|
dw.dxfReal( 20, data.ipy );
|
|
dw.dxfReal( 30, data.ipz );
|
|
dw.dxfReal( 40, data.height );
|
|
dw.dxfString( 1, data.text );
|
|
dw.dxfReal( 50, data.angle / (2 * M_PI) * 360.0 );
|
|
dw.dxfReal( 41, data.xScaleFactor );
|
|
dw.dxfString( 7, data.style );
|
|
|
|
dw.dxfInt( 71, data.textGenerationFlags );
|
|
dw.dxfInt( 72, data.hJustification );
|
|
|
|
dw.dxfReal( 11, data.apx );
|
|
dw.dxfReal( 21, data.apy );
|
|
dw.dxfReal( 31, data.apz );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbAttribute" );
|
|
}
|
|
|
|
dw.dxfString( 2, data.tag );
|
|
dw.dxfInt( 74, data.vJustification );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeDimStyleOverrides( DL_WriterA& dw,
|
|
const DL_DimensionData& data )
|
|
{
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 1001, "ACAD" );
|
|
dw.dxfString( 1000, "DSTYLE" );
|
|
dw.dxfString( 1002, "{" );
|
|
dw.dxfInt( 1070, 144 );
|
|
dw.dxfReal( 1040, data.linearFactor );
|
|
dw.dxfInt( 1070, 40 );
|
|
dw.dxfReal( 1040, data.dimScale );
|
|
dw.dxfString( 1002, "}" );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an aligned dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific aligned dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimAligned( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimAlignedData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbAlignedDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 13, edata.epx1 );
|
|
dw.dxfReal( 23, edata.epy1 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
|
|
dw.dxfReal( 14, edata.epx2 );
|
|
dw.dxfReal( 24, edata.epy2 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
|
|
writeDimStyleOverrides( dw, data );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a linear dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific linear dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimLinear( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimLinearData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbAlignedDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 13, edata.dpx1 );
|
|
dw.dxfReal( 23, edata.dpy1 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
|
|
dw.dxfReal( 14, edata.dpx2 );
|
|
dw.dxfReal( 24, edata.dpy2 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
|
|
dw.dxfReal( 50, edata.angle / (2.0 * M_PI) * 360.0 );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbRotatedDimension" );
|
|
}
|
|
|
|
writeDimStyleOverrides( dw, data );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a radial dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific radial dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimRadial( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimRadialData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbRadialDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 15, edata.dpx );
|
|
dw.dxfReal( 25, edata.dpy );
|
|
dw.dxfReal( 35, 0.0 );
|
|
|
|
dw.dxfReal( 40, edata.leader );
|
|
|
|
writeDimStyleOverrides( dw, data );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a diametric dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific diametric dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimDiametric( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimDiametricData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDiametricDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 15, edata.dpx );
|
|
dw.dxfReal( 25, edata.dpy );
|
|
dw.dxfReal( 35, 0.0 );
|
|
|
|
dw.dxfReal( 40, edata.leader );
|
|
|
|
writeDimStyleOverrides( dw, data );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an angular dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific angular dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimAngular( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimAngularData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDb2LineAngularDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 13, edata.dpx1 );
|
|
dw.dxfReal( 23, edata.dpy1 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
|
|
dw.dxfReal( 14, edata.dpx2 );
|
|
dw.dxfReal( 24, edata.dpy2 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
|
|
dw.dxfReal( 15, edata.dpx3 );
|
|
dw.dxfReal( 25, edata.dpy3 );
|
|
dw.dxfReal( 35, 0.0 );
|
|
|
|
dw.dxfReal( 16, edata.dpx4 );
|
|
dw.dxfReal( 26, edata.dpy4 );
|
|
dw.dxfReal( 36, 0.0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an angular dimension entity (3 points version) to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific angular dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimAngular3P( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimAngular3PData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
dw.dxfInt( 70, data.type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfReal( 42, data.angle );
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDb3PointAngularDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 13, edata.dpx1 );
|
|
dw.dxfReal( 23, edata.dpy1 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
|
|
dw.dxfReal( 14, edata.dpx2 );
|
|
dw.dxfReal( 24, edata.dpy2 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
|
|
dw.dxfReal( 15, edata.dpx3 );
|
|
dw.dxfReal( 25, edata.dpy3 );
|
|
dw.dxfReal( 35, 0.0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an ordinate dimension entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Generic dimension data for from the file
|
|
* @param data Specific ordinate dimension data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeDimOrdinate( DL_WriterA& dw,
|
|
const DL_DimensionData& data,
|
|
const DL_DimOrdinateData& edata,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "DIMENSION" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 10, data.dpx );
|
|
dw.dxfReal( 20, data.dpy );
|
|
dw.dxfReal( 30, data.dpz );
|
|
|
|
dw.dxfReal( 11, data.mpx );
|
|
dw.dxfReal( 21, data.mpy );
|
|
dw.dxfReal( 31, 0.0 );
|
|
|
|
int type = data.type;
|
|
|
|
if( edata.xtype )
|
|
{
|
|
type |= 0x40;
|
|
}
|
|
|
|
dw.dxfInt( 70, type );
|
|
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, data.attachmentPoint );
|
|
dw.dxfInt( 72, data.lineSpacingStyle ); // opt
|
|
dw.dxfReal( 41, data.lineSpacingFactor ); // opt
|
|
}
|
|
|
|
dw.dxfString( 1, data.text ); // opt
|
|
// dw.dxfString(3, data.style);
|
|
dw.dxfString( 3, "Standard" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbOrdinateDimension" );
|
|
}
|
|
|
|
dw.dxfReal( 13, edata.dpx1 );
|
|
dw.dxfReal( 23, edata.dpy1 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
|
|
dw.dxfReal( 14, edata.dpx2 );
|
|
dw.dxfReal( 24, edata.dpy2 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a leader entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
* @see writeVertex
|
|
*/
|
|
void DL_Dxf::writeLeader( DL_WriterA& dw,
|
|
const DL_LeaderData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.entity( "LEADER" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbLeader" );
|
|
}
|
|
|
|
dw.dxfString( 3, "Standard" );
|
|
dw.dxfInt( 71, data.arrowHeadFlag );
|
|
dw.dxfInt( 72, data.leaderPathType );
|
|
dw.dxfInt( 73, data.leaderCreationFlag );
|
|
dw.dxfInt( 74, data.hooklineDirectionFlag );
|
|
dw.dxfInt( 75, data.hooklineFlag );
|
|
dw.dxfReal( 40, data.textAnnotationHeight );
|
|
dw.dxfReal( 41, data.textAnnotationWidth );
|
|
dw.dxfInt( 76, data.number );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single vertex of a leader to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data
|
|
*/
|
|
void DL_Dxf::writeLeaderVertex( DL_WriterA& dw,
|
|
const DL_LeaderVertexData& data )
|
|
{
|
|
if( version>DL_VERSION_R12 )
|
|
{
|
|
dw.dxfReal( 10, data.x );
|
|
dw.dxfReal( 20, data.y );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the beginning of a hatch entity to the file.
|
|
* This must be followed by one or more writeHatchLoop()
|
|
* calls and a writeHatch2() call.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data.
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeHatch1( DL_WriterA& dw,
|
|
const DL_HatchData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
dw.entity( "HATCH" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbHatch" );
|
|
}
|
|
|
|
dw.dxfReal( 10, 0.0 ); // elevation
|
|
dw.dxfReal( 20, 0.0 );
|
|
dw.dxfReal( 30, 0.0 );
|
|
dw.dxfReal( 210, 0.0 ); // extrusion dir.
|
|
dw.dxfReal( 220, 0.0 );
|
|
dw.dxfReal( 230, 1.0 );
|
|
|
|
if( data.solid==false )
|
|
{
|
|
dw.dxfString( 2, data.pattern );
|
|
}
|
|
else
|
|
{
|
|
dw.dxfString( 2, "SOLID" );
|
|
}
|
|
|
|
dw.dxfInt( 70, (int) data.solid );
|
|
dw.dxfInt( 71, 0 ); // non-associative
|
|
dw.dxfInt( 91, data.numLoops );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the end of a hatch entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data.
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeHatch2( DL_WriterA& dw,
|
|
const DL_HatchData& data,
|
|
const DL_Attributes& /*aAttrib*/ )
|
|
{
|
|
dw.dxfInt( 75, 0 ); // odd parity
|
|
dw.dxfInt( 76, 1 ); // pattern type
|
|
|
|
if( data.solid==false )
|
|
{
|
|
dw.dxfReal( 52, data.angle );
|
|
dw.dxfReal( 41, data.scale );
|
|
dw.dxfInt( 77, 0 ); // not double
|
|
// dw.dxfInt(78, 0);
|
|
dw.dxfInt( 78, 1 );
|
|
dw.dxfReal( 53, 45.0 );
|
|
dw.dxfReal( 43, 0.0 );
|
|
dw.dxfReal( 44, 0.0 );
|
|
dw.dxfReal( 45, -0.0883883476483184 );
|
|
dw.dxfReal( 46, 0.0883883476483185 );
|
|
dw.dxfInt( 79, 0 );
|
|
}
|
|
|
|
dw.dxfInt( 98, 0 );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 1001, "ACAD" );
|
|
dw.dxfReal( 1010, data.originX );
|
|
dw.dxfReal( 1020, data.originY );
|
|
dw.dxfInt( 1030, 0.0 );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the beginning of a hatch loop to the file. This
|
|
* must happen after writing the beginning of a hatch entity.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data.
|
|
*/
|
|
void DL_Dxf::writeHatchLoop1( DL_WriterA& dw,
|
|
const DL_HatchLoopData& data )
|
|
{
|
|
dw.dxfInt( 92, 1 );
|
|
dw.dxfInt( 93, data.numEdges );
|
|
// dw.dxfInt(97, 0);
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the end of a hatch loop to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data.
|
|
*/
|
|
void DL_Dxf::writeHatchLoop2( DL_WriterA& dw,
|
|
const DL_HatchLoopData& /*data*/ )
|
|
{
|
|
dw.dxfInt( 97, 0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the beginning of a hatch entity to the file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data.
|
|
*/
|
|
void DL_Dxf::writeHatchEdge( DL_WriterA& dw,
|
|
const DL_HatchEdgeData& data )
|
|
{
|
|
if( data.type<1 || data.type>4 )
|
|
{
|
|
printf( "WARNING: unsupported hatch edge type: %d", data.type );
|
|
}
|
|
|
|
dw.dxfInt( 72, data.type );
|
|
|
|
switch( data.type )
|
|
{
|
|
// line:
|
|
case 1:
|
|
dw.dxfReal( 10, data.x1 );
|
|
dw.dxfReal( 20, data.y1 );
|
|
dw.dxfReal( 11, data.x2 );
|
|
dw.dxfReal( 21, data.y2 );
|
|
break;
|
|
|
|
// arc:
|
|
case 2:
|
|
dw.dxfReal( 10, data.cx );
|
|
dw.dxfReal( 20, data.cy );
|
|
dw.dxfReal( 40, data.radius );
|
|
dw.dxfReal( 50, data.angle1 / (2 * M_PI) * 360.0 );
|
|
dw.dxfReal( 51, data.angle2 / (2 * M_PI) * 360.0 );
|
|
dw.dxfInt( 73, (int) (data.ccw) );
|
|
break;
|
|
|
|
// ellipse arc:
|
|
case 3:
|
|
dw.dxfReal( 10, data.cx );
|
|
dw.dxfReal( 20, data.cy );
|
|
dw.dxfReal( 11, data.mx );
|
|
dw.dxfReal( 21, data.my );
|
|
dw.dxfReal( 40, data.ratio );
|
|
dw.dxfReal( 50, data.angle1 / (2 * M_PI) * 360.0 );
|
|
dw.dxfReal( 51, data.angle2 / (2 * M_PI) * 360.0 );
|
|
dw.dxfInt( 73, (int) (data.ccw) );
|
|
break;
|
|
|
|
// spline:
|
|
case 4:
|
|
dw.dxfInt( 94, data.degree );
|
|
dw.dxfBool( 73, data.rational );
|
|
dw.dxfBool( 74, data.periodic );
|
|
dw.dxfInt( 95, data.nKnots );
|
|
dw.dxfInt( 96, data.nControl );
|
|
|
|
for( unsigned int i = 0; i<data.knots.size(); i++ )
|
|
{
|
|
dw.dxfReal( 40, data.knots[i] );
|
|
}
|
|
|
|
for( unsigned int i = 0; i<data.controlPoints.size(); i++ )
|
|
{
|
|
dw.dxfReal( 10, data.controlPoints[i][0] );
|
|
dw.dxfReal( 20, data.controlPoints[i][1] );
|
|
}
|
|
|
|
for( unsigned int i = 0; i<data.weights.size(); i++ )
|
|
{
|
|
dw.dxfReal( 42, data.weights[i] );
|
|
}
|
|
|
|
if( data.nFit>0 )
|
|
{
|
|
dw.dxfInt( 97, data.nFit );
|
|
|
|
for( unsigned int i = 0; i<data.fitPoints.size(); i++ )
|
|
{
|
|
dw.dxfReal( 11, data.fitPoints[i][0] );
|
|
dw.dxfReal( 21, data.fitPoints[i][1] );
|
|
}
|
|
}
|
|
|
|
if( fabs( data.startTangentX )>1.0e-4 || fabs( data.startTangentY )>1.0e-4 )
|
|
{
|
|
dw.dxfReal( 12, data.startTangentX );
|
|
dw.dxfReal( 22, data.startTangentY );
|
|
}
|
|
|
|
if( fabs( data.endTangentX )>1.0e-4 || fabs( data.endTangentY )>1.0e-4 )
|
|
{
|
|
dw.dxfReal( 13, data.endTangentX );
|
|
dw.dxfReal( 23, data.endTangentY );
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an image entity.
|
|
*
|
|
* @return IMAGEDEF handle. Needed for the IMAGEDEF counterpart.
|
|
*/
|
|
int DL_Dxf::writeImage( DL_WriterA& dw,
|
|
const DL_ImageData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
/*if (data.file.empty()) {
|
|
* std::cerr << "DL_Dxf::writeImage: "
|
|
* << "Image file must not be empty\n";
|
|
* return;
|
|
* }*/
|
|
|
|
dw.entity( "IMAGE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbEntity" );
|
|
}
|
|
|
|
dw.entityAttributes( aAttrib );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbRasterImage" );
|
|
dw.dxfInt( 90, 0 );
|
|
}
|
|
|
|
// insertion point
|
|
dw.dxfReal( 10, data.ipx );
|
|
dw.dxfReal( 20, data.ipy );
|
|
dw.dxfReal( 30, data.ipz );
|
|
|
|
// vector along bottom side (1 pixel long)
|
|
dw.dxfReal( 11, data.ux );
|
|
dw.dxfReal( 21, data.uy );
|
|
dw.dxfReal( 31, data.uz );
|
|
|
|
// vector along left side (1 pixel long)
|
|
dw.dxfReal( 12, data.vx );
|
|
dw.dxfReal( 22, data.vy );
|
|
dw.dxfReal( 32, data.vz );
|
|
|
|
// image size in pixel
|
|
dw.dxfReal( 13, data.width );
|
|
dw.dxfReal( 23, data.height );
|
|
|
|
// handle of IMAGEDEF object
|
|
int handle = dw.incHandle();
|
|
dw.dxfHex( 340, handle );
|
|
|
|
// flags
|
|
dw.dxfInt( 70, 15 );
|
|
|
|
// clipping:
|
|
dw.dxfInt( 280, 0 );
|
|
|
|
// brightness, contrast, fade
|
|
dw.dxfInt( 281, data.brightness );
|
|
dw.dxfInt( 282, data.contrast );
|
|
dw.dxfInt( 283, data.fade );
|
|
|
|
return handle;
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes an image definiition entity.
|
|
*/
|
|
void DL_Dxf::writeImageDef( DL_WriterA& dw,
|
|
int handle,
|
|
const DL_ImageData& data )
|
|
{
|
|
/*if (data.file.empty()) {
|
|
* std::cerr << "DL_Dxf::writeImage: "
|
|
* << "Image file must not be empty\n";
|
|
* return;
|
|
* }*/
|
|
|
|
dw.dxfString( 0, "IMAGEDEF" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, handle );
|
|
}
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbRasterImageDef" );
|
|
dw.dxfInt( 90, 0 );
|
|
}
|
|
|
|
// file name:
|
|
dw.dxfString( 1, data.ref );
|
|
|
|
// image size in pixel
|
|
dw.dxfReal( 10, data.width );
|
|
dw.dxfReal( 20, data.height );
|
|
|
|
dw.dxfReal( 11, 1.0 );
|
|
dw.dxfReal( 21, 1.0 );
|
|
|
|
// loaded:
|
|
dw.dxfInt( 280, 1 );
|
|
// units:
|
|
dw.dxfInt( 281, 0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a layer to the file. Layers are stored in the
|
|
* tables section of a DXF file.
|
|
*
|
|
* @param dw DXF writer
|
|
* @param data Entity data from the file
|
|
* @param aAttrib Attributes
|
|
*/
|
|
void DL_Dxf::writeLayer( DL_WriterA& dw,
|
|
const DL_LayerData& data,
|
|
const DL_Attributes& aAttrib )
|
|
{
|
|
if( data.name.empty() )
|
|
{
|
|
std::cerr << "DL_Dxf::writeLayer: "
|
|
<< "Layer name must not be empty\n";
|
|
return;
|
|
}
|
|
|
|
int color = aAttrib.getColor();
|
|
|
|
if( color>=256 )
|
|
{
|
|
std::cerr << "Layer color cannot be " << color << ". Changed to 7.\n";
|
|
color = 7;
|
|
}
|
|
|
|
if( data.off )
|
|
{
|
|
// negative color value means layer is off:
|
|
color = -color;
|
|
}
|
|
|
|
if( data.name == "0" )
|
|
{
|
|
dw.tableLayerEntry( 0x10 );
|
|
}
|
|
else
|
|
{
|
|
dw.tableLayerEntry();
|
|
}
|
|
|
|
dw.dxfString( 2, data.name );
|
|
dw.dxfInt( 70, data.flags );
|
|
dw.dxfInt( 62, color );
|
|
|
|
if( version>=DL_VERSION_2000 && aAttrib.getColor24()!=-1 )
|
|
{
|
|
dw.dxfInt( 420, attrib.getColor24() );
|
|
}
|
|
|
|
dw.dxfString( 6, ( attrib.getLinetype().length()==0 ?
|
|
std::string( "CONTINUOUS" ) : attrib.getLinetype() ) );
|
|
|
|
if( version>=DL_VERSION_2000 )
|
|
{
|
|
// layer defpoints cannot be plotted
|
|
std::string lstr = data.name;
|
|
std::transform( lstr.begin(), lstr.end(), lstr.begin(), tolower );
|
|
|
|
if( lstr=="defpoints" )
|
|
{
|
|
dw.dxfInt( 290, 0 );
|
|
}
|
|
}
|
|
|
|
if( version>=DL_VERSION_2000 && attrib.getWidth()!=-1 )
|
|
{
|
|
dw.dxfInt( 370, attrib.getWidth() );
|
|
}
|
|
|
|
if( version>=DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 390, 0xF );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a line type to the file. Line types are stored in the
|
|
* tables section of a DXF file.
|
|
*/
|
|
void DL_Dxf::writeLinetype( DL_WriterA& dw,
|
|
const DL_LinetypeData& data )
|
|
{
|
|
std::string nameUpper = data.name;
|
|
std::transform( nameUpper.begin(), nameUpper.end(), nameUpper.begin(), ::toupper );
|
|
|
|
if( data.name.empty() )
|
|
{
|
|
std::cerr << "DL_Dxf::writeLinetype: "
|
|
<< "Line type name must not be empty\n";
|
|
return;
|
|
}
|
|
|
|
// ignore BYLAYER, BYBLOCK for R12
|
|
if( version<DL_VERSION_2000 )
|
|
{
|
|
if( nameUpper=="BYBLOCK" || nameUpper=="BYLAYER" )
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
// write id (not for R12)
|
|
if( nameUpper=="BYBLOCK" )
|
|
{
|
|
dw.tableLinetypeEntry( 0x14 );
|
|
}
|
|
else if( nameUpper=="BYLAYER" )
|
|
{
|
|
dw.tableLinetypeEntry( 0x15 );
|
|
}
|
|
else if( nameUpper=="CONTINUOUS" )
|
|
{
|
|
dw.tableLinetypeEntry( 0x16 );
|
|
}
|
|
else
|
|
{
|
|
dw.tableLinetypeEntry();
|
|
}
|
|
|
|
dw.dxfString( 2, data.name );
|
|
dw.dxfInt( 70, data.flags );
|
|
|
|
if( nameUpper=="BYBLOCK" )
|
|
{
|
|
dw.dxfString( 3, "" );
|
|
dw.dxfInt( 72, 65 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfReal( 40, 0.0 );
|
|
}
|
|
else if( nameUpper=="BYLAYER" )
|
|
{
|
|
dw.dxfString( 3, "" );
|
|
dw.dxfInt( 72, 65 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfReal( 40, 0.0 );
|
|
}
|
|
else if( nameUpper=="CONTINUOUS" )
|
|
{
|
|
dw.dxfString( 3, "Solid line" );
|
|
dw.dxfInt( 72, 65 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfReal( 40, 0.0 );
|
|
}
|
|
else
|
|
{
|
|
dw.dxfString( 3, data.description );
|
|
dw.dxfInt( 72, 65 );
|
|
dw.dxfInt( 73, data.numberOfDashes );
|
|
dw.dxfReal( 40, data.patternLength );
|
|
|
|
for( int i = 0; i < data.numberOfDashes; i++ )
|
|
{
|
|
dw.dxfReal( 49, data.pattern[i] );
|
|
|
|
if( version>=DL_VERSION_R13 )
|
|
{
|
|
dw.dxfInt( 74, 0 );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the APPID section to the DXF file.
|
|
*
|
|
* @param name Application name
|
|
*/
|
|
void DL_Dxf::writeAppid( DL_WriterA& dw, const std::string& name )
|
|
{
|
|
if( name.empty() )
|
|
{
|
|
std::cerr << "DL_Dxf::writeAppid: "
|
|
<< "Application name must not be empty\n";
|
|
return;
|
|
}
|
|
|
|
std::string n = name;
|
|
std::transform( n.begin(), n.end(), n.begin(), ::toupper );
|
|
|
|
if( n=="ACAD" )
|
|
{
|
|
dw.tableAppidEntry( 0x12 );
|
|
}
|
|
else
|
|
{
|
|
dw.tableAppidEntry();
|
|
}
|
|
|
|
dw.dxfString( 2, name );
|
|
dw.dxfInt( 70, 0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a block's definition (no entities) to the DXF file.
|
|
*/
|
|
void DL_Dxf::writeBlock( DL_WriterA& dw, const DL_BlockData& data )
|
|
{
|
|
if( data.name.empty() )
|
|
{
|
|
std::cerr << "DL_Dxf::writeBlock: "
|
|
<< "Block name must not be empty\n";
|
|
return;
|
|
}
|
|
|
|
std::string n = data.name;
|
|
std::transform( n.begin(), n.end(), n.begin(), ::toupper );
|
|
|
|
if( n=="*PAPER_SPACE" )
|
|
{
|
|
dw.sectionBlockEntry( 0x1C );
|
|
}
|
|
else if( n=="*MODEL_SPACE" )
|
|
{
|
|
dw.sectionBlockEntry( 0x20 );
|
|
}
|
|
else if( n=="*PAPER_SPACE0" )
|
|
{
|
|
dw.sectionBlockEntry( 0x24 );
|
|
}
|
|
else
|
|
{
|
|
dw.sectionBlockEntry();
|
|
}
|
|
|
|
dw.dxfString( 2, data.name );
|
|
dw.dxfInt( 70, 0 );
|
|
dw.coord( 10, data.bpx, data.bpy, data.bpz );
|
|
dw.dxfString( 3, data.name );
|
|
dw.dxfString( 1, "" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a block end.
|
|
*
|
|
* @param name Block name
|
|
*/
|
|
void DL_Dxf::writeEndBlock( DL_WriterA& dw, const std::string& name )
|
|
{
|
|
std::string n = name;
|
|
std::transform( n.begin(), n.end(), n.begin(), ::toupper );
|
|
|
|
if( n=="*PAPER_SPACE" )
|
|
{
|
|
dw.sectionBlockEntryEnd( 0x1D );
|
|
}
|
|
else if( n=="*MODEL_SPACE" )
|
|
{
|
|
dw.sectionBlockEntryEnd( 0x21 );
|
|
}
|
|
else if( n=="*PAPER_SPACE0" )
|
|
{
|
|
dw.sectionBlockEntryEnd( 0x25 );
|
|
}
|
|
else
|
|
{
|
|
dw.sectionBlockEntryEnd();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a viewport section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked VPORT section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeVPort( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "TABLE" );
|
|
dw.dxfString( 2, "VPORT" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 0x8 );
|
|
}
|
|
|
|
// dw.dxfHex(330, 0);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTable" );
|
|
}
|
|
|
|
dw.dxfInt( 70, 1 );
|
|
dw.dxfString( 0, "VPORT" );
|
|
|
|
// dw.dxfHex(5, 0x2F);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.handle();
|
|
}
|
|
|
|
// dw.dxfHex(330, 8);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbViewportTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, "*Active" );
|
|
dw.dxfInt( 70, 0 );
|
|
dw.dxfReal( 10, 0.0 );
|
|
dw.dxfReal( 20, 0.0 );
|
|
dw.dxfReal( 11, 1.0 );
|
|
dw.dxfReal( 21, 1.0 );
|
|
dw.dxfReal( 12, 286.3055555555555 );
|
|
dw.dxfReal( 22, 148.5 );
|
|
dw.dxfReal( 13, 0.0 );
|
|
dw.dxfReal( 23, 0.0 );
|
|
dw.dxfReal( 14, 10.0 );
|
|
dw.dxfReal( 24, 10.0 );
|
|
dw.dxfReal( 15, 10.0 );
|
|
dw.dxfReal( 25, 10.0 );
|
|
dw.dxfReal( 16, 0.0 );
|
|
dw.dxfReal( 26, 0.0 );
|
|
dw.dxfReal( 36, 1.0 );
|
|
dw.dxfReal( 17, 0.0 );
|
|
dw.dxfReal( 27, 0.0 );
|
|
dw.dxfReal( 37, 0.0 );
|
|
dw.dxfReal( 40, 297.0 );
|
|
dw.dxfReal( 41, 1.92798353909465 );
|
|
dw.dxfReal( 42, 50.0 );
|
|
dw.dxfReal( 43, 0.0 );
|
|
dw.dxfReal( 44, 0.0 );
|
|
dw.dxfReal( 50, 0.0 );
|
|
dw.dxfReal( 51, 0.0 );
|
|
dw.dxfInt( 71, 0 );
|
|
dw.dxfInt( 72, 100 );
|
|
dw.dxfInt( 73, 1 );
|
|
dw.dxfInt( 74, 3 );
|
|
dw.dxfInt( 75, 1 );
|
|
dw.dxfInt( 76, 1 );
|
|
dw.dxfInt( 77, 0 );
|
|
dw.dxfInt( 78, 0 );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfInt( 281, 0 );
|
|
dw.dxfInt( 65, 1 );
|
|
dw.dxfReal( 110, 0.0 );
|
|
dw.dxfReal( 120, 0.0 );
|
|
dw.dxfReal( 130, 0.0 );
|
|
dw.dxfReal( 111, 1.0 );
|
|
dw.dxfReal( 121, 0.0 );
|
|
dw.dxfReal( 131, 0.0 );
|
|
dw.dxfReal( 112, 0.0 );
|
|
dw.dxfReal( 122, 1.0 );
|
|
dw.dxfReal( 132, 0.0 );
|
|
dw.dxfInt( 79, 0 );
|
|
dw.dxfReal( 146, 0.0 );
|
|
}
|
|
|
|
dw.dxfString( 0, "ENDTAB" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a style section. This section is needed in DL_VERSION_R13.
|
|
*/
|
|
void DL_Dxf::writeStyle( DL_WriterA& dw, const DL_StyleData& style )
|
|
{
|
|
// dw.dxfString( 0, "TABLE");
|
|
// dw.dxfString( 2, "STYLE");
|
|
// if (version==DL_VERSION_2000) {
|
|
// dw.dxfHex(5, 3);
|
|
// }
|
|
// dw.dxfHex(330, 0);
|
|
// if (version==DL_VERSION_2000) {
|
|
// dw.dxfString(100, "AcDbSymbolTable");
|
|
// }
|
|
// dw.dxfInt( 70, 1);
|
|
dw.dxfString( 0, "STYLE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
if( style.name=="Standard" )
|
|
{
|
|
// dw.dxfHex(5, 0x11);
|
|
styleHandleStd = dw.handle();
|
|
}
|
|
else
|
|
{
|
|
dw.handle();
|
|
}
|
|
}
|
|
|
|
// dw.dxfHex(330, 3);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbTextStyleTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, style.name );
|
|
dw.dxfInt( 70, style.flags );
|
|
dw.dxfReal( 40, style.fixedTextHeight );
|
|
dw.dxfReal( 41, style.widthFactor );
|
|
dw.dxfReal( 50, style.obliqueAngle );
|
|
dw.dxfInt( 71, style.textGenerationFlags );
|
|
dw.dxfReal( 42, style.lastHeightUsed );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 3, "" );
|
|
dw.dxfString( 4, "" );
|
|
dw.dxfString( 1001, "ACAD" );
|
|
// dw.dxfString(1000, style.name);
|
|
dw.dxfString( 1000, style.primaryFontFile );
|
|
int xFlags = 0;
|
|
|
|
if( style.bold )
|
|
{
|
|
xFlags = xFlags | 0x2000000;
|
|
}
|
|
|
|
if( style.italic )
|
|
{
|
|
xFlags = xFlags | 0x1000000;
|
|
}
|
|
|
|
dw.dxfInt( 1071, xFlags );
|
|
}
|
|
else
|
|
{
|
|
dw.dxfString( 3, style.primaryFontFile );
|
|
dw.dxfString( 4, style.bigFontFile );
|
|
}
|
|
|
|
// dw.dxfString( 0, "ENDTAB");
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a view section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked VIEW section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeView( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "TABLE" );
|
|
dw.dxfString( 2, "VIEW" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 6 );
|
|
}
|
|
|
|
// dw.dxfHex(330, 0);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTable" );
|
|
}
|
|
|
|
dw.dxfInt( 70, 0 );
|
|
dw.dxfString( 0, "ENDTAB" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a ucs section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked UCS section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeUcs( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "TABLE" );
|
|
dw.dxfString( 2, "UCS" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 7 );
|
|
}
|
|
|
|
// dw.dxfHex(330, 0);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTable" );
|
|
}
|
|
|
|
dw.dxfInt( 70, 0 );
|
|
dw.dxfString( 0, "ENDTAB" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a dimstyle section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked DIMSTYLE section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeDimStyle( DL_WriterA& dw,
|
|
double dimasz, double dimexe, double dimexo,
|
|
double dimgap, double dimtxt )
|
|
{
|
|
dw.dxfString( 0, "TABLE" );
|
|
dw.dxfString( 2, "DIMSTYLE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 0xA );
|
|
dw.dxfString( 100, "AcDbSymbolTable" );
|
|
}
|
|
|
|
dw.dxfInt( 70, 1 );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbDimStyleTable" );
|
|
dw.dxfInt( 71, 0 );
|
|
}
|
|
|
|
|
|
dw.dxfString( 0, "DIMSTYLE" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 105, 0x27 );
|
|
}
|
|
|
|
// dw.handle(105);
|
|
// dw.dxfHex(330, 0xA);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbDimStyleTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, "Standard" );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfString( 3, "" );
|
|
dw.dxfString( 4, "" );
|
|
dw.dxfString( 5, "" );
|
|
dw.dxfString( 6, "" );
|
|
dw.dxfString( 7, "" );
|
|
dw.dxfReal( 40, 1.0 );
|
|
}
|
|
|
|
dw.dxfReal( 41, dimasz );
|
|
dw.dxfReal( 42, dimexo );
|
|
dw.dxfReal( 43, 3.75 );
|
|
dw.dxfReal( 44, dimexe );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfReal( 45, 0.0 );
|
|
dw.dxfReal( 46, 0.0 );
|
|
dw.dxfReal( 47, 0.0 );
|
|
dw.dxfReal( 48, 0.0 );
|
|
}
|
|
|
|
dw.dxfInt( 70, 0 );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 71, 0 );
|
|
dw.dxfInt( 72, 0 );
|
|
}
|
|
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfInt( 74, 0 );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 75, 0 );
|
|
dw.dxfInt( 76, 0 );
|
|
}
|
|
|
|
dw.dxfInt( 77, 1 );
|
|
dw.dxfInt( 78, 8 );
|
|
dw.dxfReal( 140, dimtxt );
|
|
dw.dxfReal( 141, 2.5 );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfReal( 142, 0.0 );
|
|
}
|
|
|
|
dw.dxfReal( 143, 0.03937007874016 );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfReal( 144, 1.0 );
|
|
dw.dxfReal( 145, 0.0 );
|
|
dw.dxfReal( 146, 1.0 );
|
|
}
|
|
|
|
dw.dxfReal( 147, dimgap );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 170, 0 );
|
|
}
|
|
|
|
dw.dxfInt( 171, 3 );
|
|
dw.dxfInt( 172, 1 );
|
|
|
|
if( version==DL_VERSION_R12 )
|
|
{
|
|
dw.dxfInt( 173, 0 );
|
|
dw.dxfInt( 174, 0 );
|
|
dw.dxfInt( 175, 0 );
|
|
dw.dxfInt( 176, 0 );
|
|
dw.dxfInt( 177, 0 );
|
|
dw.dxfInt( 178, 0 );
|
|
}
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfInt( 271, 2 );
|
|
dw.dxfInt( 272, 2 );
|
|
dw.dxfInt( 274, 3 );
|
|
dw.dxfInt( 278, 44 );
|
|
dw.dxfInt( 283, 0 );
|
|
dw.dxfInt( 284, 8 );
|
|
dw.dxfHex( 340, styleHandleStd );
|
|
// dw.dxfHex(340, 0x11);
|
|
}
|
|
|
|
// * /
|
|
dw.dxfString( 0, "ENDTAB" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a blockrecord section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked BLOCKRECORD section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeBlockRecord( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "TABLE" );
|
|
dw.dxfString( 2, "BLOCK_RECORD" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 1 );
|
|
}
|
|
|
|
// dw.dxfHex(330, 0);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTable" );
|
|
}
|
|
|
|
dw.dxfInt( 70, 1 );
|
|
|
|
dw.dxfString( 0, "BLOCK_RECORD" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 0x1F );
|
|
}
|
|
|
|
// int msh = dw.handle();
|
|
// dw.setModelSpaceHandle(msh);
|
|
// dw.dxfHex(330, 1);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbBlockTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, "*Model_Space" );
|
|
dw.dxfHex( 340, 0x22 );
|
|
|
|
dw.dxfString( 0, "BLOCK_RECORD" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 0x1B );
|
|
}
|
|
|
|
// int psh = dw.handle();
|
|
// dw.setPaperSpaceHandle(psh);
|
|
// dw.dxfHex(330, 1);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbBlockTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, "*Paper_Space" );
|
|
dw.dxfHex( 340, 0x1E );
|
|
|
|
dw.dxfString( 0, "BLOCK_RECORD" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfHex( 5, 0x23 );
|
|
}
|
|
|
|
// int ps0h = dw.handle();
|
|
// dw.setPaperSpace0Handle(ps0h);
|
|
// dw.dxfHex(330, 1);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbBlockTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, "*Paper_Space0" );
|
|
dw.dxfHex( 340, 0x26 );
|
|
|
|
// dw.dxfString( 0, "ENDTAB");
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a single block record with the given name.
|
|
*/
|
|
void DL_Dxf::writeBlockRecord( DL_WriterA& dw, const std::string& name )
|
|
{
|
|
dw.dxfString( 0, "BLOCK_RECORD" );
|
|
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.handle();
|
|
}
|
|
|
|
// dw->dxfHex(330, 1);
|
|
if( version==DL_VERSION_2000 )
|
|
{
|
|
dw.dxfString( 100, "AcDbSymbolTableRecord" );
|
|
dw.dxfString( 100, "AcDbBlockTableRecord" );
|
|
}
|
|
|
|
dw.dxfString( 2, name );
|
|
dw.dxfHex( 340, 0 );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a objects section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked OBJECTS section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeObjects( DL_WriterA& dw, const std::string& appDictionaryName )
|
|
{
|
|
dw.dxfString( 0, "SECTION" );
|
|
dw.dxfString( 2, "OBJECTS" );
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
dw.dxfHex( 5, 0xC );
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 281, 1 );
|
|
dw.dxfString( 3, "ACAD_GROUP" );
|
|
dw.dxfHex( 350, 0xD );
|
|
dw.dxfString( 3, "ACAD_LAYOUT" );
|
|
dw.dxfHex( 350, 0x1A );
|
|
dw.dxfString( 3, "ACAD_MLINESTYLE" );
|
|
dw.dxfHex( 350, 0x17 );
|
|
dw.dxfString( 3, "ACAD_PLOTSETTINGS" );
|
|
dw.dxfHex( 350, 0x19 );
|
|
dw.dxfString( 3, "ACAD_PLOTSTYLENAME" );
|
|
dw.dxfHex( 350, 0xE );
|
|
dw.dxfString( 3, "AcDbVariableDictionary" );
|
|
int acDbVariableDictionaryHandle = dw.handle( 350 );
|
|
// int acDbVariableDictionaryHandle = dw.getNextHandle();
|
|
// dw.dxfHex(350, acDbVariableDictionaryHandle);
|
|
// dw.incHandle();
|
|
|
|
if( appDictionaryName.length()!=0 )
|
|
{
|
|
dw.dxfString( 3, appDictionaryName );
|
|
appDictionaryHandle = dw.handle( 350 );
|
|
// appDictionaryHandle = dw.getNextHandle();
|
|
// dw.dxfHex(350, appDictionaryHandle);
|
|
// dw.incHandle();
|
|
}
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
dw.dxfHex( 5, 0xD );
|
|
// dw.handle(); // D
|
|
// dw.dxfHex(330, 0xC);
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 281, 1 );
|
|
|
|
|
|
dw.dxfString( 0, "ACDBDICTIONARYWDFLT" );
|
|
dw.dxfHex( 5, 0xE );
|
|
// dicId4 = dw.handle(); // E
|
|
// dw.dxfHex(330, 0xC); // C
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 281, 1 );
|
|
dw.dxfString( 3, "Normal" );
|
|
dw.dxfHex( 350, 0xF );
|
|
// dw.dxfHex(350, dw.getNextHandle()+5); // F
|
|
dw.dxfString( 100, "AcDbDictionaryWithDefault" );
|
|
dw.dxfHex( 340, 0xF );
|
|
// dw.dxfHex(340, dw.getNextHandle()+5); // F
|
|
|
|
|
|
dw.dxfString( 0, "ACDBPLACEHOLDER" );
|
|
dw.dxfHex( 5, 0xF );
|
|
// dw.handle(); // F
|
|
// dw.dxfHex(330, dicId4); // E
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
// dicId3 = dw.handle(); // 17
|
|
dw.dxfHex( 5, 0x17 );
|
|
// dw.dxfHex(330, 0xC); // C
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 281, 1 );
|
|
dw.dxfString( 3, "Standard" );
|
|
dw.dxfHex( 350, 0x18 );
|
|
// dw.dxfHex(350, dw.getNextHandle()+5); // 18
|
|
|
|
|
|
dw.dxfString( 0, "MLINESTYLE" );
|
|
dw.dxfHex( 5, 0x18 );
|
|
// dw.handle(); // 18
|
|
// dw.dxfHex(330, dicId3); // 17
|
|
dw.dxfString( 100, "AcDbMlineStyle" );
|
|
dw.dxfString( 2, "STANDARD" );
|
|
dw.dxfInt( 70, 0 );
|
|
dw.dxfString( 3, "" );
|
|
dw.dxfInt( 62, 256 );
|
|
dw.dxfReal( 51, 90.0 );
|
|
dw.dxfReal( 52, 90.0 );
|
|
dw.dxfInt( 71, 2 );
|
|
dw.dxfReal( 49, 0.5 );
|
|
dw.dxfInt( 62, 256 );
|
|
dw.dxfString( 6, "BYLAYER" );
|
|
dw.dxfReal( 49, -0.5 );
|
|
dw.dxfInt( 62, 256 );
|
|
dw.dxfString( 6, "BYLAYER" );
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
dw.dxfHex( 5, 0x19 );
|
|
// dw.handle(); // 17
|
|
// dw.dxfHex(330, 0xC); // C
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 281, 1 );
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
// dicId2 = dw.handle(); // 1A
|
|
dw.dxfHex( 5, 0x1A );
|
|
// dw.dxfHex(330, 0xC);
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 281, 1 );
|
|
dw.dxfString( 3, "Layout1" );
|
|
dw.dxfHex( 350, 0x1E );
|
|
// dw.dxfHex(350, dw.getNextHandle()+2); // 1E
|
|
dw.dxfString( 3, "Layout2" );
|
|
dw.dxfHex( 350, 0x26 );
|
|
// dw.dxfHex(350, dw.getNextHandle()+4); // 26
|
|
dw.dxfString( 3, "Model" );
|
|
dw.dxfHex( 350, 0x22 );
|
|
// dw.dxfHex(350, dw.getNextHandle()+5); // 22
|
|
|
|
|
|
dw.dxfString( 0, "LAYOUT" );
|
|
dw.dxfHex( 5, 0x1E );
|
|
// dw.handle(); // 1E
|
|
// dw.dxfHex(330, dicId2); // 1A
|
|
dw.dxfString( 100, "AcDbPlotSettings" );
|
|
dw.dxfString( 1, "" );
|
|
dw.dxfString( 2, "none_device" );
|
|
dw.dxfString( 4, "" );
|
|
dw.dxfString( 6, "" );
|
|
dw.dxfReal( 40, 0.0 );
|
|
dw.dxfReal( 41, 0.0 );
|
|
dw.dxfReal( 42, 0.0 );
|
|
dw.dxfReal( 43, 0.0 );
|
|
dw.dxfReal( 44, 0.0 );
|
|
dw.dxfReal( 45, 0.0 );
|
|
dw.dxfReal( 46, 0.0 );
|
|
dw.dxfReal( 47, 0.0 );
|
|
dw.dxfReal( 48, 0.0 );
|
|
dw.dxfReal( 49, 0.0 );
|
|
dw.dxfReal( 140, 0.0 );
|
|
dw.dxfReal( 141, 0.0 );
|
|
dw.dxfReal( 142, 1.0 );
|
|
dw.dxfReal( 143, 1.0 );
|
|
dw.dxfInt( 70, 688 );
|
|
dw.dxfInt( 72, 0 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfInt( 74, 5 );
|
|
dw.dxfString( 7, "" );
|
|
dw.dxfInt( 75, 16 );
|
|
dw.dxfReal( 147, 1.0 );
|
|
dw.dxfReal( 148, 0.0 );
|
|
dw.dxfReal( 149, 0.0 );
|
|
dw.dxfString( 100, "AcDbLayout" );
|
|
dw.dxfString( 1, "Layout1" );
|
|
dw.dxfInt( 70, 1 );
|
|
dw.dxfInt( 71, 1 );
|
|
dw.dxfReal( 10, 0.0 );
|
|
dw.dxfReal( 20, 0.0 );
|
|
dw.dxfReal( 11, 420.0 );
|
|
dw.dxfReal( 21, 297.0 );
|
|
dw.dxfReal( 12, 0.0 );
|
|
dw.dxfReal( 22, 0.0 );
|
|
dw.dxfReal( 32, 0.0 );
|
|
dw.dxfReal( 14, 1.000000000000000E+20 );
|
|
dw.dxfReal( 24, 1.000000000000000E+20 );
|
|
dw.dxfReal( 34, 1.000000000000000E+20 );
|
|
dw.dxfReal( 15, -1.000000000000000E+20 );
|
|
dw.dxfReal( 25, -1.000000000000000E+20 );
|
|
dw.dxfReal( 35, -1.000000000000000E+20 );
|
|
dw.dxfReal( 146, 0.0 );
|
|
dw.dxfReal( 13, 0.0 );
|
|
dw.dxfReal( 23, 0.0 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
dw.dxfReal( 16, 1.0 );
|
|
dw.dxfReal( 26, 0.0 );
|
|
dw.dxfReal( 36, 0.0 );
|
|
dw.dxfReal( 17, 0.0 );
|
|
dw.dxfReal( 27, 1.0 );
|
|
dw.dxfReal( 37, 0.0 );
|
|
dw.dxfInt( 76, 0 );
|
|
// dw.dxfHex(330, dw.getPaperSpaceHandle()); // 1B
|
|
dw.dxfHex( 330, 0x1B );
|
|
|
|
|
|
dw.dxfString( 0, "LAYOUT" );
|
|
dw.dxfHex( 5, 0x22 );
|
|
// dw.handle(); // 22
|
|
// dw.dxfHex(330, dicId2); // 1A
|
|
dw.dxfString( 100, "AcDbPlotSettings" );
|
|
dw.dxfString( 1, "" );
|
|
dw.dxfString( 2, "none_device" );
|
|
dw.dxfString( 4, "" );
|
|
dw.dxfString( 6, "" );
|
|
dw.dxfReal( 40, 0.0 );
|
|
dw.dxfReal( 41, 0.0 );
|
|
dw.dxfReal( 42, 0.0 );
|
|
dw.dxfReal( 43, 0.0 );
|
|
dw.dxfReal( 44, 0.0 );
|
|
dw.dxfReal( 45, 0.0 );
|
|
dw.dxfReal( 46, 0.0 );
|
|
dw.dxfReal( 47, 0.0 );
|
|
dw.dxfReal( 48, 0.0 );
|
|
dw.dxfReal( 49, 0.0 );
|
|
dw.dxfReal( 140, 0.0 );
|
|
dw.dxfReal( 141, 0.0 );
|
|
dw.dxfReal( 142, 1.0 );
|
|
dw.dxfReal( 143, 1.0 );
|
|
dw.dxfInt( 70, 1712 );
|
|
dw.dxfInt( 72, 0 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfInt( 74, 0 );
|
|
dw.dxfString( 7, "" );
|
|
dw.dxfInt( 75, 0 );
|
|
dw.dxfReal( 147, 1.0 );
|
|
dw.dxfReal( 148, 0.0 );
|
|
dw.dxfReal( 149, 0.0 );
|
|
dw.dxfString( 100, "AcDbLayout" );
|
|
dw.dxfString( 1, "Model" );
|
|
dw.dxfInt( 70, 1 );
|
|
dw.dxfInt( 71, 0 );
|
|
dw.dxfReal( 10, 0.0 );
|
|
dw.dxfReal( 20, 0.0 );
|
|
dw.dxfReal( 11, 12.0 );
|
|
dw.dxfReal( 21, 9.0 );
|
|
dw.dxfReal( 12, 0.0 );
|
|
dw.dxfReal( 22, 0.0 );
|
|
dw.dxfReal( 32, 0.0 );
|
|
dw.dxfReal( 14, 0.0 );
|
|
dw.dxfReal( 24, 0.0 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
dw.dxfReal( 15, 0.0 );
|
|
dw.dxfReal( 25, 0.0 );
|
|
dw.dxfReal( 35, 0.0 );
|
|
dw.dxfReal( 146, 0.0 );
|
|
dw.dxfReal( 13, 0.0 );
|
|
dw.dxfReal( 23, 0.0 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
dw.dxfReal( 16, 1.0 );
|
|
dw.dxfReal( 26, 0.0 );
|
|
dw.dxfReal( 36, 0.0 );
|
|
dw.dxfReal( 17, 0.0 );
|
|
dw.dxfReal( 27, 1.0 );
|
|
dw.dxfReal( 37, 0.0 );
|
|
dw.dxfInt( 76, 0 );
|
|
// dw.dxfHex(330, dw.getModelSpaceHandle()); // 1F
|
|
dw.dxfHex( 330, 0x1F );
|
|
|
|
|
|
dw.dxfString( 0, "LAYOUT" );
|
|
// dw.handle(); // 26
|
|
dw.dxfHex( 5, 0x26 );
|
|
// dw.dxfHex(330, dicId2); // 1A
|
|
dw.dxfString( 100, "AcDbPlotSettings" );
|
|
dw.dxfString( 1, "" );
|
|
dw.dxfString( 2, "none_device" );
|
|
dw.dxfString( 4, "" );
|
|
dw.dxfString( 6, "" );
|
|
dw.dxfReal( 40, 0.0 );
|
|
dw.dxfReal( 41, 0.0 );
|
|
dw.dxfReal( 42, 0.0 );
|
|
dw.dxfReal( 43, 0.0 );
|
|
dw.dxfReal( 44, 0.0 );
|
|
dw.dxfReal( 45, 0.0 );
|
|
dw.dxfReal( 46, 0.0 );
|
|
dw.dxfReal( 47, 0.0 );
|
|
dw.dxfReal( 48, 0.0 );
|
|
dw.dxfReal( 49, 0.0 );
|
|
dw.dxfReal( 140, 0.0 );
|
|
dw.dxfReal( 141, 0.0 );
|
|
dw.dxfReal( 142, 1.0 );
|
|
dw.dxfReal( 143, 1.0 );
|
|
dw.dxfInt( 70, 688 );
|
|
dw.dxfInt( 72, 0 );
|
|
dw.dxfInt( 73, 0 );
|
|
dw.dxfInt( 74, 5 );
|
|
dw.dxfString( 7, "" );
|
|
dw.dxfInt( 75, 16 );
|
|
dw.dxfReal( 147, 1.0 );
|
|
dw.dxfReal( 148, 0.0 );
|
|
dw.dxfReal( 149, 0.0 );
|
|
dw.dxfString( 100, "AcDbLayout" );
|
|
dw.dxfString( 1, "Layout2" );
|
|
dw.dxfInt( 70, 1 );
|
|
dw.dxfInt( 71, 2 );
|
|
dw.dxfReal( 10, 0.0 );
|
|
dw.dxfReal( 20, 0.0 );
|
|
dw.dxfReal( 11, 12.0 );
|
|
dw.dxfReal( 21, 9.0 );
|
|
dw.dxfReal( 12, 0.0 );
|
|
dw.dxfReal( 22, 0.0 );
|
|
dw.dxfReal( 32, 0.0 );
|
|
dw.dxfReal( 14, 0.0 );
|
|
dw.dxfReal( 24, 0.0 );
|
|
dw.dxfReal( 34, 0.0 );
|
|
dw.dxfReal( 15, 0.0 );
|
|
dw.dxfReal( 25, 0.0 );
|
|
dw.dxfReal( 35, 0.0 );
|
|
dw.dxfReal( 146, 0.0 );
|
|
dw.dxfReal( 13, 0.0 );
|
|
dw.dxfReal( 23, 0.0 );
|
|
dw.dxfReal( 33, 0.0 );
|
|
dw.dxfReal( 16, 1.0 );
|
|
dw.dxfReal( 26, 0.0 );
|
|
dw.dxfReal( 36, 0.0 );
|
|
dw.dxfReal( 17, 0.0 );
|
|
dw.dxfReal( 27, 1.0 );
|
|
dw.dxfReal( 37, 0.0 );
|
|
dw.dxfInt( 76, 0 );
|
|
// dw.dxfHex(330, dw.getPaperSpace0Handle()); // 23
|
|
dw.dxfHex( 330, 0x23 );
|
|
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
// dw.dxfHex(5, 0x2C);
|
|
// dicId5 =
|
|
dw.dxfHex( 5, acDbVariableDictionaryHandle );
|
|
// dw.handle(); // 2C
|
|
// dw.dxfHex(330, 0xC); // C
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 281, 1 );
|
|
dw.dxfString( 3, "DIMASSOC" );
|
|
// dw.dxfHex(350, 0x2F);
|
|
dw.dxfHex( 350, dw.getNextHandle() + 1 ); // 2E
|
|
dw.dxfString( 3, "HIDETEXT" );
|
|
// dw.dxfHex(350, 0x2E);
|
|
dw.dxfHex( 350, dw.getNextHandle() ); // 2D
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARYVAR" );
|
|
// dw.dxfHex(5, 0x2E);
|
|
dw.handle(); // 2E
|
|
// dw.dxfHex(330, dicId5); // 2C
|
|
dw.dxfString( 100, "DictionaryVariables" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 1, 2 );
|
|
|
|
|
|
dw.dxfString( 0, "DICTIONARYVAR" );
|
|
// dw.dxfHex(5, 0x2D);
|
|
dw.handle(); // 2D
|
|
// dw.dxfHex(330, dicId5); // 2C
|
|
dw.dxfString( 100, "DictionaryVariables" );
|
|
dw.dxfInt( 280, 0 );
|
|
dw.dxfInt( 1, 1 );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeAppDictionary( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "DICTIONARY" );
|
|
// dw.handle();
|
|
dw.dxfHex( 5, appDictionaryHandle );
|
|
dw.dxfString( 100, "AcDbDictionary" );
|
|
dw.dxfInt( 281, 1 );
|
|
}
|
|
|
|
|
|
int DL_Dxf::writeDictionaryEntry( DL_WriterA& dw, const std::string& name )
|
|
{
|
|
dw.dxfString( 3, name );
|
|
int handle = dw.getNextHandle();
|
|
dw.dxfHex( 350, handle );
|
|
dw.incHandle();
|
|
return handle;
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeXRecord( DL_WriterA& dw, int handle, int value )
|
|
{
|
|
dw.dxfString( 0, "XRECORD" );
|
|
dw.dxfHex( 5, handle );
|
|
dw.dxfHex( 330, appDictionaryHandle );
|
|
dw.dxfString( 100, "AcDbXrecord" );
|
|
dw.dxfInt( 280, 1 );
|
|
dw.dxfInt( 90, value );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeXRecord( DL_WriterA& dw, int handle, double value )
|
|
{
|
|
dw.dxfString( 0, "XRECORD" );
|
|
dw.dxfHex( 5, handle );
|
|
dw.dxfHex( 330, appDictionaryHandle );
|
|
dw.dxfString( 100, "AcDbXrecord" );
|
|
dw.dxfInt( 280, 1 );
|
|
dw.dxfReal( 40, value );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeXRecord( DL_WriterA& dw, int handle, bool value )
|
|
{
|
|
dw.dxfString( 0, "XRECORD" );
|
|
dw.dxfHex( 5, handle );
|
|
dw.dxfHex( 330, appDictionaryHandle );
|
|
dw.dxfString( 100, "AcDbXrecord" );
|
|
dw.dxfInt( 280, 1 );
|
|
dw.dxfBool( 290, value );
|
|
}
|
|
|
|
|
|
void DL_Dxf::writeXRecord( DL_WriterA& dw, int handle, const std::string& value )
|
|
{
|
|
dw.dxfString( 0, "XRECORD" );
|
|
dw.dxfHex( 5, handle );
|
|
dw.dxfHex( 330, appDictionaryHandle );
|
|
dw.dxfString( 100, "AcDbXrecord" );
|
|
dw.dxfInt( 280, 1 );
|
|
dw.dxfString( 1000, value );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes the end of the objects section. This section is needed in DL_VERSION_R13.
|
|
* Note that this method currently only writes a faked OBJECTS section
|
|
* to make the file readable by Aut*cad.
|
|
*/
|
|
void DL_Dxf::writeObjectsEnd( DL_WriterA& dw )
|
|
{
|
|
dw.dxfString( 0, "ENDSEC" );
|
|
}
|
|
|
|
|
|
/**
|
|
* Writes a comment to the DXF file.
|
|
*/
|
|
void DL_Dxf::writeComment( DL_WriterA& dw, const std::string& comment )
|
|
{
|
|
dw.dxfString( 999, comment );
|
|
}
|
|
|
|
|
|
/**
|
|
* Checks if the given variable is known by the given DXF version.
|
|
*/
|
|
bool DL_Dxf::checkVariable( const char* var, DL_Codes::version version )
|
|
{
|
|
if( version>=DL_VERSION_2000 )
|
|
{
|
|
return true;
|
|
}
|
|
else if( version==DL_VERSION_R12 )
|
|
{
|
|
// these are all the variables recognized by dxf r12:
|
|
if( !strcmp( var, "$ACADVER" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ACADVER" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ANGBASE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ANGDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ATTDIA" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ATTMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ATTREQ" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$AUNITS" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$AUPREC" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$AXISMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$AXISUNIT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$BLIPMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$CECOLOR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$CELTYPE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$CHAMFERA" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$CHAMFERB" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$CLAYER" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$COORDS" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMALT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMALTD" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMALTF" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMAPOST" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMASO" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMASZ" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMBLK" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMBLK1" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMBLK2" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMCEN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMCLRD" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMCLRE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMCLRT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMDLE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMDLI" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMEXE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMEXO" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMGAP" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMLFAC" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMLIM" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMPOST" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMRND" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSAH" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSCALE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSE1" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSE2" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSHO" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSOXD" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMSTYLE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTAD" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTFAC" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTIH" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTIX" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTM" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTOFL" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTOH" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTOL" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTP" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTSZ" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTVP" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMTXT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DIMZIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DWGCODEPAGE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$DRAGMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ELEVATION" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$EXTMAX" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$EXTMIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$FILLETRAD" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$FILLMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$HANDLING" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$HANDSEED" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$INSBASE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LIMCHECK" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LIMMAX" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LIMMIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LTSCALE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LUNITS" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$LUPREC" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$MAXACTVP" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$MENU" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$MIRRTEXT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$ORTHOMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$OSMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PDMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PDSIZE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PELEVATION" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PEXTMAX" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PEXTMIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PLIMCHECK" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PLIMMAX" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PLIMMIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PLINEGEN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PLINEWID" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PSLTSCALE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PUCSNAME" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PUCSORG" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PUCSXDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$PUCSYDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$QTEXTMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$REGENMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SHADEDGE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SHADEDIF" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SKETCHINC" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SKPOLY" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SPLFRAME" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SPLINESEGS" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SPLINETYPE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SURFTAB1" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SURFTAB2" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SURFTYPE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SURFU" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SURFV" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TDCREATE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TDINDWG" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TDUPDATE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TDUSRTIMER" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TEXTSIZE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TEXTSTYLE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$THICKNESS" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TILEMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$TRACEWID" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$UCSNAME" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$UCSORG" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$UCSXDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$UCSYDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$UNITMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$USERI1" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$USERR1" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$USRTIMER" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$VISRETAIN" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$WORLDVIEW" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$FASTZOOM" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$GRIDMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$GRIDUNIT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPANG" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPBASE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPISOPAIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPMODE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPSTYLE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$SNAPUNIT" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$VIEWCTR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$VIEWDIR" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if( !strcmp( var, "$VIEWSIZE" ) )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/**
|
|
* @returns the library version as int (4 bytes, each byte one version number).
|
|
* e.g. if str = "2.0.2.0" getLibVersion returns 0x02000200
|
|
*/
|
|
int DL_Dxf::getLibVersion( const std::string& str )
|
|
{
|
|
int d[4];
|
|
int idx = 0;
|
|
// char v[4][5];
|
|
std::string v[4];
|
|
int ret = 0;
|
|
|
|
for( unsigned int i = 0; i<str.length() && idx<3; ++i )
|
|
{
|
|
if( str[i]=='.' )
|
|
{
|
|
d[idx] = i;
|
|
idx++;
|
|
}
|
|
}
|
|
|
|
if( idx>=2 )
|
|
{
|
|
d[3] = str.length();
|
|
|
|
v[0] = str.substr( 0, d[0] );
|
|
v[1] = str.substr( d[0] + 1, d[1] - d[0] - 1 );
|
|
v[2] = str.substr( d[1] + 1, d[2] - d[1] - 1 );
|
|
|
|
if( idx>=3 )
|
|
{
|
|
v[3] = str.substr( d[2] + 1, d[3] - d[2] - 1 );
|
|
}
|
|
else
|
|
{
|
|
v[3] = "0";
|
|
}
|
|
|
|
ret = ( atoi( v[0].c_str() ) << (3 * 8) ) +
|
|
( atoi( v[1].c_str() ) << (2 * 8) ) +
|
|
( atoi( v[2].c_str() ) << (1 * 8) ) +
|
|
( atoi( v[3].c_str() ) << (0 * 8) );
|
|
|
|
return ret;
|
|
}
|
|
else
|
|
{
|
|
std::cerr << "DL_Dxf::getLibVersion: invalid version number: " << str << "\n";
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Converts the given string into a double or returns the given
|
|
* default valud (def) if value is NULL or empty.
|
|
*/
|
|
// double DL_Dxf::toReal(const char* value, double def) {
|
|
// if (value!=NULL && value[0] != '\0') {
|
|
// printf("toReal: not empty: %s\n", value);
|
|
// printf("toReal: val: %f\n", atof(value));
|
|
// printf("toReal: 0: %d\n", value[0]);
|
|
// printf("toReal: 1: %d\n", value[1]);
|
|
// printf("toReal: 2: %d\n", value[2]);
|
|
// double ret;
|
|
// if (strchr(value, ',') != NULL) {
|
|
// char* tmp = new char[strlen(value)+1];
|
|
// strcpy(tmp, value);
|
|
// DL_WriterA::strReplace(tmp, ',', '.');
|
|
// ret = atof(tmp);
|
|
// delete[] tmp;
|
|
// }
|
|
// else {
|
|
// ret = atof(value);
|
|
// }
|
|
// return ret;
|
|
// } else {
|
|
// return def;
|
|
// }
|
|
// }
|
|
|
|
|
|
/**
|
|
* Some test routines.
|
|
*/
|
|
void DL_Dxf::test()
|
|
{
|
|
char* buf1;
|
|
char* buf2;
|
|
char* buf3;
|
|
char* buf4;
|
|
char* buf5;
|
|
char* buf6;
|
|
|
|
buf1 = new char[10];
|
|
buf2 = new char[10];
|
|
buf3 = new char[10];
|
|
buf4 = new char[10];
|
|
buf5 = new char[10];
|
|
buf6 = new char[10];
|
|
|
|
strcpy( buf1, " 10\n" );
|
|
strcpy( buf2, "10" );
|
|
strcpy( buf3, "10\n" );
|
|
strcpy( buf4, " 10 \n" );
|
|
strcpy( buf5, " 10 \r" );
|
|
strcpy( buf6, "\t10 \n" );
|
|
|
|
std::cout << "1 buf1: '" << buf1 << "'\n";
|
|
stripWhiteSpace( &buf1 );
|
|
std::cout << "2 buf1: '" << buf1 << "'\n";
|
|
// assert(!strcmp(buf1, "10"));
|
|
|
|
std::cout << "1 buf2: '" << buf2 << "'\n";
|
|
stripWhiteSpace( &buf2 );
|
|
std::cout << "2 buf2: '" << buf2 << "'\n";
|
|
|
|
std::cout << "1 buf3: '" << buf3 << "'\n";
|
|
stripWhiteSpace( &buf3 );
|
|
std::cout << "2 buf3: '" << buf3 << "'\n";
|
|
|
|
std::cout << "1 buf4: '" << buf4 << "'\n";
|
|
stripWhiteSpace( &buf4 );
|
|
std::cout << "2 buf4: '" << buf4 << "'\n";
|
|
|
|
std::cout << "1 buf5: '" << buf5 << "'\n";
|
|
stripWhiteSpace( &buf5 );
|
|
std::cout << "2 buf5: '" << buf5 << "'\n";
|
|
|
|
std::cout << "1 buf6: '" << buf6 << "'\n";
|
|
stripWhiteSpace( &buf6 );
|
|
std::cout << "2 buf6: '" << buf6 << "'\n";
|
|
|
|
delete( buf1 );
|
|
delete( buf2 );
|
|
delete( buf3 );
|
|
delete( buf4 );
|
|
delete( buf5 );
|
|
delete( buf6 );
|
|
}
|