diff options
Diffstat (limited to 'pcbnew/printout_controler.cpp')
-rw-r--r-- | pcbnew/printout_controler.cpp | 353 |
1 files changed, 353 insertions, 0 deletions
diff --git a/pcbnew/printout_controler.cpp b/pcbnew/printout_controler.cpp new file mode 100644 index 0000000..7ad346f --- /dev/null +++ b/pcbnew/printout_controler.cpp @@ -0,0 +1,353 @@ +/* + * This program source code file is part of KiCad, a free EDA CAD application. + * + * Copyright (C) 2009 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr + * Copyright (C) 1992-2011 KiCad Developers, see AUTHORS.txt for contributors. + * + * 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 + */ + +/** + * @file printout_controler.cpp + * @brief Board print handler implementation file. + */ + + +// Set this to 1 if you want to test PostScript printing under MSW. +#define wxTEST_POSTSCRIPT_IN_MSW 1 + +#include <fctsys.h> +#include <pgm_base.h> +#include <gr_basic.h> +#include <class_drawpanel.h> +#include <confirm.h> +#include <base_units.h> +#include <wxBasePcbFrame.h> +#include <class_board.h> +#include <pcbnew.h> + +#include <printout_controler.h> + + +/** + * Definition for enabling and disabling print controller trace output. See the + * wxWidgets documentation on using the WXTRACE environment variable. + */ +static const wxString tracePrinting( wxT( "KicadPrinting" ) ); + + +PRINT_PARAMETERS::PRINT_PARAMETERS() +{ + m_PenDefaultSize = Millimeter2iu( 0.2 ); // A reasonable default value to draw items + // which do not have a specified line width + m_PrintScale = 1.0; + m_XScaleAdjust = 1.0; + m_YScaleAdjust = 1.0; + m_Print_Sheet_Ref = false; + m_PrintMaskLayer.set(); + m_PrintMirror = false; + m_Print_Black_and_White = true; + m_OptionPrintPage = 1; + m_PageCount = 1; + m_ForceCentered = false; + m_Flags = 0; + m_DrillShapeOpt = PRINT_PARAMETERS::SMALL_DRILL_SHAPE; + m_PageSetupData = NULL; +} + + +BOARD_PRINTOUT_CONTROLLER::BOARD_PRINTOUT_CONTROLLER( const PRINT_PARAMETERS& aParams, + EDA_DRAW_FRAME* aParent, + const wxString& aTitle ) : + wxPrintout( aTitle ) +{ + m_PrintParams = aParams; // Make a local copy of the print parameters. + m_Parent = aParent; +} + + +bool BOARD_PRINTOUT_CONTROLLER::OnPrintPage( int aPage ) +{ + LSET lset = m_PrintParams.m_PrintMaskLayer; + + // compute layer mask from page number if we want one page per layer + if( m_PrintParams.m_OptionPrintPage == 0 ) // One page per layer + { + // This sequence is TBD, call a different + // sequencer if needed, such as Seq(). Could not find documentation on + // page order. + LSEQ seq = lset.UIOrder(); + + // aPage starts at 1, not 0 + if( unsigned( aPage-1 ) < seq.size() ) + m_PrintParams.m_PrintMaskLayer = LSET( seq[aPage-1] ); + } + + if( !m_PrintParams.m_PrintMaskLayer.any() ) + return false; + + // In Pcbnew we can want the layer EDGE always printed + if( m_PrintParams.m_Flags == 1 ) + m_PrintParams.m_PrintMaskLayer.set( Edge_Cuts ); + + DrawPage(); + + m_PrintParams.m_PrintMaskLayer = lset; + + return true; +} + + +void BOARD_PRINTOUT_CONTROLLER::GetPageInfo( int* minPage, int* maxPage, + int* selPageFrom, int* selPageTo ) +{ + *minPage = 1; + *selPageFrom = 1; + + int icnt = 1; + + if( m_PrintParams.m_OptionPrintPage == 0 ) + icnt = m_PrintParams.m_PageCount; + + *maxPage = icnt; + *selPageTo = icnt; +} + + +void BOARD_PRINTOUT_CONTROLLER::DrawPage() +{ + wxPoint offset; + double userscale; + EDA_RECT boardBoundingBox; + EDA_RECT drawRect; + wxDC* dc = GetDC(); + BASE_SCREEN* screen = m_Parent->GetScreen(); + bool printMirror = m_PrintParams.m_PrintMirror; + wxSize pageSizeIU = m_Parent->GetPageSizeIU(); + + wxBusyCursor dummy; + + BOARD* brd = ((PCB_BASE_FRAME*) m_Parent)->GetBoard(); + boardBoundingBox = brd->ComputeBoundingBox(); + wxString titleblockFilename = brd->GetFileName(); + + // Use the page size as the drawing area when the board is shown or the user scale + // is less than 1. + if( m_PrintParams.PrintBorderAndTitleBlock() ) + boardBoundingBox = EDA_RECT( wxPoint( 0, 0 ), pageSizeIU ); + + wxLogTrace( tracePrinting, wxT( "Drawing bounding box: x=%d, y=%d, w=%d, h=%d" ), + boardBoundingBox.GetX(), boardBoundingBox.GetY(), + boardBoundingBox.GetWidth(), boardBoundingBox.GetHeight() ); + + // Compute the PCB size in internal units + userscale = m_PrintParams.m_PrintScale; + + if( m_PrintParams.m_PrintScale == 0 ) // fit in page option + { + if( boardBoundingBox.GetWidth() && boardBoundingBox.GetHeight() ) + { + int margin = Millimeter2iu( 10.0 ); // add a margin around the drawings + double scaleX = (double)(pageSizeIU.x - (2 * margin)) / + boardBoundingBox.GetWidth(); + double scaleY = (double)(pageSizeIU.y - (2 * margin)) / + boardBoundingBox.GetHeight(); + userscale = (scaleX < scaleY) ? scaleX : scaleY; + } + else + userscale = 1.0; + } + + wxSize scaledPageSize = pageSizeIU; + drawRect.SetSize( scaledPageSize ); + scaledPageSize.x = wxRound( scaledPageSize.x / userscale ); + scaledPageSize.y = wxRound( scaledPageSize.y / userscale ); + + + if( m_PrintParams.m_PageSetupData ) + { + wxLogTrace( tracePrinting, wxT( "Fit size to page margins: x=%d, y=%d" ), + scaledPageSize.x, scaledPageSize.y ); + + // Always scale to the size of the paper. + FitThisSizeToPageMargins( scaledPageSize, *m_PrintParams.m_PageSetupData ); + } + + // Compute Accurate scale 1 + if( m_PrintParams.m_PrintScale == 1.0 ) + { + // We want a 1:1 scale, regardless the page setup + // like page size, margin ... + MapScreenSizeToPaper(); // set best scale and offset (scale is not used) + int w, h; + GetPPIPrinter( &w, &h ); + double accurate_Xscale = (double) w / (IU_PER_MILS*1000); + double accurate_Yscale = (double) h / (IU_PER_MILS*1000); + + if( IsPreview() ) // Scale must take in account the DC size in Preview + { + // Get the size of the DC in pixels + wxSize PlotAreaSize; + dc->GetSize( &PlotAreaSize.x, &PlotAreaSize.y ); + GetPageSizePixels( &w, &h ); + accurate_Xscale *= (double)PlotAreaSize.x / w; + accurate_Yscale *= (double)PlotAreaSize.y / h; + } + // Fine scale adjust + accurate_Xscale *= m_PrintParams.m_XScaleAdjust; + accurate_Yscale *= m_PrintParams.m_YScaleAdjust; + + // Set print scale for 1:1 exact scale + dc->SetUserScale( accurate_Xscale, accurate_Yscale ); + } + + // Get the final size of the DC in pixels + wxSize PlotAreaSizeInPixels; + dc->GetSize( &PlotAreaSizeInPixels.x, &PlotAreaSizeInPixels.y ); + wxLogTrace( tracePrinting, wxT( "Plot area in pixels: x=%d, y=%d" ), + PlotAreaSizeInPixels.x, PlotAreaSizeInPixels.y ); + double scalex, scaley; + dc->GetUserScale( &scalex, &scaley ); + wxLogTrace( tracePrinting, wxT( "DC user scale: x=%g, y=%g" ), + scalex, scaley ); + + wxSize PlotAreaSizeInUserUnits; + PlotAreaSizeInUserUnits.x = KiROUND( PlotAreaSizeInPixels.x / scalex ); + PlotAreaSizeInUserUnits.y = KiROUND( PlotAreaSizeInPixels.y / scaley ); + wxLogTrace( tracePrinting, wxT( "Scaled plot area in user units: x=%d, y=%d" ), + PlotAreaSizeInUserUnits.x, PlotAreaSizeInUserUnits.y ); + + // In module editor, the module is located at 0,0 but for printing + // it is moved to pageSizeIU.x/2, pageSizeIU.y/2. + // So the equivalent board must be moved to the center of the page: + if( m_Parent->IsType( FRAME_PCB_MODULE_EDITOR ) ) + { + boardBoundingBox.Move( wxPoint( pageSizeIU.x/2, pageSizeIU.y/2 ) ); + } + + // In some cases the plot origin is the centre of the board outline rather than the center + // of the selected paper size. + if( m_PrintParams.CenterOnBoardOutline() ) + { + // Here we are only drawing the board and it's contents. + drawRect = boardBoundingBox; + offset.x += wxRound( (double) -scaledPageSize.x / 2.0 ); + offset.y += wxRound( (double) -scaledPageSize.y / 2.0 ); + + wxPoint center = boardBoundingBox.Centre(); + + if( printMirror ) + { + // Calculate the mirrored center of the board. + center.x = m_Parent->GetPageSizeIU().x - boardBoundingBox.Centre().x; + } + + offset += center; + } + + GRResetPenAndBrush( dc ); + + EDA_DRAW_PANEL* panel = m_Parent->GetCanvas(); + EDA_RECT tmp = *panel->GetClipBox(); + + // Set clip box to the max size + #define MAX_VALUE (INT_MAX/2) // MAX_VALUE is the max we can use in an integer + // and that allows calculations without overflow + panel->SetClipBox( EDA_RECT( wxPoint( 0, 0 ), wxSize( MAX_VALUE, MAX_VALUE ) ) ); + + screen->m_IsPrinting = true; + EDA_COLOR_T bg_color = m_Parent->GetDrawBgColor(); + + // Print frame reference, if requested, before + if( m_PrintParams.m_Print_Black_and_White ) + GRForceBlackPen( true ); + + if( m_PrintParams.PrintBorderAndTitleBlock() ) + m_Parent->DrawWorkSheet( dc, screen, m_PrintParams.m_PenDefaultSize, + IU_PER_MILS, titleblockFilename ); + + if( printMirror ) + { + // To plot mirror, we reverse the x axis, and modify the plot x origin + dc->SetAxisOrientation( false, false); + + /* Plot offset x is moved by the x plot area size in order to have + * the old draw area in the new draw area, because the draw origin has not moved + * (this is the upper left corner) but the X axis is reversed, therefore the plotting area + * is the x coordinate values from - PlotAreaSize.x to 0 */ + int x_dc_offset = PlotAreaSizeInPixels.x; + x_dc_offset = KiROUND( x_dc_offset * userscale ); + dc->SetDeviceOrigin( x_dc_offset, 0 ); + + wxLogTrace( tracePrinting, wxT( "Device origin: x=%d, y=%d" ), + x_dc_offset, 0 ); + + panel->SetClipBox( EDA_RECT( wxPoint( -MAX_VALUE/2, -MAX_VALUE/2 ), + panel->GetClipBox()->GetSize() ) ); + } + + // screen->m_DrawOrg = offset; + dc->SetLogicalOrigin( offset.x, offset.y ); + + wxLogTrace( tracePrinting, wxT( "Logical origin: x=%d, y=%d" ), + offset.x, offset.y ); + +#if defined(wxUSE_LOG_TRACE) && defined( DEBUG ) + wxRect paperRect = GetPaperRectPixels(); + wxLogTrace( tracePrinting, wxT( "Paper rectangle: left=%d, top=%d, " + "right=%d, bottom=%d" ), + paperRect.GetLeft(), paperRect.GetTop(), paperRect.GetRight(), + paperRect.GetBottom() ); + + int devLeft = dc->LogicalToDeviceX( drawRect.GetX() ); + int devTop = dc->LogicalToDeviceY( drawRect.GetY() ); + int devRight = dc->LogicalToDeviceX( drawRect.GetRight() ); + int devBottom = dc->LogicalToDeviceY( drawRect.GetBottom() ); + wxLogTrace( tracePrinting, wxT( "Final device rectangle: left=%d, top=%d, " + "right=%d, bottom=%d\n" ), + devLeft, devTop, devRight, devBottom ); +#endif + + m_Parent->SetDrawBgColor( WHITE ); + + /* when printing in color mode, we use the graphic OR mode that gives the same look as + * the screen but because the background is white when printing, we must use a trick: + * In order to plot on a white background in OR mode we must: + * 1 - Plot all items in black, this creates a local black background + * 2 - Plot in OR mode on black "local" background + */ + if( !m_PrintParams.m_Print_Black_and_White ) + { + // Creates a "local" black background + GRForceBlackPen( true ); + m_Parent->PrintPage( dc, m_PrintParams.m_PrintMaskLayer, + printMirror, &m_PrintParams ); + GRForceBlackPen( false ); + } + else + GRForceBlackPen( true ); + + + m_Parent->PrintPage( dc, m_PrintParams.m_PrintMaskLayer, printMirror, + &m_PrintParams ); + + m_Parent->SetDrawBgColor( bg_color ); + screen->m_IsPrinting = false; + panel->SetClipBox( tmp ); + GRForceBlackPen( false ); +} |