2012-05-03 19:08:14 +00:00
|
|
|
/**
|
2018-01-28 18:12:26 +00:00
|
|
|
* @file PDF_plotter.cpp
|
|
|
|
* @brief Kicad: specialized plotter for PDF files format
|
2012-05-03 19:08:14 +00:00
|
|
|
*/
|
|
|
|
|
2012-06-08 09:56:42 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 1992-2012 Lorenzo Marcantonio, l.marcantonio@logossrl.com
|
2017-01-20 21:00:20 +00:00
|
|
|
* Copyright (C) 1992-2017 KiCad Developers, see AUTHORS.txt for contributors.
|
2012-06-08 09:56:42 +00:00
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <trigo.h>
|
2018-01-29 15:39:40 +00:00
|
|
|
#include <eda_base_frame.h>
|
2012-05-03 19:08:14 +00:00
|
|
|
#include <base_struct.h>
|
|
|
|
#include <common.h>
|
2018-01-28 18:12:26 +00:00
|
|
|
#include <plotter.h>
|
2012-05-03 19:08:14 +00:00
|
|
|
#include <macros.h>
|
2012-05-05 04:55:36 +00:00
|
|
|
#include <wx/zstream.h>
|
|
|
|
#include <wx/mstream.h>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <math/util.h> // for KiROUND
|
2012-05-03 19:08:14 +00:00
|
|
|
|
2020-07-11 09:37:22 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2012-10-13 18:54:33 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Open or create the plot file aFullFilename
|
|
|
|
* return true if success, false if the file cannot be created/opened
|
|
|
|
*
|
|
|
|
* Opens the PDF file in binary mode
|
|
|
|
*/
|
|
|
|
bool PDF_PLOTTER::OpenFile( const wxString& aFullFilename )
|
|
|
|
{
|
|
|
|
filename = aFullFilename;
|
|
|
|
|
|
|
|
wxASSERT( !outputFile );
|
|
|
|
|
|
|
|
// Open the PDF file in binary mode
|
|
|
|
outputFile = wxFopen( filename, wxT( "wb" ) );
|
|
|
|
|
|
|
|
if( outputFile == NULL )
|
|
|
|
return false ;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
void PDF_PLOTTER::SetViewport( const wxPoint& aOffset, double aIusPerDecimil,
|
|
|
|
double aScale, bool aMirror )
|
|
|
|
{
|
2013-12-06 18:31:15 +00:00
|
|
|
m_plotMirror = aMirror;
|
2012-05-03 19:08:14 +00:00
|
|
|
plotOffset = aOffset;
|
|
|
|
plotScale = aScale;
|
2012-08-29 20:13:47 +00:00
|
|
|
m_IUsPerDecimil = aIusPerDecimil;
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
// The CTM is set to 1 user unit per decimil
|
|
|
|
iuPerDeviceUnit = 1.0 / aIusPerDecimil;
|
|
|
|
|
|
|
|
/* The paper size in this engined is handled page by page
|
|
|
|
Look in the StartPage function */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pen width setting for PDF. Since the specs *explicitly* says that a 0
|
|
|
|
* width is a bad thing to use (since it results in 1 pixel traces), we
|
2015-05-21 09:04:47 +00:00
|
|
|
* convert such requests to the minimal width (like 1)
|
|
|
|
* Note pen width = 0 is used in plot polygons to plot filled polygons with
|
|
|
|
* no outline thickness
|
|
|
|
* use in this case pen width = 1 does not actally change the polygon
|
2012-05-03 19:08:14 +00:00
|
|
|
*/
|
2020-04-14 12:25:00 +00:00
|
|
|
void PDF_PLOTTER::SetCurrentLineWidth( int aWidth, void* aData )
|
2012-05-03 19:08:14 +00:00
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
|
2020-05-13 16:44:21 +00:00
|
|
|
if( aWidth == DO_NOT_SET_LINE_WIDTH )
|
|
|
|
return;
|
|
|
|
else if( aWidth == USE_DEFAULT_LINE_WIDTH )
|
|
|
|
aWidth = m_renderSettings->GetDefaultPenWidth();
|
2020-05-27 01:17:16 +00:00
|
|
|
|
|
|
|
if( aWidth == 0 )
|
2020-04-14 12:25:00 +00:00
|
|
|
aWidth = 1;
|
2012-05-03 19:08:14 +00:00
|
|
|
|
2020-05-13 16:44:21 +00:00
|
|
|
wxASSERT_MSG( aWidth > 0, "Plotter called to set negative pen width" );
|
|
|
|
|
2020-04-14 12:25:00 +00:00
|
|
|
if( aWidth != currentPenWidth )
|
|
|
|
fprintf( workFile, "%g w\n", userToDeviceSize( aWidth ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
|
2020-04-14 12:25:00 +00:00
|
|
|
currentPenWidth = aWidth;
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PDF supports colors fully. It actually has distinct fill and pen colors,
|
|
|
|
* but we set both at the same time.
|
|
|
|
*
|
|
|
|
* XXX Keeping them divided could result in a minor optimization in
|
|
|
|
* eeschema filled shapes, but would propagate to all the other plot
|
|
|
|
* engines. Also arcs are filled as pies but only the arc is stroked so
|
|
|
|
* it would be difficult to handle anyway.
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::emitSetRGBColor( double r, double g, double b )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
fprintf( workFile, "%g %g %g rg %g %g %g RG\n",
|
|
|
|
r, g, b, r, g, b );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PDF supports dashed lines
|
|
|
|
*/
|
2019-12-28 00:55:11 +00:00
|
|
|
void PDF_PLOTTER::SetDash( PLOT_DASH_TYPE dashed )
|
2012-05-03 19:08:14 +00:00
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
2017-11-13 03:53:27 +00:00
|
|
|
switch( dashed )
|
|
|
|
{
|
2019-12-28 00:55:11 +00:00
|
|
|
case PLOT_DASH_TYPE::DASH:
|
2015-03-10 20:00:50 +00:00
|
|
|
fprintf( workFile, "[%d %d] 0 d\n",
|
2017-11-13 03:53:27 +00:00
|
|
|
(int) GetDashMarkLenIU(), (int) GetDashGapLenIU() );
|
|
|
|
break;
|
2019-12-28 00:55:11 +00:00
|
|
|
case PLOT_DASH_TYPE::DOT:
|
2017-11-13 03:53:27 +00:00
|
|
|
fprintf( workFile, "[%d %d] 0 d\n",
|
|
|
|
(int) GetDotMarkLenIU(), (int) GetDashGapLenIU() );
|
|
|
|
break;
|
2019-12-28 00:55:11 +00:00
|
|
|
case PLOT_DASH_TYPE::DASHDOT:
|
2017-11-13 03:53:27 +00:00
|
|
|
fprintf( workFile, "[%d %d %d %d] 0 d\n",
|
|
|
|
(int) GetDashMarkLenIU(), (int) GetDashGapLenIU(),
|
|
|
|
(int) GetDotMarkLenIU(), (int) GetDashGapLenIU() );
|
|
|
|
break;
|
|
|
|
default:
|
2012-05-03 19:08:14 +00:00
|
|
|
fputs( "[] 0 d\n", workFile );
|
2017-11-13 03:53:27 +00:00
|
|
|
}
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Rectangles in PDF. Supported by the native operator
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::Rect( const wxPoint& p1, const wxPoint& p2, FILL_T fill, int width )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
DPOINT p1_dev = userToDeviceCoordinates( p1 );
|
|
|
|
DPOINT p2_dev = userToDeviceCoordinates( p2 );
|
|
|
|
|
|
|
|
SetCurrentLineWidth( width );
|
|
|
|
fprintf( workFile, "%g %g %g %g re %c\n", p1_dev.x, p1_dev.y,
|
|
|
|
p2_dev.x - p1_dev.x, p2_dev.y - p1_dev.y,
|
|
|
|
fill == NO_FILL ? 'S' : 'B' );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Circle drawing for PDF. They're approximated by curves, but fill is supported
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::Circle( const wxPoint& pos, int diametre, FILL_T aFill, int width )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
DPOINT pos_dev = userToDeviceCoordinates( pos );
|
|
|
|
double radius = userToDeviceSize( diametre / 2.0 );
|
|
|
|
|
|
|
|
/* OK. Here's a trick. PDF doesn't support circles or circular angles, that's
|
|
|
|
a fact. You'll have to do with cubic beziers. These *can't* represent
|
|
|
|
circular arcs (NURBS can, beziers don't). But there is a widely known
|
2015-03-18 19:50:42 +00:00
|
|
|
approximation which is really good
|
|
|
|
*/
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
SetCurrentLineWidth( width );
|
2018-03-14 01:01:01 +00:00
|
|
|
|
|
|
|
// If diameter is less than width, switch to filled mode
|
|
|
|
if( aFill == NO_FILL && diametre < width )
|
|
|
|
{
|
|
|
|
aFill = FILLED_SHAPE;
|
|
|
|
SetCurrentLineWidth( 0 );
|
|
|
|
|
|
|
|
radius = userToDeviceSize( ( diametre / 2.0 ) + ( width / 2.0 ) );
|
|
|
|
}
|
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
double magic = radius * 0.551784; // You don't want to know where this come from
|
|
|
|
|
|
|
|
// This is the convex hull for the bezier approximated circle
|
|
|
|
fprintf( workFile, "%g %g m "
|
|
|
|
"%g %g %g %g %g %g c "
|
|
|
|
"%g %g %g %g %g %g c "
|
|
|
|
"%g %g %g %g %g %g c "
|
|
|
|
"%g %g %g %g %g %g c %c\n",
|
|
|
|
pos_dev.x - radius, pos_dev.y,
|
|
|
|
|
|
|
|
pos_dev.x - radius, pos_dev.y + magic,
|
|
|
|
pos_dev.x - magic, pos_dev.y + radius,
|
|
|
|
pos_dev.x, pos_dev.y + radius,
|
|
|
|
|
|
|
|
pos_dev.x + magic, pos_dev.y + radius,
|
|
|
|
pos_dev.x + radius, pos_dev.y + magic,
|
|
|
|
pos_dev.x + radius, pos_dev.y,
|
|
|
|
|
|
|
|
pos_dev.x + radius, pos_dev.y - magic,
|
|
|
|
pos_dev.x + magic, pos_dev.y - radius,
|
|
|
|
pos_dev.x, pos_dev.y - radius,
|
|
|
|
|
|
|
|
pos_dev.x - magic, pos_dev.y - radius,
|
|
|
|
pos_dev.x - radius, pos_dev.y - magic,
|
|
|
|
pos_dev.x - radius, pos_dev.y,
|
|
|
|
|
|
|
|
aFill == NO_FILL ? 's' : 'b' );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The PDF engine can't directly plot arcs, it uses the base emulation.
|
|
|
|
* So no filled arcs (not a great loss... )
|
|
|
|
*/
|
2013-05-05 07:17:48 +00:00
|
|
|
void PDF_PLOTTER::Arc( const wxPoint& centre, double StAngle, double EndAngle, int radius,
|
2012-05-03 19:08:14 +00:00
|
|
|
FILL_T fill, int width )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
if( radius <= 0 )
|
2018-03-14 01:01:01 +00:00
|
|
|
{
|
|
|
|
Circle( centre, width, FILLED_SHAPE, 0 );
|
2012-05-03 19:08:14 +00:00
|
|
|
return;
|
2018-03-14 01:01:01 +00:00
|
|
|
}
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
/* Arcs are not so easily approximated by beziers (in the general case),
|
|
|
|
so we approximate them in the old way */
|
|
|
|
wxPoint start, end;
|
|
|
|
const int delta = 50; // increment (in 0.1 degrees) to draw circles
|
|
|
|
|
|
|
|
if( StAngle > EndAngle )
|
2015-06-26 13:41:56 +00:00
|
|
|
std::swap( StAngle, EndAngle );
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
SetCurrentLineWidth( width );
|
|
|
|
|
|
|
|
// Usual trig arc plotting routine...
|
2013-05-02 18:06:58 +00:00
|
|
|
start.x = centre.x + KiROUND( cosdecideg( radius, -StAngle ) );
|
|
|
|
start.y = centre.y + KiROUND( sindecideg( radius, -StAngle ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
DPOINT pos_dev = userToDeviceCoordinates( start );
|
|
|
|
fprintf( workFile, "%g %g m ", pos_dev.x, pos_dev.y );
|
|
|
|
for( int ii = StAngle + delta; ii < EndAngle; ii += delta )
|
|
|
|
{
|
2013-05-02 18:06:58 +00:00
|
|
|
end.x = centre.x + KiROUND( cosdecideg( radius, -ii ) );
|
|
|
|
end.y = centre.y + KiROUND( sindecideg( radius, -ii ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
pos_dev = userToDeviceCoordinates( end );
|
|
|
|
fprintf( workFile, "%g %g l ", pos_dev.x, pos_dev.y );
|
|
|
|
}
|
|
|
|
|
2013-05-02 18:06:58 +00:00
|
|
|
end.x = centre.x + KiROUND( cosdecideg( radius, -EndAngle ) );
|
|
|
|
end.y = centre.y + KiROUND( sindecideg( radius, -EndAngle ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
pos_dev = userToDeviceCoordinates( end );
|
|
|
|
fprintf( workFile, "%g %g l ", pos_dev.x, pos_dev.y );
|
|
|
|
|
|
|
|
// The arc is drawn... if not filled we stroke it, otherwise we finish
|
|
|
|
// closing the pie at the center
|
|
|
|
if( fill == NO_FILL )
|
|
|
|
{
|
|
|
|
fputs( "S\n", workFile );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pos_dev = userToDeviceCoordinates( centre );
|
|
|
|
fprintf( workFile, "%g %g l b\n", pos_dev.x, pos_dev.y );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Polygon plotting for PDF. Everything is supported
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::PlotPoly( const std::vector< wxPoint >& aCornerList,
|
2016-09-19 11:01:36 +00:00
|
|
|
FILL_T aFill, int aWidth, void * aData )
|
2012-05-03 19:08:14 +00:00
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
if( aCornerList.size() <= 1 )
|
|
|
|
return;
|
|
|
|
|
|
|
|
SetCurrentLineWidth( aWidth );
|
|
|
|
|
|
|
|
DPOINT pos = userToDeviceCoordinates( aCornerList[0] );
|
|
|
|
fprintf( workFile, "%g %g m\n", pos.x, pos.y );
|
|
|
|
|
|
|
|
for( unsigned ii = 1; ii < aCornerList.size(); ii++ )
|
|
|
|
{
|
|
|
|
pos = userToDeviceCoordinates( aCornerList[ii] );
|
|
|
|
fprintf( workFile, "%g %g l\n", pos.x, pos.y );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close path and stroke(/fill)
|
|
|
|
fprintf( workFile, "%c\n", aFill == NO_FILL ? 'S' : 'b' );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PDF_PLOTTER::PenTo( const wxPoint& pos, char plume )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
if( plume == 'Z' )
|
|
|
|
{
|
|
|
|
if( penState != 'Z' )
|
|
|
|
{
|
|
|
|
fputs( "S\n", workFile );
|
|
|
|
penState = 'Z';
|
|
|
|
penLastpos.x = -1;
|
|
|
|
penLastpos.y = -1;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( penState != plume || pos != penLastpos )
|
|
|
|
{
|
|
|
|
DPOINT pos_dev = userToDeviceCoordinates( pos );
|
|
|
|
fprintf( workFile, "%g %g %c\n",
|
|
|
|
pos_dev.x, pos_dev.y,
|
|
|
|
( plume=='D' ) ? 'l' : 'm' );
|
|
|
|
}
|
|
|
|
penState = plume;
|
|
|
|
penLastpos = pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PDF images are handles as inline, not XObject streams...
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::PlotImage( const wxImage & aImage, const wxPoint& aPos,
|
|
|
|
double aScaleFactor )
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
wxSize pix_size( aImage.GetWidth(), aImage.GetHeight() );
|
|
|
|
|
|
|
|
// Requested size (in IUs)
|
|
|
|
DPOINT drawsize( aScaleFactor * pix_size.x,
|
|
|
|
aScaleFactor * pix_size.y );
|
|
|
|
|
|
|
|
// calculate the bitmap start position
|
|
|
|
wxPoint start( aPos.x - drawsize.x / 2,
|
|
|
|
aPos.y + drawsize.y / 2);
|
|
|
|
|
|
|
|
DPOINT dev_start = userToDeviceCoordinates( start );
|
|
|
|
|
|
|
|
/* PDF has an uhm... simplified coordinate system handling. There is
|
|
|
|
*one* operator to do everything (the PS concat equivalent). At least
|
|
|
|
they kept the matrix stack to save restore environments. Also images
|
|
|
|
are always emitted at the origin with a size of 1x1 user units.
|
|
|
|
What we need to do is:
|
2019-08-20 17:22:30 +00:00
|
|
|
1) save the CTM end establish the new one
|
2012-05-03 19:08:14 +00:00
|
|
|
2) plot the image
|
|
|
|
3) restore the CTM
|
|
|
|
4) profit
|
|
|
|
*/
|
|
|
|
fprintf( workFile, "q %g 0 0 %g %g %g cm\n", // Step 1
|
|
|
|
userToDeviceSize( drawsize.x ),
|
|
|
|
userToDeviceSize( drawsize.y ),
|
|
|
|
dev_start.x, dev_start.y );
|
|
|
|
|
|
|
|
/* An inline image is a cross between a dictionary and a stream.
|
|
|
|
A real ugly construct (compared with the elegance of the PDF
|
|
|
|
format). Also it accepts some 'abbreviations', which is stupid
|
|
|
|
since the content stream is usually compressed anyway... */
|
|
|
|
fprintf( workFile,
|
|
|
|
"BI\n"
|
|
|
|
" /BPC 8\n"
|
|
|
|
" /CS %s\n"
|
|
|
|
" /W %d\n"
|
|
|
|
" /H %d\n"
|
|
|
|
"ID\n", colorMode ? "/RGB" : "/G", pix_size.x, pix_size.y );
|
|
|
|
|
|
|
|
/* Here comes the stream (in binary!). I *could* have hex or ascii84
|
|
|
|
encoded it, but who cares? I'll go through zlib anyway */
|
|
|
|
for( int y = 0; y < pix_size.y; y++ )
|
|
|
|
{
|
|
|
|
for( int x = 0; x < pix_size.x; x++ )
|
|
|
|
{
|
|
|
|
unsigned char r = aImage.GetRed( x, y ) & 0xFF;
|
|
|
|
unsigned char g = aImage.GetGreen( x, y ) & 0xFF;
|
|
|
|
unsigned char b = aImage.GetBlue( x, y ) & 0xFF;
|
2019-05-27 16:35:05 +00:00
|
|
|
|
|
|
|
// PDF inline images don't support alpha, so premultiply against white background
|
|
|
|
if( aImage.HasAlpha() )
|
|
|
|
{
|
|
|
|
unsigned char alpha = aImage.GetAlpha( x, y ) & 0xFF;
|
|
|
|
|
|
|
|
if( alpha < 0xFF )
|
|
|
|
{
|
|
|
|
float a = 1.0 - ( (float) alpha / 255.0 );
|
|
|
|
r = ( int )( r + ( a * 0xFF ) ) & 0xFF;
|
|
|
|
g = ( int )( g + ( a * 0xFF ) ) & 0xFF;
|
|
|
|
b = ( int )( b + ( a * 0xFF ) ) & 0xFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-16 20:27:58 +00:00
|
|
|
if( aImage.HasMask() )
|
|
|
|
{
|
|
|
|
if( r == aImage.GetMaskRed() && g == aImage.GetMaskGreen() && b == aImage.GetMaskBlue() )
|
|
|
|
{
|
|
|
|
r = 0xFF;
|
|
|
|
g = 0xFF;
|
|
|
|
b = 0xFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
// As usual these days, stdio buffering has to suffeeeeerrrr
|
|
|
|
if( colorMode )
|
|
|
|
{
|
2019-06-16 20:27:58 +00:00
|
|
|
putc( r, workFile );
|
|
|
|
putc( g, workFile );
|
|
|
|
putc( b, workFile );
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-18 01:35:41 +00:00
|
|
|
// Greyscale conversion (CIE 1931)
|
|
|
|
unsigned char grey = KiROUND( r * 0.2126 + g * 0.7152 + b * 0.0722 );
|
|
|
|
putc( grey, workFile );
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fputs( "EI Q\n", workFile ); // Finish step 2 and do step 3
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allocate a new handle in the table of the PDF object. The
|
|
|
|
* handle must be completed using startPdfObject. It's an in-RAM operation
|
|
|
|
* only, no output is done.
|
|
|
|
*/
|
|
|
|
int PDF_PLOTTER::allocPdfObject()
|
|
|
|
{
|
|
|
|
xrefTable.push_back( 0 );
|
|
|
|
return xrefTable.size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Open a new PDF object and returns the handle if the parameter is -1.
|
|
|
|
* Otherwise fill in the xref entry for the passed object
|
|
|
|
*/
|
|
|
|
int PDF_PLOTTER::startPdfObject(int handle)
|
|
|
|
{
|
|
|
|
wxASSERT( outputFile );
|
|
|
|
wxASSERT( !workFile );
|
2017-04-21 12:16:40 +00:00
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
if( handle < 0)
|
|
|
|
handle = allocPdfObject();
|
|
|
|
|
|
|
|
xrefTable[handle] = ftell( outputFile );
|
|
|
|
fprintf( outputFile, "%d 0 obj\n", handle );
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Close the current PDF object
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::closePdfObject()
|
|
|
|
{
|
|
|
|
wxASSERT( outputFile );
|
|
|
|
wxASSERT( !workFile );
|
|
|
|
fputs( "endobj\n", outputFile );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts a PDF stream (for the page). Returns the object handle opened
|
|
|
|
* Pass -1 (default) for a fresh object. Especially from PDF 1.5 streams
|
|
|
|
* can contain a lot of things, but for the moment we only handle page
|
|
|
|
* content.
|
|
|
|
*/
|
|
|
|
int PDF_PLOTTER::startPdfStream(int handle)
|
|
|
|
{
|
|
|
|
wxASSERT( outputFile );
|
|
|
|
wxASSERT( !workFile );
|
|
|
|
handle = startPdfObject( handle );
|
|
|
|
|
|
|
|
// This is guaranteed to be handle+1 but needs to be allocated since
|
|
|
|
// you could allocate more object during stream preparation
|
|
|
|
streamLengthHandle = allocPdfObject();
|
|
|
|
fprintf( outputFile,
|
|
|
|
"<< /Length %d 0 R /Filter /FlateDecode >>\n" // Length is deferred
|
|
|
|
"stream\n", handle + 1 );
|
|
|
|
|
|
|
|
// Open a temporary file to accumulate the stream
|
|
|
|
workFilename = filename + wxT(".tmp");
|
|
|
|
workFile = wxFopen( workFilename, wxT( "w+b" ));
|
|
|
|
wxASSERT( workFile );
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finish the current PDF stream (writes the deferred length, too)
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::closePdfStream()
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
|
2015-02-27 14:33:13 +00:00
|
|
|
long stream_len = ftell( workFile );
|
2015-04-02 11:18:19 +00:00
|
|
|
|
|
|
|
if( stream_len < 0 )
|
|
|
|
{
|
|
|
|
wxASSERT( false );
|
|
|
|
return;
|
|
|
|
}
|
2012-05-03 19:08:14 +00:00
|
|
|
|
2012-05-09 00:26:15 +00:00
|
|
|
// Rewind the file, read in the page stream and DEFLATE it
|
2012-05-03 19:08:14 +00:00
|
|
|
fseek( workFile, 0, SEEK_SET );
|
|
|
|
unsigned char *inbuf = new unsigned char[stream_len];
|
|
|
|
|
|
|
|
int rc = fread( inbuf, 1, stream_len, workFile );
|
|
|
|
wxASSERT( rc == stream_len );
|
2012-05-09 00:26:15 +00:00
|
|
|
(void) rc;
|
2012-05-03 19:08:14 +00:00
|
|
|
|
2012-05-09 00:26:15 +00:00
|
|
|
// We are done with the temporary file, junk it
|
2012-05-03 19:08:14 +00:00
|
|
|
fclose( workFile );
|
|
|
|
workFile = 0;
|
|
|
|
::wxRemoveFile( workFilename );
|
|
|
|
|
2012-05-06 20:10:43 +00:00
|
|
|
// NULL means memos owns the memory, but provide a hint on optimum size needed.
|
2015-02-27 14:33:13 +00:00
|
|
|
wxMemoryOutputStream memos( NULL, std::max( 2000l, stream_len ) ) ;
|
2012-05-05 04:55:36 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
/* Somewhat standard parameters to compress in DEFLATE. The PDF spec is
|
2015-04-02 11:18:19 +00:00
|
|
|
* misleading, it says it wants a DEFLATE stream but it really want a ZLIB
|
|
|
|
* stream! (a DEFLATE stream would be generated with -15 instead of 15)
|
|
|
|
* rc = deflateInit2( &zstrm, Z_BEST_COMPRESSION, Z_DEFLATED, 15,
|
|
|
|
* 8, Z_DEFAULT_STRATEGY );
|
|
|
|
*/
|
2012-05-05 04:55:36 +00:00
|
|
|
|
|
|
|
wxZlibOutputStream zos( memos, wxZ_BEST_COMPRESSION, wxZLIB_ZLIB );
|
|
|
|
|
|
|
|
zos.Write( inbuf, stream_len );
|
|
|
|
|
2012-05-09 00:26:15 +00:00
|
|
|
delete[] inbuf;
|
|
|
|
|
|
|
|
} // flush the zip stream using zos destructor
|
2012-05-05 04:55:36 +00:00
|
|
|
|
|
|
|
wxStreamBuffer* sb = memos.GetOutputStreamBuffer();
|
|
|
|
|
|
|
|
unsigned out_count = sb->Tell();
|
|
|
|
|
|
|
|
fwrite( sb->GetBufferStart(), 1, out_count, outputFile );
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
fputs( "endstream\n", outputFile );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
// Writing the deferred length as an indirect object
|
|
|
|
startPdfObject( streamLengthHandle );
|
2012-05-05 04:55:36 +00:00
|
|
|
fprintf( outputFile, "%u\n", out_count );
|
2012-05-03 19:08:14 +00:00
|
|
|
closePdfObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts a new page in the PDF document
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::StartPage()
|
|
|
|
{
|
|
|
|
wxASSERT( outputFile );
|
|
|
|
wxASSERT( !workFile );
|
|
|
|
|
|
|
|
// Compute the paper size in IUs
|
|
|
|
paperSize = pageInfo.GetSizeMils();
|
|
|
|
paperSize.x *= 10.0 / iuPerDeviceUnit;
|
|
|
|
paperSize.y *= 10.0 / iuPerDeviceUnit;
|
|
|
|
|
|
|
|
// Open the content stream; the page object will go later
|
|
|
|
pageStreamHandle = startPdfStream();
|
|
|
|
|
|
|
|
/* Now, until ClosePage *everything* must be wrote in workFile, to be
|
|
|
|
compressed later in closePdfStream */
|
|
|
|
|
|
|
|
// Default graphic settings (coordinate system, default color and line style)
|
|
|
|
fprintf( workFile,
|
|
|
|
"%g 0 0 %g 0 0 cm 1 J 1 j 0 0 0 rg 0 0 0 RG %g w\n",
|
|
|
|
0.0072 * plotScaleAdjX, 0.0072 * plotScaleAdjY,
|
2020-04-14 12:25:00 +00:00
|
|
|
userToDeviceSize( m_renderSettings->GetDefaultPenWidth() ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Close the current page in the PDF document (and emit its compressed stream)
|
|
|
|
*/
|
|
|
|
void PDF_PLOTTER::ClosePage()
|
|
|
|
{
|
|
|
|
wxASSERT( workFile );
|
|
|
|
|
|
|
|
// Close the page stream (and compress it)
|
|
|
|
closePdfStream();
|
|
|
|
|
|
|
|
// Emit the page object and put it in the page list for later
|
|
|
|
pageHandles.push_back( startPdfObject() );
|
|
|
|
|
|
|
|
/* Page size is in 1/72 of inch (default user space units)
|
|
|
|
Works like the bbox in postscript but there is no need for
|
|
|
|
swapping the sizes, since PDF doesn't require a portrait page.
|
|
|
|
We use the MediaBox but PDF has lots of other less used boxes
|
|
|
|
to use */
|
|
|
|
|
|
|
|
const double BIGPTsPERMIL = 0.072;
|
|
|
|
wxSize psPaperSize = pageInfo.GetSizeMils();
|
|
|
|
|
|
|
|
fprintf( outputFile,
|
|
|
|
"<<\n"
|
|
|
|
"/Type /Page\n"
|
|
|
|
"/Parent %d 0 R\n"
|
|
|
|
"/Resources <<\n"
|
|
|
|
" /ProcSet [/PDF /Text /ImageC /ImageB]\n"
|
|
|
|
" /Font %d 0 R >>\n"
|
|
|
|
"/MediaBox [0 0 %d %d]\n"
|
|
|
|
"/Contents %d 0 R\n"
|
|
|
|
">>\n",
|
|
|
|
pageTreeHandle,
|
|
|
|
fontResDictHandle,
|
|
|
|
int( ceil( psPaperSize.x * BIGPTsPERMIL ) ),
|
|
|
|
int( ceil( psPaperSize.y * BIGPTsPERMIL ) ),
|
|
|
|
pageStreamHandle );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
// Mark the page stream as idle
|
|
|
|
pageStreamHandle = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The PDF engine supports multiple pages; the first one is opened
|
|
|
|
* 'for free' the following are to be closed and reopened. Between
|
|
|
|
* each page parameters can be set
|
|
|
|
*/
|
2012-10-13 18:54:33 +00:00
|
|
|
bool PDF_PLOTTER::StartPlot()
|
2012-05-03 19:08:14 +00:00
|
|
|
{
|
2012-10-13 18:54:33 +00:00
|
|
|
wxASSERT( outputFile );
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
// First things first: the customary null object
|
|
|
|
xrefTable.clear();
|
|
|
|
xrefTable.push_back( 0 );
|
|
|
|
|
|
|
|
/* The header (that's easy!). The second line is binary junk required
|
|
|
|
to make the file binary from the beginning (the important thing is
|
|
|
|
that they must have the bit 7 set) */
|
|
|
|
fputs( "%PDF-1.5\n%\200\201\202\203\n", outputFile );
|
|
|
|
|
|
|
|
/* Allocate an entry for the page tree root, it will go in every page
|
|
|
|
parent entry */
|
|
|
|
pageTreeHandle = allocPdfObject();
|
|
|
|
|
|
|
|
/* In the same way, the font resource dictionary is used by every page
|
|
|
|
(it *could* be inherited via the Pages tree */
|
|
|
|
fontResDictHandle = allocPdfObject();
|
|
|
|
|
|
|
|
/* Now, the PDF is read from the end, (more or less)... so we start
|
|
|
|
with the page stream for page 1. Other more important stuff is written
|
|
|
|
at the end */
|
|
|
|
StartPage();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool PDF_PLOTTER::EndPlot()
|
|
|
|
{
|
|
|
|
wxASSERT( outputFile );
|
|
|
|
|
|
|
|
// Close the current page (often the only one)
|
|
|
|
ClosePage();
|
|
|
|
|
|
|
|
/* We need to declare the resources we're using (fonts in particular)
|
|
|
|
The useful standard one is the Helvetica family. Adding external fonts
|
|
|
|
is *very* involved! */
|
|
|
|
struct {
|
|
|
|
const char *psname;
|
|
|
|
const char *rsname;
|
|
|
|
int font_handle;
|
|
|
|
} fontdefs[4] = {
|
|
|
|
{ "/Helvetica", "/KicadFont", 0 },
|
|
|
|
{ "/Helvetica-Oblique", "/KicadFontI", 0 },
|
|
|
|
{ "/Helvetica-Bold", "/KicadFontB", 0 },
|
|
|
|
{ "/Helvetica-BoldOblique", "/KicadFontBI", 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Declare the font resources. Since they're builtin fonts, no descriptors (yay!)
|
2019-08-20 17:22:30 +00:00
|
|
|
We'll need metrics anyway to do any alignment (these are in the shared with
|
2012-05-03 19:08:14 +00:00
|
|
|
the postscript engine) */
|
|
|
|
for( int i = 0; i < 4; i++ )
|
|
|
|
{
|
|
|
|
fontdefs[i].font_handle = startPdfObject();
|
|
|
|
fprintf( outputFile,
|
|
|
|
"<< /BaseFont %s\n"
|
|
|
|
" /Type /Font\n"
|
|
|
|
" /Subtype /Type1\n"
|
|
|
|
|
|
|
|
/* Adobe is so Mac-based that the nearest thing to Latin1 is
|
|
|
|
the Windows ANSI encoding! */
|
|
|
|
" /Encoding /WinAnsiEncoding\n"
|
|
|
|
">>\n",
|
|
|
|
fontdefs[i].psname );
|
|
|
|
closePdfObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Named font dictionary (was allocated, now we emit it)
|
|
|
|
startPdfObject( fontResDictHandle );
|
|
|
|
fputs( "<<\n", outputFile );
|
|
|
|
for( int i = 0; i < 4; i++ )
|
|
|
|
{
|
|
|
|
fprintf( outputFile, " %s %d 0 R\n",
|
|
|
|
fontdefs[i].rsname, fontdefs[i].font_handle );
|
|
|
|
}
|
|
|
|
fputs( ">>\n", outputFile );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
/* The page tree: it's a B-tree but luckily we only have few pages!
|
|
|
|
So we use just an array... The handle was allocated at the beginning,
|
|
|
|
now we instantiate the corresponding object */
|
|
|
|
startPdfObject( pageTreeHandle );
|
|
|
|
fputs( "<<\n"
|
|
|
|
"/Type /Pages\n"
|
|
|
|
"/Kids [\n", outputFile );
|
|
|
|
|
|
|
|
for( unsigned i = 0; i < pageHandles.size(); i++ )
|
|
|
|
fprintf( outputFile, "%d 0 R\n", pageHandles[i] );
|
|
|
|
|
|
|
|
fprintf( outputFile,
|
|
|
|
"]\n"
|
|
|
|
"/Count %ld\n"
|
|
|
|
">>\n", (long) pageHandles.size() );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
|
|
|
|
// The info dictionary
|
|
|
|
int infoDictHandle = startPdfObject();
|
|
|
|
char date_buf[250];
|
|
|
|
time_t ltime = time( NULL );
|
|
|
|
strftime( date_buf, 250, "D:%Y%m%d%H%M%S",
|
|
|
|
localtime( <ime ) );
|
2017-01-20 21:00:20 +00:00
|
|
|
|
|
|
|
if( title.IsEmpty() )
|
|
|
|
{
|
|
|
|
// Windows uses '\' and other platforms ue '/' as sepatator
|
|
|
|
title = filename.AfterLast('\\');
|
|
|
|
title = title.AfterLast('/');
|
|
|
|
}
|
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
fprintf( outputFile,
|
|
|
|
"<<\n"
|
|
|
|
"/Producer (KiCAD PDF)\n"
|
|
|
|
"/CreationDate (%s)\n"
|
|
|
|
"/Creator (%s)\n"
|
|
|
|
"/Title (%s)\n"
|
2020-05-18 00:42:04 +00:00
|
|
|
"/Trapped False\n",
|
2012-05-03 19:08:14 +00:00
|
|
|
date_buf,
|
|
|
|
TO_UTF8( creator ),
|
2017-01-20 21:00:20 +00:00
|
|
|
TO_UTF8( title ) );
|
2012-05-03 19:08:14 +00:00
|
|
|
|
|
|
|
fputs( ">>\n", outputFile );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
// The catalog, at last
|
|
|
|
int catalogHandle = startPdfObject();
|
|
|
|
fprintf( outputFile,
|
|
|
|
"<<\n"
|
|
|
|
"/Type /Catalog\n"
|
|
|
|
"/Pages %d 0 R\n"
|
|
|
|
"/Version /1.5\n"
|
|
|
|
"/PageMode /UseNone\n"
|
|
|
|
"/PageLayout /SinglePage\n"
|
|
|
|
">>\n", pageTreeHandle );
|
|
|
|
closePdfObject();
|
|
|
|
|
|
|
|
/* Emit the xref table (format is crucial to the byte, each entry must
|
|
|
|
be 20 bytes long, and object zero must be done in that way). Also
|
|
|
|
the offset must be kept along for the trailer */
|
|
|
|
long xref_start = ftell( outputFile );
|
|
|
|
fprintf( outputFile,
|
|
|
|
"xref\n"
|
|
|
|
"0 %ld\n"
|
|
|
|
"0000000000 65535 f \n", (long) xrefTable.size() );
|
|
|
|
for( unsigned i = 1; i < xrefTable.size(); i++ )
|
|
|
|
{
|
|
|
|
fprintf( outputFile, "%010ld 00000 n \n", xrefTable[i] );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Done the xref, go for the trailer
|
|
|
|
fprintf( outputFile,
|
|
|
|
"trailer\n"
|
|
|
|
"<< /Size %lu /Root %d 0 R /Info %d 0 R >>\n"
|
|
|
|
"startxref\n"
|
|
|
|
"%ld\n" // The offset we saved before
|
2014-06-20 08:55:30 +00:00
|
|
|
"%%%%EOF\n",
|
2012-05-03 19:08:14 +00:00
|
|
|
(unsigned long) xrefTable.size(), catalogHandle, infoDictHandle, xref_start );
|
|
|
|
|
|
|
|
fclose( outputFile );
|
|
|
|
outputFile = NULL;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PDF_PLOTTER::Text( const wxPoint& aPos,
|
2017-02-20 16:57:41 +00:00
|
|
|
const COLOR4D aColor,
|
2012-05-03 19:08:14 +00:00
|
|
|
const wxString& aText,
|
2013-05-05 07:17:48 +00:00
|
|
|
double aOrient,
|
2012-05-03 19:08:14 +00:00
|
|
|
const wxSize& aSize,
|
|
|
|
enum EDA_TEXT_HJUSTIFY_T aH_justify,
|
|
|
|
enum EDA_TEXT_VJUSTIFY_T aV_justify,
|
|
|
|
int aWidth,
|
|
|
|
bool aItalic,
|
2014-04-28 16:13:18 +00:00
|
|
|
bool aBold,
|
2016-09-19 11:01:36 +00:00
|
|
|
bool aMultilineAllowed,
|
|
|
|
void* aData )
|
2012-05-03 19:08:14 +00:00
|
|
|
{
|
2015-03-18 19:50:42 +00:00
|
|
|
// PDF files do not like 0 sized texts which create broken files.
|
|
|
|
if( aSize.x == 0 || aSize.y == 0 )
|
|
|
|
return;
|
|
|
|
|
2020-05-18 10:34:30 +00:00
|
|
|
// Render phantom text (which will be searchable) behind the stroke font. This won't
|
|
|
|
// be pixel-accurate, but it doesn't matter for searching.
|
|
|
|
int render_mode = 3; // invisible
|
2017-08-15 13:52:46 +00:00
|
|
|
|
2020-05-18 12:37:05 +00:00
|
|
|
const char *fontname = aItalic ? ( aBold ? "/KicadFontBI" : "/KicadFontI" )
|
|
|
|
: ( aBold ? "/KicadFontB" : "/KicadFont" );
|
2017-08-15 13:52:46 +00:00
|
|
|
|
2019-08-20 17:22:30 +00:00
|
|
|
// Compute the copious transformation parameters of the Curent Transform Matrix
|
2017-08-15 13:52:46 +00:00
|
|
|
double ctm_a, ctm_b, ctm_c, ctm_d, ctm_e, ctm_f;
|
|
|
|
double wideningFactor, heightFactor;
|
|
|
|
|
|
|
|
computeTextParameters( aPos, aText, aOrient, aSize, m_plotMirror, aH_justify,
|
2020-04-14 12:25:00 +00:00
|
|
|
aV_justify, aWidth, aItalic, aBold,
|
|
|
|
&wideningFactor, &ctm_a, &ctm_b, &ctm_c,
|
|
|
|
&ctm_d, &ctm_e, &ctm_f, &heightFactor );
|
2017-08-15 13:52:46 +00:00
|
|
|
|
|
|
|
SetColor( aColor );
|
|
|
|
SetCurrentLineWidth( aWidth, aData );
|
|
|
|
|
|
|
|
/* We use the full CTM instead of the text matrix because the same
|
|
|
|
coordinate system will be used for the overlining. Also the %f
|
|
|
|
for the trig part of the matrix to avoid %g going in exponential
|
2020-05-18 10:34:30 +00:00
|
|
|
format (which is not supported) */
|
2017-08-15 13:52:46 +00:00
|
|
|
fprintf( workFile, "q %f %f %f %f %g %g cm BT %s %g Tf %d Tr %g Tz ",
|
2020-05-18 12:37:05 +00:00
|
|
|
ctm_a, ctm_b, ctm_c, ctm_d, ctm_e, ctm_f,
|
|
|
|
fontname, heightFactor, render_mode, wideningFactor * 100 );
|
2017-08-15 13:52:46 +00:00
|
|
|
|
|
|
|
// The text must be escaped correctly
|
|
|
|
fputsPostscriptString( workFile, aText );
|
|
|
|
fputs( " Tj ET\n", workFile );
|
|
|
|
|
2020-05-18 10:34:30 +00:00
|
|
|
// Restore the CTM
|
|
|
|
fputs( "Q\n", workFile );
|
2017-08-15 13:52:46 +00:00
|
|
|
|
2012-05-03 19:08:14 +00:00
|
|
|
// Plot the stroked text (if requested)
|
2020-05-18 10:34:30 +00:00
|
|
|
PLOTTER::Text( aPos, aColor, aText, aOrient, aSize, aH_justify, aV_justify, aWidth,
|
|
|
|
aItalic, aBold, aMultilineAllowed );
|
2012-05-03 19:08:14 +00:00
|
|
|
}
|
|
|
|
|