/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace vcl; using namespace com::sun::star; using namespace com::sun::star::uno; using namespace com::sun::star::container; using namespace com::sun::star::beans; enum { ORIENTATION_AUTOMATIC, ORIENTATION_PORTRAIT, ORIENTATION_LANDSCAPE }; namespace { bool lcl_ListBoxCompare( const OUString& rStr1, const OUString& rStr2 ) { return vcl::NaturalSortCompare( rStr1, rStr2 ) < 0; } } PrintDialog::PrintPreviewWindow::PrintPreviewWindow(PrintDialog* pDialog) : mpDialog(pDialog) , maOrigSize( 10, 10 ) , mnDPIX(Application::GetDefaultDevice()->GetDPIX()) , mnDPIY(Application::GetDefaultDevice()->GetDPIY()) , mbGreyscale( false ) { } PrintDialog::PrintPreviewWindow::~PrintPreviewWindow() { } void PrintDialog::PrintPreviewWindow::Resize() { Size aNewSize(GetOutputSizePixel()); tools::Long nTextHeight = GetDrawingArea()->get_text_height(); // leave small space for decoration aNewSize.AdjustWidth( -(nTextHeight + 2) ); aNewSize.AdjustHeight( -(nTextHeight + 2) ); Size aScaledSize; double fScale = 1.0; // #i106435# catch corner case of Size(0,0) Size aOrigSize( maOrigSize ); if( aOrigSize.Width() < 1 ) aOrigSize.setWidth( aNewSize.Width() ); if( aOrigSize.Height() < 1 ) aOrigSize.setHeight( aNewSize.Height() ); if( aOrigSize.Width() > aOrigSize.Height() ) { aScaledSize = Size( aNewSize.Width(), aNewSize.Width() * aOrigSize.Height() / aOrigSize.Width() ); if( aScaledSize.Height() > aNewSize.Height() ) fScale = double(aNewSize.Height())/double(aScaledSize.Height()); } else { aScaledSize = Size( aNewSize.Height() * aOrigSize.Width() / aOrigSize.Height(), aNewSize.Height() ); if( aScaledSize.Width() > aNewSize.Width() ) fScale = double(aNewSize.Width())/double(aScaledSize.Width()); } aScaledSize.setWidth( tools::Long(aScaledSize.Width()*fScale) ); aScaledSize.setHeight( tools::Long(aScaledSize.Height()*fScale) ); maPreviewSize = aScaledSize; // check and evtl. recreate preview bitmap preparePreviewBitmap(); } void PrintDialog::PrintPreviewWindow::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle&) { rRenderContext.Push(); weld::SetPointFont(rRenderContext, rRenderContext.GetSettings().GetStyleSettings().GetLabelFont()); rRenderContext.SetTextColor(rRenderContext.GetSettings().GetStyleSettings().GetLabelTextColor()); rRenderContext.SetBackground(Wallpaper(Application::GetSettings().GetStyleSettings().GetDialogColor())); rRenderContext.Erase(); auto nTextHeight = rRenderContext.GetTextHeight(); Size aSize(GetOutputSizePixel()); Point aOffset((aSize.Width() - maPreviewSize.Width() + nTextHeight) / 2, (aSize.Height() - maPreviewSize.Height() + nTextHeight) / 2); // horizontal line { auto nWidth = rRenderContext.GetTextWidth(maHorzText); auto nStart = aOffset.X() + (maPreviewSize.Width() - nWidth) / 2; rRenderContext.DrawText(Point(nStart, aOffset.Y() - nTextHeight), maHorzText, 0, maHorzText.getLength()); DecorationView aDecoView(&rRenderContext); auto nTop = aOffset.Y() - (nTextHeight / 2); aDecoView.DrawSeparator(Point(aOffset.X(), nTop), Point(nStart - 2, nTop), false); aDecoView.DrawSeparator(Point(nStart + nWidth + 2, nTop), Point(aOffset.X() + maPreviewSize.Width(), nTop), false); } // vertical line { rRenderContext.Push(PushFlags::FONT); vcl::Font aFont(rRenderContext.GetFont()); aFont.SetOrientation(900_deg10); rRenderContext.SetFont(aFont); auto nLeft = aOffset.X() - nTextHeight; auto nWidth = rRenderContext.GetTextWidth(maVertText); auto nStart = aOffset.Y() + (maPreviewSize.Height() + nWidth) / 2; rRenderContext.DrawText(Point(nLeft, nStart), maVertText, 0, maVertText.getLength()); DecorationView aDecoView(&rRenderContext); nLeft = aOffset.X() - (nTextHeight / 2); aDecoView.DrawSeparator(Point(nLeft, aOffset.Y()), Point(nLeft, nStart - nWidth - 2), true); aDecoView.DrawSeparator(Point(nLeft, nStart + 2), Point(nLeft, aOffset.Y() + maPreviewSize.Height()), true); rRenderContext.Pop(); } if (!maReplacementString.isEmpty()) { // replacement is active tools::Rectangle aTextRect(aOffset + Point(2, 2), Size(maPreviewSize.Width() - 4, maPreviewSize.Height() - 4)); rRenderContext.DrawText(aTextRect, maReplacementString, DrawTextFlags::Center | DrawTextFlags::VCenter | DrawTextFlags::WordBreak | DrawTextFlags::MultiLine); } else { BitmapEx aPreviewBitmap(maPreviewBitmap); // This explicit force-to-scale allows us to get the // mentioned best quality here. Unfortunately this is // currently not sure when using just ::DrawBitmap with // a defined size or ::DrawOutDev aPreviewBitmap.Scale(maPreviewSize, BmpScaleFlag::BestQuality); rRenderContext.DrawBitmapEx(aOffset, aPreviewBitmap); } tools::Rectangle aFrameRect(aOffset + Point(-1, -1), Size(maPreviewSize.Width() + 2, maPreviewSize.Height() + 2)); DecorationView aDecorationView(&rRenderContext); aDecorationView.DrawFrame(aFrameRect, DrawFrameStyle::Group); rRenderContext.Pop(); } bool PrintDialog::PrintPreviewWindow::Command( const CommandEvent& rEvt ) { if( rEvt.GetCommand() == CommandEventId::Wheel ) { const CommandWheelData* pWheelData = rEvt.GetWheelData(); if(pWheelData->GetDelta() > 0) mpDialog->previewForward(); else if (pWheelData->GetDelta() < 0) mpDialog->previewBackward(); return true; } return CustomWidgetController::Command(rEvt); } void PrintDialog::PrintPreviewWindow::setPreview( const GDIMetaFile& i_rNewPreview, const Size& i_rOrigSize, std::u16string_view i_rPaperName, const OUString& i_rReplacement, sal_Int32 i_nDPIX, sal_Int32 i_nDPIY, bool i_bGreyscale ) { maMtf = i_rNewPreview; mnDPIX = i_nDPIX; mnDPIY = i_nDPIY; maOrigSize = i_rOrigSize; maReplacementString = i_rReplacement; mbGreyscale = i_bGreyscale; // use correct measurements const LocaleDataWrapper& rLocWrap(Application::GetSettings().GetLocaleDataWrapper()); o3tl::Length eUnit = o3tl::Length::mm; int nDigits = 0; if( rLocWrap.getMeasurementSystemEnum() == MeasurementSystem::US ) { eUnit = o3tl::Length::in100; nDigits = 2; } Size aLogicPaperSize(o3tl::convert(i_rOrigSize, o3tl::Length::mm100, eUnit)); OUString aNumText( rLocWrap.getNum( aLogicPaperSize.Width(), nDigits ) ); OUStringBuffer aBuf( aNumText + " " ); aBuf.appendAscii( eUnit == o3tl::Length::mm ? "mm" : "in" ); if( !i_rPaperName.empty() ) { aBuf.append( OUString::Concat(" (") + i_rPaperName + ")" ); } maHorzText = aBuf.makeStringAndClear(); aNumText = rLocWrap.getNum( aLogicPaperSize.Height(), nDigits ); aBuf.append( aNumText + " " ); aBuf.appendAscii( eUnit == o3tl::Length::mm ? "mm" : "in" ); maVertText = aBuf.makeStringAndClear(); // We have a new Metafile and evtl. a new page, so we need to reset // the PreviewBitmap to force new creation maPreviewBitmap = Bitmap(); // sets/calculates e.g. maPreviewSize // also triggers preparePreviewBitmap() Resize(); Invalidate(); } void PrintDialog::PrintPreviewWindow::preparePreviewBitmap() { if(maPreviewSize.IsEmpty()) { // not yet fully initialized, no need to prepare anything return; } // define an allowed number of pixels, also see // defaults for primitive renderers and similar. This // might be centralized and made dependent of 32/64bit const sal_uInt32 nMaxSquarePixels(500000); // check how big (squarePixels) the preview is currently (with // max value of MaxSquarePixels) const sal_uInt32 nCurrentSquarePixels( std::min( nMaxSquarePixels, static_cast(maPreviewBitmap.GetSizePixel().getWidth()) * static_cast(maPreviewBitmap.GetSizePixel().getHeight()))); // check how big (squarePixels) the preview needs to be (with // max value of MaxSquarePixels) const sal_uInt32 nRequiredSquarePixels( std::min( nMaxSquarePixels, static_cast(maPreviewSize.getWidth()) * static_cast(maPreviewSize.getHeight()))); // check if preview is big enough. Use a scaling value in // the comparison to not get bigger at the last possible moment // what may look awkward and pixelated (again). This means // to use a percentage value - if we have at least // that value of required pixels, we are good. static const double fPreventAwkwardFactor(1.35); // 35% if(nCurrentSquarePixels >= static_cast(nRequiredSquarePixels * fPreventAwkwardFactor)) { // at this place we also could add a mechanism to let the preview // bitmap 'shrink' again if it is currently 'too big' -> bigger // than required. I think this is not necessary for now. // already sufficient, done. return; } // check if we have enough square pixels e.g for 8x8 pixels if(nRequiredSquarePixels < 64) { // too small preview - let it empty return; } // Calculate nPlannedSquarePixels which is the required size // expanded by a percentage (with max value of MaxSquarePixels) static const double fExtraSpaceFactor(1.65); // 65% const sal_uInt32 nPlannedSquarePixels( std::min( nMaxSquarePixels, static_cast(maPreviewSize.getWidth() * fExtraSpaceFactor) * static_cast(maPreviewSize.getHeight() * fExtraSpaceFactor))); // calculate back new width and height - it might have been // truncated by MaxSquarePixels. // We know that w*h == nPlannedSquarePixels and w/h == ratio const double fRatio(static_cast(maPreviewSize.getWidth()) / static_cast(maPreviewSize.getHeight())); const double fNewWidth(sqrt(static_cast(nPlannedSquarePixels) * fRatio)); const double fNewHeight(sqrt(static_cast(nPlannedSquarePixels) / fRatio)); const Size aScaledSize(basegfx::fround(fNewWidth), basegfx::fround(fNewHeight)); // check if this eventual maximum is already reached // due to having hit the MaxSquarePixels. Due to using // an integer AspectRatio, we cannot make a numeric exact // comparison - we need to compare if we are close const double fScaledSizeSquare(static_cast(aScaledSize.getWidth() * aScaledSize.getHeight())); const double fPreviewSizeSquare(static_cast(maPreviewBitmap.GetSizePixel().getWidth() * maPreviewBitmap.GetSizePixel().getHeight())); // test as equal up to 0.1% (0.001) if(fPreviewSizeSquare != 0.0 && fabs((fScaledSizeSquare / fPreviewSizeSquare) - 1.0) < 0.001) { // maximum is reached, avoid bigger scaling return; } // create temporary VDev with requested Size and DPI. // CAUTION: DPI *is* important here - it DIFFERS from 75x75, usually 600x600 is used ScopedVclPtrInstance pPrerenderVDev(*Application::GetDefaultDevice()); pPrerenderVDev->SetOutputSizePixel(aScaledSize, false); pPrerenderVDev->SetReferenceDevice( mnDPIX, mnDPIY ); // calculate needed Scale for Metafile (using Size and DPI from VDev) Size aLogicSize( pPrerenderVDev->PixelToLogic( pPrerenderVDev->GetOutputSizePixel(), MapMode( MapUnit::Map100thMM ) ) ); Size aOrigSize( maOrigSize ); if( aOrigSize.Width() < 1 ) aOrigSize.setWidth( aLogicSize.Width() ); if( aOrigSize.Height() < 1 ) aOrigSize.setHeight( aLogicSize.Height() ); double fScale = double(aLogicSize.Width())/double(aOrigSize.Width()); // tdf#141761 // The display quality of the Preview is pretty ugly when // FormControls are used. I made a deep-dive why this happens, // and in principle the reason is the Mteafile::Scale used // below. Since Metafile actions are integer, that floating point // scale leads to rounding errors that make the lines painting // the FormControls disappear in the surrounding ClipRegions. // That Scale cannot be avoided since the Metafile contains it's // own SetMapMode commands which *will* be executed on ::Play, // so the ::Scale is the only possibility fr Metafile currently: // Giving a Size as parameter in ::Play will *not* work due to // the relativeMapMode that gets created will fail on // ::SetMapMode actions in the Metafile - and FormControls DO // use ::SetMapMode(MapPixel). // This can only be solved better in the future using Primitives // which would allow any scale by embedding to a Transformation, // but that would be a bigger rework. // Until then, use this little 'trick' to improve quality. // It uses the fact to empirically having tested that the quality // gets really bad for FormControls starting by a scale factor // smaller than 0.2 - that makes the ClipRegion overlap start. // So - for now - try not to go below that. static const double fMinimumScale(0.2); double fFactor(0.0); if(fScale < fMinimumScale) { fFactor = fMinimumScale / fScale; fScale = fMinimumScale; double fWidth(aScaledSize.getWidth() * fFactor); double fHeight(aScaledSize.getHeight() * fFactor); const double fNewNeededPixels(fWidth * fHeight); // to not risk using too big bitmaps and running into // memory problems, still limit to a useful factor is // necessary, also empirically estimated to // avoid the quality from collapsing (using a direct // in-between , ceil'd result) static const double fMaximumQualitySquare(1396221.0); if(fNewNeededPixels > fMaximumQualitySquare) { const double fCorrection(fMaximumQualitySquare/fNewNeededPixels); fWidth *= fCorrection; fHeight *= fCorrection; fScale *= fCorrection; } const Size aScaledSize2(basegfx::fround(fWidth), basegfx::fround(fHeight)); pPrerenderVDev->SetOutputSizePixel(aScaledSize2, false); aLogicSize = pPrerenderVDev->PixelToLogic( aScaledSize2, MapMode( MapUnit::Map100thMM ) ); } pPrerenderVDev->EnableOutput(); pPrerenderVDev->SetBackground( Wallpaper(COL_WHITE) ); pPrerenderVDev->Erase(); pPrerenderVDev->SetMapMode(MapMode(MapUnit::Map100thMM)); if( mbGreyscale ) pPrerenderVDev->SetDrawMode( pPrerenderVDev->GetDrawMode() | ( DrawModeFlags::GrayLine | DrawModeFlags::GrayFill | DrawModeFlags::GrayText | DrawModeFlags::GrayBitmap | DrawModeFlags::GrayGradient ) ); // Copy, Scale and Paint Metafile GDIMetaFile aMtf( maMtf ); aMtf.WindStart(); aMtf.Scale( fScale, fScale ); aMtf.WindStart(); aMtf.Play(*pPrerenderVDev, Point(0, 0), aLogicSize); pPrerenderVDev->SetMapMode(MapMode(MapUnit::MapPixel)); maPreviewBitmap = pPrerenderVDev->GetBitmapEx(Point(0, 0), pPrerenderVDev->GetOutputSizePixel()); if(0.0 != fFactor) { // Correct to needed size, BmpScaleFlag::Interpolate is acceptable, // but BmpScaleFlag::BestQuality is just better. In case of time // constraints, change to Interpolate would be possible maPreviewBitmap.Scale(aScaledSize, BmpScaleFlag::BestQuality); } } PrintDialog::ShowNupOrderWindow::ShowNupOrderWindow() : mnOrderMode( NupOrderType::LRTB ) , mnRows( 1 ) , mnColumns( 1 ) { } void PrintDialog::ShowNupOrderWindow::SetDrawingArea(weld::DrawingArea* pDrawingArea) { Size aSize(70, 70); pDrawingArea->set_size_request(aSize.Width(), aSize.Height()); CustomWidgetController::SetDrawingArea(pDrawingArea); SetOutputSizePixel(aSize); } void PrintDialog::ShowNupOrderWindow::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle& /*i_rRect*/) { rRenderContext.SetMapMode(MapMode(MapUnit::MapPixel)); rRenderContext.SetTextColor(rRenderContext.GetSettings().GetStyleSettings().GetFieldTextColor()); rRenderContext.SetBackground(Wallpaper(Application::GetSettings().GetStyleSettings().GetFieldColor())); rRenderContext.Erase(); int nPages = mnRows * mnColumns; Font aFont(rRenderContext.GetSettings().GetStyleSettings().GetFieldFont()); aFont.SetFontSize(Size(0, 24)); rRenderContext.SetFont(aFont); Size aSampleTextSize(rRenderContext.GetTextWidth(OUString::number(nPages + 1)), rRenderContext.GetTextHeight()); Size aOutSize(GetOutputSizePixel()); Size aSubSize(aOutSize.Width() / mnColumns, aOutSize.Height() / mnRows); // calculate font size: shrink the sample text so it fits double fX = double(aSubSize.Width()) / double(aSampleTextSize.Width()); double fY = double(aSubSize.Height()) / double(aSampleTextSize.Height()); double fScale = (fX < fY) ? fX : fY; tools::Long nFontHeight = tools::Long(24.0 * fScale) - 3; if (nFontHeight < 5) nFontHeight = 5; aFont.SetFontSize(Size( 0, nFontHeight)); rRenderContext.SetFont(aFont); tools::Long nTextHeight = rRenderContext.GetTextHeight(); for (int i = 0; i < nPages; i++) { OUString aPageText(OUString::number(i + 1)); int nX = 0, nY = 0; switch (mnOrderMode) { case NupOrderType::LRTB: nX = (i % mnColumns); nY = (i / mnColumns); break; case NupOrderType::TBLR: nX = (i / mnRows); nY = (i % mnRows); break; case NupOrderType::RLTB: nX = mnColumns - 1 - (i % mnColumns); nY = (i / mnColumns); break; case NupOrderType::TBRL: nX = mnColumns - 1 - (i / mnRows); nY = (i % mnRows); break; } Size aTextSize(rRenderContext.GetTextWidth(aPageText), nTextHeight); int nDeltaX = (aSubSize.Width() - aTextSize.Width()) / 2; int nDeltaY = (aSubSize.Height() - aTextSize.Height()) / 2; rRenderContext.DrawText(Point(nX * aSubSize.Width() + nDeltaX, nY * aSubSize.Height() + nDeltaY), aPageText); } DecorationView aDecorationView(&rRenderContext); aDecorationView.DrawFrame(tools::Rectangle(Point(0, 0), aOutSize), DrawFrameStyle::Group); } Size const & PrintDialog::getJobPageSize() { if( maFirstPageSize.IsEmpty() ) { maFirstPageSize = maNupPortraitSize; GDIMetaFile aMtf; if( maPController->getPageCountProtected() > 0 ) { PrinterController::PageSize aPageSize = maPController->getPageFile( 0, aMtf, true ); maFirstPageSize = aPageSize.aSize; } } return maFirstPageSize; } PrintDialog::PrintDialog(weld::Window* i_pWindow, std::shared_ptr i_xController) : GenericDialogController(i_pWindow, u"vcl/ui/printdialog.ui"_ustr, u"PrintDialog"_ustr) , maPController(std::move( i_xController )) , mxTabCtrl(m_xBuilder->weld_notebook(u"tabcontrol"_ustr)) , mxScrolledWindow(m_xBuilder->weld_scrolled_window(u"scrolledwindow"_ustr)) , mxPageLayoutFrame(m_xBuilder->weld_frame(u"layoutframe"_ustr)) , mxPrinters(m_xBuilder->weld_combo_box(u"printersbox"_ustr)) , mxStatusTxt(m_xBuilder->weld_label(u"status"_ustr)) , mxSetupButton(m_xBuilder->weld_button(u"setup"_ustr)) , mxCopyCountField(m_xBuilder->weld_spin_button(u"copycount"_ustr)) , mxCollateBox(m_xBuilder->weld_check_button(u"collate"_ustr)) , mxCollateImage(m_xBuilder->weld_image(u"collateimage"_ustr)) , mxPageRangeEdit(m_xBuilder->weld_entry(u"pagerange"_ustr)) , mxPageRangesRadioButton(m_xBuilder->weld_radio_button(u"rbRangePages"_ustr)) , mxPaperSidesBox(m_xBuilder->weld_combo_box(u"sidesbox"_ustr)) , mxSingleJobsBox(m_xBuilder->weld_check_button(u"singlejobs"_ustr)) , mxReverseOrderBox(m_xBuilder->weld_check_button(u"reverseorder"_ustr)) , mxOKButton(m_xBuilder->weld_button(u"ok"_ustr)) , mxCancelButton(m_xBuilder->weld_button(u"cancel"_ustr)) , mxBackwardBtn(m_xBuilder->weld_button(u"backward"_ustr)) , mxForwardBtn(m_xBuilder->weld_button(u"forward"_ustr)) , mxFirstBtn(m_xBuilder->weld_button(u"btnFirst"_ustr)) , mxLastBtn(m_xBuilder->weld_button(u"btnLast"_ustr)) , mxPreviewBox(m_xBuilder->weld_check_button(u"previewbox"_ustr)) , mxNumPagesText(m_xBuilder->weld_label(u"totalnumpages"_ustr)) , mxPreview(new PrintPreviewWindow(this)) , mxPreviewWindow(new weld::CustomWeld(*m_xBuilder, u"preview"_ustr, *mxPreview)) , mxPageEdit(m_xBuilder->weld_entry(u"pageedit"_ustr)) , mxPagesBtn(m_xBuilder->weld_radio_button(u"pagespersheetbtn"_ustr)) , mxBrochureBtn(m_xBuilder->weld_radio_button(u"brochure"_ustr)) , mxPagesBoxTitleTxt(m_xBuilder->weld_label(u"pagespersheettxt"_ustr)) , mxNupPagesBox(m_xBuilder->weld_combo_box(u"pagespersheetbox"_ustr)) , mxNupNumPagesTxt(m_xBuilder->weld_label(u"pagestxt"_ustr)) , mxNupColEdt(m_xBuilder->weld_spin_button(u"pagecols"_ustr)) , mxNupTimesTxt(m_xBuilder->weld_label(u"by"_ustr)) , mxNupRowsEdt(m_xBuilder->weld_spin_button(u"pagerows"_ustr)) , mxPageMarginTxt1(m_xBuilder->weld_label(u"pagemargintxt1"_ustr)) , mxPageMarginEdt(m_xBuilder->weld_metric_spin_button(u"pagemarginsb"_ustr, FieldUnit::MM)) , mxPageMarginTxt2(m_xBuilder->weld_label(u"pagemargintxt2"_ustr)) , mxSheetMarginTxt1(m_xBuilder->weld_label(u"sheetmargintxt1"_ustr)) , mxSheetMarginEdt(m_xBuilder->weld_metric_spin_button(u"sheetmarginsb"_ustr, FieldUnit::MM)) , mxSheetMarginTxt2(m_xBuilder->weld_label(u"sheetmargintxt2"_ustr)) , mxPaperSizeBox(m_xBuilder->weld_combo_box(u"papersizebox"_ustr)) , mxOrientationBox(m_xBuilder->weld_combo_box(u"pageorientationbox"_ustr)) , mxNupOrderTxt(m_xBuilder->weld_label(u"labelorder"_ustr)) , mxNupOrderBox(m_xBuilder->weld_combo_box(u"orderbox"_ustr)) , mxNupOrder(new ShowNupOrderWindow) , mxNupOrderWin(new weld::CustomWeld(*m_xBuilder, u"orderpreview"_ustr, *mxNupOrder)) , mxBorderCB(m_xBuilder->weld_check_button(u"bordercb"_ustr)) , mxRangeExpander(m_xBuilder->weld_expander(u"exRangeExpander"_ustr)) , mxLayoutExpander(m_xBuilder->weld_expander(u"exLayoutExpander"_ustr)) , mxCustom(m_xBuilder->weld_widget(u"customcontents"_ustr)) , maPrintToFileText( VclResId( SV_PRINT_TOFILE_TXT ) ) , maDefPrtText( VclResId( SV_PRINT_DEFPRT_TXT ) ) , maNoPageStr( VclResId( SV_PRINT_NOPAGES ) ) , maNoPreviewStr( VclResId( SV_PRINT_NOPREVIEW ) ) , mnCurPage( 0 ) , mnCachedPages( 0 ) , mbCollateAlwaysOff(false) , mbShowLayoutFrame( true ) , maUpdatePreviewIdle("Print Dialog Update Preview Idle") , maUpdatePreviewNoCacheIdle("Print Dialog Update Preview (no cache) Idle") { // save printbutton text, gets exchanged occasionally with print to file maPrintText = mxOKButton->get_label(); maPageStr = mxNumPagesText->get_label(); mxPrinters->append_text(maPrintToFileText); // fill printer listbox std::vector< OUString > rQueues( Printer::GetPrinterQueues() ); std::sort( rQueues.begin(), rQueues.end(), lcl_ListBoxCompare ); for( const auto& rQueue : rQueues ) { mxPrinters->append_text(rQueue); } // select current printer if (mxPrinters->find_text(maPController->getPrinter()->GetName()) != -1) mxPrinters->set_active_text(maPController->getPrinter()->GetName()); else { // fall back to last printer SettingsConfigItem* pItem = SettingsConfigItem::get(); OUString aValue( pItem->getValue( u"PrintDialog"_ustr, u"LastPrinter"_ustr ) ); if (mxPrinters->find_text(aValue) != -1) { mxPrinters->set_active_text(aValue); maPController->setPrinter( VclPtrInstance( aValue ) ); } else { // fall back to default printer mxPrinters->set_active_text(Printer::GetDefaultPrinterName()); maPController->setPrinter( VclPtrInstance( Printer::GetDefaultPrinterName() ) ); } } // not printing to file maPController->resetPrinterOptions( false ); // update the text fields for the printer updatePrinterText(); // setup dependencies checkControlDependencies(); // setup paper sides box setupPaperSidesBox(); // set initial focus to "Printer" mxPrinters->grab_focus(); // setup sizes for N-Up Size aNupSize( maPController->getPrinter()->PixelToLogic( maPController->getPrinter()->GetPaperSizePixel(), MapMode( MapUnit::Map100thMM ) ) ); if( maPController->getPrinter()->GetOrientation() == Orientation::Landscape ) { maNupLandscapeSize = aNupSize; // coverity[swapped_arguments : FALSE] - this is in the correct order maNupPortraitSize = Size( aNupSize.Height(), aNupSize.Width() ); } else { maNupPortraitSize = aNupSize; // coverity[swapped_arguments : FALSE] - this is in the correct order maNupLandscapeSize = Size( aNupSize.Height(), aNupSize.Width() ); } maUpdatePreviewIdle.SetPriority(TaskPriority::POST_PAINT); maUpdatePreviewIdle.SetInvokeHandler(LINK( this, PrintDialog, updatePreviewIdle)); maUpdatePreviewNoCacheIdle.SetPriority(TaskPriority::POST_PAINT); maUpdatePreviewNoCacheIdle.SetInvokeHandler(LINK(this, PrintDialog, updatePreviewNoCacheIdle)); initFromMultiPageSetup( maPController->getMultipage() ); updatePageSize(mxOrientationBox->get_active()); // setup optional UI options set by application setupOptionalUI(); // hide layout frame if unwanted mxPageLayoutFrame->set_visible(mbShowLayoutFrame); // restore settings from last run readFromSettings(); // setup click hdl mxOKButton->connect_clicked(LINK(this, PrintDialog, ClickHdl)); mxCancelButton->connect_clicked(LINK(this, PrintDialog, ClickHdl)); mxSetupButton->connect_clicked( LINK( this, PrintDialog, ClickHdl ) ); mxBackwardBtn->connect_clicked(LINK(this, PrintDialog, ClickHdl)); mxForwardBtn->connect_clicked(LINK(this, PrintDialog, ClickHdl)); mxFirstBtn->connect_clicked(LINK(this, PrintDialog, ClickHdl)); mxLastBtn->connect_clicked( LINK( this, PrintDialog, ClickHdl ) ); // setup toggle hdl mxReverseOrderBox->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); mxCollateBox->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); mxSingleJobsBox->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); mxBrochureBtn->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); mxPreviewBox->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); mxBorderCB->connect_toggled( LINK( this, PrintDialog, ToggleHdl ) ); // setup select hdl mxPrinters->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); mxPaperSidesBox->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); mxNupPagesBox->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); mxOrientationBox->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); mxNupOrderBox->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); mxPaperSizeBox->connect_changed( LINK( this, PrintDialog, SelectHdl ) ); // setup modify hdl mxPageEdit->connect_activate( LINK( this, PrintDialog, ActivateHdl ) ); mxPageEdit->connect_focus_out( LINK( this, PrintDialog, FocusOutHdl ) ); mxCopyCountField->connect_value_changed( LINK( this, PrintDialog, SpinModifyHdl ) ); mxNupColEdt->connect_value_changed( LINK( this, PrintDialog, SpinModifyHdl ) ); mxNupRowsEdt->connect_value_changed( LINK( this, PrintDialog, SpinModifyHdl ) ); mxPageMarginEdt->connect_value_changed( LINK( this, PrintDialog, MetricSpinModifyHdl ) ); mxSheetMarginEdt->connect_value_changed( LINK( this, PrintDialog, MetricSpinModifyHdl ) ); updateNupFromPages(); // tdf#129180 Delay setting the default value in the Paper Size list // set paper sizes listbox setPaperSizes(); mxRangeExpander->set_expanded( officecfg::Office::Common::Print::Dialog::RangeSectionExpanded::get()); mxLayoutExpander->set_expanded( officecfg::Office::Common::Print::Dialog::LayoutSectionExpanded::get()); // lock the dialog height, regardless of later expander state mxScrolledWindow->set_size_request( mxScrolledWindow->get_preferred_size().Width() + mxScrolledWindow->get_scroll_thickness(), 450); m_xDialog->set_centered_on_parent(true); } PrintDialog::~PrintDialog() { std::shared_ptr batch(comphelper::ConfigurationChanges::create()); officecfg::Office::Common::Print::Dialog::RangeSectionExpanded::set(mxRangeExpander->get_expanded(), batch); officecfg::Office::Common::Print::Dialog::LayoutSectionExpanded::set(mxLayoutExpander->get_expanded(), batch); batch->commit(); } void PrintDialog::setupPaperSidesBox() { DuplexMode eDuplex = maPController->getPrinter()->GetDuplexMode(); if ( eDuplex == DuplexMode::Unknown || isPrintToFile() ) { mxPaperSidesBox->set_active( 0 ); mxPaperSidesBox->set_sensitive( false ); } else { mxPaperSidesBox->set_active( static_cast(eDuplex) - 1 ); mxPaperSidesBox->set_sensitive( true ); } } void PrintDialog::storeToSettings() { SettingsConfigItem* pItem = SettingsConfigItem::get(); pItem->setValue( u"PrintDialog"_ustr, u"LastPrinter"_ustr, isPrintToFile() ? Printer::GetDefaultPrinterName() : mxPrinters->get_active_text() ); pItem->setValue( u"PrintDialog"_ustr, u"LastPage"_ustr, mxTabCtrl->get_tab_label_text(mxTabCtrl->get_current_page_ident())); pItem->setValue( u"PrintDialog"_ustr, u"WindowState"_ustr, m_xDialog->get_window_state(vcl::WindowDataMask::All) ); pItem->setValue( u"PrintDialog"_ustr, u"CopyCount"_ustr, mxCopyCountField->get_text() ); pItem->setValue( u"PrintDialog"_ustr, u"Collate"_ustr, mxCollateBox->get_active() ? u"true"_ustr : u"false"_ustr ); pItem->setValue( u"PrintDialog"_ustr, u"CollateSingleJobs"_ustr, mxSingleJobsBox->get_active() ? u"true"_ustr : u"false"_ustr ); pItem->setValue( u"PrintDialog"_ustr, u"HasPreview"_ustr, hasPreview() ? u"true"_ustr : u"false"_ustr ); pItem->Commit(); } void PrintDialog::readFromSettings() { SettingsConfigItem* pItem = SettingsConfigItem::get(); // read last selected tab page; if it exists, activate it OUString aValue = pItem->getValue( u"PrintDialog"_ustr, u"LastPage"_ustr ); sal_uInt16 nCount = mxTabCtrl->get_n_pages(); for (sal_uInt16 i = 0; i < nCount; ++i) { OUString sPageId = mxTabCtrl->get_page_ident(i); if (aValue == mxTabCtrl->get_tab_label_text(sPageId)) { mxTabCtrl->set_current_page(sPageId); break; } } // persistent window state aValue = pItem->getValue( u"PrintDialog"_ustr, u"WindowState"_ustr ); if (!aValue.isEmpty()) m_xDialog->set_window_state(aValue); // collate aValue = pItem->getValue( u"PrintDialog"_ustr, u"CollateBox"_ustr ); if( aValue.equalsIgnoreAsciiCase("alwaysoff") ) { mbCollateAlwaysOff = true; mxCollateBox->set_active( false ); mxCollateBox->set_sensitive( false ); } else { mbCollateAlwaysOff = false; aValue = pItem->getValue( u"PrintDialog"_ustr, u"Collate"_ustr ); mxCollateBox->set_active( aValue.equalsIgnoreAsciiCase("true") ); } // collate single jobs aValue = pItem->getValue( u"PrintDialog"_ustr, u"CollateSingleJobs"_ustr ); mxSingleJobsBox->set_active(aValue.equalsIgnoreAsciiCase("true")); // preview box aValue = pItem->getValue( u"PrintDialog"_ustr, u"HasPreview"_ustr ); if ( aValue.equalsIgnoreAsciiCase("false") ) mxPreviewBox->set_active( false ); else mxPreviewBox->set_active( true ); } void PrintDialog::setPaperSizes() { mxPaperSizeBox->clear(); mxPaperSizeBox->set_active(-1); VclPtr aPrt( maPController->getPrinter() ); mePaper = aPrt->GetPaper(); if ( isPrintToFile() ) { mxPaperSizeBox->set_sensitive( false ); } else { int nExactMatch = -1; int nSizeMatch = -1; int nRotatedSizeMatch = -1; Size aSizeOfPaper = aPrt->GetSizeOfPaper(); PaperInfo aPaperInfo(aSizeOfPaper.getWidth(), aSizeOfPaper.getHeight()); // coverity[swapped_arguments : FALSE] - this is in the correct order PaperInfo aRotatedPaperInfo(aSizeOfPaper.getHeight(), aSizeOfPaper.getWidth()); const LocaleDataWrapper& rLocWrap(Application::GetSettings().GetLocaleDataWrapper()); o3tl::Length eUnit = o3tl::Length::mm; int nDigits = 0; if( rLocWrap.getMeasurementSystemEnum() == MeasurementSystem::US ) { eUnit = o3tl::Length::in100; nDigits = 2; } for (int nPaper = 0; nPaper < aPrt->GetPaperInfoCount(); nPaper++) { PaperInfo aInfo = aPrt->GetPaperInfo( nPaper ); aInfo.doSloppyFit(true); Paper ePaper = aInfo.getPaper(); Size aSize = aPrt->GetPaperSize( nPaper ); Size aLogicPaperSize( o3tl::convert(aSize, o3tl::Length::mm100, eUnit) ); OUString aWidth( rLocWrap.getNum( aLogicPaperSize.Width(), nDigits ) ); OUString aHeight( rLocWrap.getNum( aLogicPaperSize.Height(), nDigits ) ); OUString aUnit = eUnit == o3tl::Length::mm ? u"mm"_ustr : u"in"_ustr; OUString aPaperName; // Paper sizes that we don't know of but the system printer driver lists are not "User // Defined". Displaying them as such is just confusing. if (ePaper != PAPER_USER) aPaperName = Printer::GetPaperName( ePaper ) + " "; aPaperName += aWidth + aUnit + " x " + aHeight + aUnit; mxPaperSizeBox->append_text(aPaperName); if (ePaper != PAPER_USER && ePaper == mePaper) nExactMatch = nPaper; if (ePaper == PAPER_USER && aInfo.sloppyEqual(aPaperInfo)) nSizeMatch = nPaper; if (ePaper == PAPER_USER && aInfo.sloppyEqual(aRotatedPaperInfo)) nRotatedSizeMatch = nPaper; } if (nExactMatch != -1) mxPaperSizeBox->set_active(nExactMatch); else if (nSizeMatch != -1) mxPaperSizeBox->set_active(nSizeMatch); else if (nRotatedSizeMatch != -1) mxPaperSizeBox->set_active(nRotatedSizeMatch); mxPaperSizeBox->set_sensitive( true ); } } void PrintDialog::updatePrinterText() { const OUString aDefPrt( Printer::GetDefaultPrinterName() ); const QueueInfo* pInfo = Printer::GetQueueInfo( mxPrinters->get_active_text(), true ); if( pInfo ) { // FIXME: status text OUString aStatus; if( aDefPrt == pInfo->GetPrinterName() ) aStatus = maDefPrtText; mxStatusTxt->set_label( aStatus ); } else { mxStatusTxt->set_label( OUString() ); } } void PrintDialog::setPreviewText() { OUString aNewText( maPageStr.replaceFirst( "%n", OUString::number( mnCachedPages ) ) ); mxNumPagesText->set_label( aNewText ); } IMPL_LINK_NOARG(PrintDialog, updatePreviewIdle, Timer*, void) { preparePreview(true); } IMPL_LINK_NOARG(PrintDialog, updatePreviewNoCacheIdle, Timer*, void) { preparePreview(false); } void PrintDialog::preparePreview( bool i_bMayUseCache ) { VclPtr aPrt( maPController->getPrinter() ); Size aCurPageSize = aPrt->PixelToLogic( aPrt->GetPaperSizePixel(), MapMode( MapUnit::Map100thMM ) ); // tdf#123076 Get paper size for the preview top label mePaper = aPrt->GetPaper(); GDIMetaFile aMtf; // page range may have changed depending on options sal_Int32 nPages = maPController->getFilteredPageCount(); mnCachedPages = nPages; if (!i_bMayUseCache) updatePageRange(nPages); setPreviewText(); if ( !hasPreview() ) { mxPreview->setPreview( aMtf, aCurPageSize, Printer::GetPaperName( mePaper ), maNoPreviewStr, aPrt->GetDPIX(), aPrt->GetDPIY(), aPrt->GetPrinterOptions().IsConvertToGreyscales() ); mxForwardBtn->set_sensitive( false ); mxBackwardBtn->set_sensitive( false ); mxFirstBtn->set_sensitive( false ); mxLastBtn->set_sensitive( false ); mxPageEdit->set_sensitive( false ); return; } if( mnCurPage >= nPages ) mnCurPage = nPages-1; if( mnCurPage < 0 ) mnCurPage = 0; mxPageEdit->set_text(OUString::number(mnCurPage + 1)); if( nPages > 0 ) { PrinterController::PageSize aPageSize = maPController->getFilteredPageFile( mnCurPage, aMtf, i_bMayUseCache ); aCurPageSize = aPrt->PixelToLogic(aPrt->GetPaperSizePixel(), MapMode(MapUnit::Map100thMM)); if( ! aPageSize.bFullPaper ) { const MapMode aMapMode( MapUnit::Map100thMM ); Point aOff( aPrt->PixelToLogic( aPrt->GetPageOffsetPixel(), aMapMode ) ); aMtf.Move( aOff.X(), aOff.Y() ); } // tdf#150561: page size may have changed so sync mePaper with it mePaper = aPrt->GetPaper(); } mxPreview->setPreview( aMtf, aCurPageSize, Printer::GetPaperName( mePaper ), nPages > 0 ? OUString() : maNoPageStr, aPrt->GetDPIX(), aPrt->GetDPIY(), aPrt->GetPrinterOptions().IsConvertToGreyscales() ); mxForwardBtn->set_sensitive( mnCurPage < nPages-1 ); mxBackwardBtn->set_sensitive( mnCurPage != 0 ); mxFirstBtn->set_sensitive( mnCurPage != 0 ); mxLastBtn->set_sensitive( mnCurPage < nPages-1 ); mxPageEdit->set_sensitive( nPages > 1 ); } void PrintDialog::updatePageRange(sal_Int32 nPages) { if (nPages > 0 && !mxPageRangesRadioButton->get_active()) { OUStringBuffer aBuf(32); aBuf.append("1"); if (nPages > 1) { aBuf.append("-" + OUString::number(nPages)); } OUString sRange = aBuf.makeStringAndClear(); mxPageRangeEdit->set_text(sRange); maPController->setValue(u"PageRange"_ustr, Any(sRange)); } } void PrintDialog::updatePageSize(int nOrientation) { VclPtr aPrt(maPController->getPrinter()); Size aSize; if (mxNupPagesBox->get_active_id() == "1") { PaperInfo aInfo = aPrt->GetPaperInfo(mxPaperSizeBox->get_active()); aSize = Size(aInfo.getWidth(), aInfo.getHeight()); if (aSize.IsEmpty()) aSize = aPrt->GetSizeOfPaper(); if (nOrientation != ORIENTATION_AUTOMATIC) { if ((nOrientation == ORIENTATION_PORTRAIT && aSize.Width() > aSize.Height()) || (nOrientation == ORIENTATION_LANDSCAPE && aSize.Width() < aSize.Height())) { // coverity[swapped_arguments : FALSE] - this is in the intended order aSize = Size(aSize.Height(), aSize.Width()); } } } else aSize = getJobPageSize(); aPrt->SetPrintPageSize(aSize); aPrt->SetUsePrintDialogSetting(true); } void PrintDialog::updateOrientationBox( const bool bAutomatic ) { if ( !bAutomatic ) { Orientation eOrientation = maPController->getPrinter()->GetOrientation(); mxOrientationBox->set_active( static_cast(eOrientation) + 1 ); } else if ( hasOrientationChanged() ) { mxOrientationBox->set_active( ORIENTATION_AUTOMATIC ); } } bool PrintDialog::hasOrientationChanged() const { const int nOrientation = mxOrientationBox->get_active(); const Orientation eOrientation = maPController->getPrinter()->GetOrientation(); return (nOrientation == ORIENTATION_LANDSCAPE && eOrientation == Orientation::Portrait) || (nOrientation == ORIENTATION_PORTRAIT && eOrientation == Orientation::Landscape); } // Always use this function to set paper orientation to make sure everything behaves well void PrintDialog::setPaperOrientation( Orientation eOrientation, bool fromUser ) { VclPtr aPrt( maPController->getPrinter() ); aPrt->SetOrientation( eOrientation ); maPController->setOrientationFromUser( eOrientation, fromUser ); } void PrintDialog::checkControlDependencies() { if (mxCopyCountField->get_value() > 1) { mxCollateBox->set_sensitive( !mbCollateAlwaysOff ); mxSingleJobsBox->set_sensitive( mxCollateBox->get_active() ); } else { mxCollateBox->set_sensitive( false ); mxSingleJobsBox->set_sensitive( false ); } OUString aImg(mxCollateBox->get_active() ? SV_PRINT_COLLATE_BMP : SV_PRINT_NOCOLLATE_BMP); mxCollateImage->set_from_icon_name(aImg); // enable setup button only for printers that can be setup bool bHaveSetup = maPController->getPrinter()->HasSupport( PrinterSupport::SetupDialog ); mxSetupButton->set_sensitive(bHaveSetup); } void PrintDialog::checkOptionalControlDependencies() { for( const auto& rEntry : maControlToPropertyMap ) { assert(rEntry.first); bool bShouldbeEnabled = maPController->isUIOptionEnabled( rEntry.second ); if (bShouldbeEnabled && dynamic_cast(rEntry.first)) { auto r_it = maControlToNumValMap.find( rEntry.first ); if( r_it != maControlToNumValMap.end() ) { bShouldbeEnabled = maPController->isUIChoiceEnabled( rEntry.second, r_it->second ); } } bool bIsEnabled = rEntry.first->get_sensitive(); // Enable does not do a change check first, so can be less cheap than expected if (bShouldbeEnabled != bIsEnabled) rEntry.first->set_sensitive( bShouldbeEnabled ); } } void PrintDialog::initFromMultiPageSetup( const vcl::PrinterController::MultiPageSetup& i_rMPS ) { mxNupOrderWin->show(); mxPagesBtn->set_active(true); mxBrochureBtn->hide(); // setup field units for metric fields const LocaleDataWrapper& rLocWrap(Application::GetSettings().GetLocaleDataWrapper()); FieldUnit eUnit = FieldUnit::MM; sal_uInt16 nDigits = 0; if( rLocWrap.getMeasurementSystemEnum() == MeasurementSystem::US ) { eUnit = FieldUnit::INCH; nDigits = 2; } // set units mxPageMarginEdt->set_unit( eUnit ); mxSheetMarginEdt->set_unit( eUnit ); // set precision mxPageMarginEdt->set_digits( nDigits ); mxSheetMarginEdt->set_digits( nDigits ); mxSheetMarginEdt->set_value( mxSheetMarginEdt->normalize( i_rMPS.nLeftMargin ), FieldUnit::MM_100TH ); mxPageMarginEdt->set_value( mxPageMarginEdt->normalize( i_rMPS.nHorizontalSpacing ), FieldUnit::MM_100TH ); mxBorderCB->set_active( i_rMPS.bDrawBorder ); mxNupRowsEdt->set_value( i_rMPS.nRows ); mxNupColEdt->set_value( i_rMPS.nColumns ); mxNupOrderBox->set_active( static_cast(i_rMPS.nOrder) ); if( i_rMPS.nRows != 1 || i_rMPS.nColumns != 1 ) { mxNupPagesBox->set_active( mxNupPagesBox->get_count()-1 ); showAdvancedControls( true ); mxNupOrder->setValues( i_rMPS.nOrder, i_rMPS.nColumns, i_rMPS.nRows ); } } void PrintDialog::updateNup( bool i_bMayUseCache ) { int nRows = mxNupRowsEdt->get_value(); int nCols = mxNupColEdt->get_value(); tools::Long nPageMargin = mxPageMarginEdt->denormalize(mxPageMarginEdt->get_value( FieldUnit::MM_100TH )); tools::Long nSheetMargin = mxSheetMarginEdt->denormalize(mxSheetMarginEdt->get_value( FieldUnit::MM_100TH )); PrinterController::MultiPageSetup aMPS; aMPS.nRows = nRows; aMPS.nColumns = nCols; aMPS.nLeftMargin = aMPS.nTopMargin = aMPS.nRightMargin = aMPS.nBottomMargin = nSheetMargin; aMPS.nHorizontalSpacing = aMPS.nVerticalSpacing = nPageMargin; aMPS.bDrawBorder = mxBorderCB->get_active(); aMPS.nOrder = static_cast(mxNupOrderBox->get_active()); int nOrientationMode = mxOrientationBox->get_active(); if( nOrientationMode == ORIENTATION_LANDSCAPE ) aMPS.aPaperSize = maNupLandscapeSize; else if( nOrientationMode == ORIENTATION_PORTRAIT ) aMPS.aPaperSize = maNupPortraitSize; else // automatic mode { updatePageSize(mxOrientationBox->get_active()); Size aPrintPageSize = maPController->getPrinter()->GetPrintPageSize(); // get size of first real page to see if it is portrait or landscape // we assume same page sizes for all the pages for this Size aPageSize = getJobPageSize(); if ((aPageSize.Width() < aPageSize.Height() && aPrintPageSize.Width() > aPrintPageSize.Height()) || (aPageSize.Width() > aPageSize.Height() && aPrintPageSize.Width() < aPrintPageSize.Height())) { tools::Long nTmp = aPageSize.Width(); aPageSize.setWidth(aPageSize.Height()); aPageSize.setHeight(nTmp); } Size aMultiSize( aPageSize.Width() * nCols, aPageSize.Height() * nRows ); if( aMultiSize.Width() > aMultiSize.Height() ) // fits better on landscape { aMPS.aPaperSize = maNupLandscapeSize; setPaperOrientation( Orientation::Landscape, false ); } else { aMPS.aPaperSize = maNupPortraitSize; setPaperOrientation( Orientation::Portrait, false ); } } maPController->setMultipage( aMPS ); mxNupOrder->setValues( aMPS.nOrder, nCols, nRows ); if (i_bMayUseCache) maUpdatePreviewIdle.Start(); else maUpdatePreviewNoCacheIdle.Start(); } void PrintDialog::updateNupFromPages( bool i_bMayUseCache ) { int nPages = mxNupPagesBox->get_active_id().toInt32(); int nRows = mxNupRowsEdt->get_value(); int nCols = mxNupColEdt->get_value(); tools::Long nPageMargin = mxPageMarginEdt->denormalize(mxPageMarginEdt->get_value( FieldUnit::MM_100TH )); tools::Long nSheetMargin = mxSheetMarginEdt->denormalize(mxSheetMarginEdt->get_value( FieldUnit::MM_100TH )); bool bCustom = false; if( nPages == 1 ) { nRows = nCols = 1; nSheetMargin = 0; nPageMargin = 0; } else if( nPages == 2 || nPages == 4 || nPages == 6 || nPages == 9 || nPages == 16 ) { Size aJobPageSize( getJobPageSize() ); bool bPortrait = aJobPageSize.Width() < aJobPageSize.Height(); if( nPages == 2 ) { if( bPortrait ) { nRows = 1; nCols = 2; } else { nRows = 2; nCols = 1; } } else if( nPages == 4 ) nRows = nCols = 2; else if( nPages == 6 ) { if( bPortrait ) { nRows = 2; nCols = 3; } else { nRows = 3; nCols = 2; } } else if( nPages == 9 ) nRows = nCols = 3; else if( nPages == 16 ) nRows = nCols = 4; nPageMargin = 0; nSheetMargin = 0; } else bCustom = true; if( nPages > 1 ) { // set upper limits for margins based on job page size and rows/columns Size aSize( getJobPageSize() ); // maximum sheet distance: 1/2 sheet tools::Long nHorzMax = aSize.Width()/2; tools::Long nVertMax = aSize.Height()/2; if( nSheetMargin > nHorzMax ) nSheetMargin = nHorzMax; if( nSheetMargin > nVertMax ) nSheetMargin = nVertMax; mxSheetMarginEdt->set_max( mxSheetMarginEdt->normalize( std::min(nHorzMax, nVertMax) ), FieldUnit::MM_100TH ); // maximum page distance nHorzMax = (aSize.Width() - 2*nSheetMargin); if( nCols > 1 ) nHorzMax /= (nCols-1); nVertMax = (aSize.Height() - 2*nSheetMargin); if( nRows > 1 ) nHorzMax /= (nRows-1); if( nPageMargin > nHorzMax ) nPageMargin = nHorzMax; if( nPageMargin > nVertMax ) nPageMargin = nVertMax; mxPageMarginEdt->set_max( mxSheetMarginEdt->normalize( std::min(nHorzMax, nVertMax ) ), FieldUnit::MM_100TH ); } mxNupRowsEdt->set_value( nRows ); mxNupColEdt->set_value( nCols ); mxPageMarginEdt->set_value( mxPageMarginEdt->normalize( nPageMargin ), FieldUnit::MM_100TH ); mxSheetMarginEdt->set_value( mxSheetMarginEdt->normalize( nSheetMargin ), FieldUnit::MM_100TH ); showAdvancedControls( bCustom ); updateNup( i_bMayUseCache ); } void PrintDialog::enableNupControls( bool bEnable ) { mxNupPagesBox->set_sensitive( bEnable ); mxNupNumPagesTxt->set_sensitive( bEnable ); mxNupColEdt->set_sensitive( bEnable ); mxNupTimesTxt->set_sensitive( bEnable ); mxNupRowsEdt->set_sensitive( bEnable ); mxPageMarginTxt1->set_sensitive( bEnable ); mxPageMarginEdt->set_sensitive( bEnable ); mxPageMarginTxt2->set_sensitive( bEnable ); mxSheetMarginTxt1->set_sensitive( bEnable ); mxSheetMarginEdt->set_sensitive( bEnable ); mxSheetMarginTxt2->set_sensitive( bEnable ); mxNupOrderTxt->set_sensitive( bEnable ); mxNupOrderBox->set_sensitive( bEnable ); mxNupOrderWin->set_sensitive( bEnable ); mxBorderCB->set_sensitive( bEnable ); } void PrintDialog::showAdvancedControls( bool i_bShow ) { mxNupNumPagesTxt->set_visible( i_bShow ); mxNupColEdt->set_visible( i_bShow ); mxNupTimesTxt->set_visible( i_bShow ); mxNupRowsEdt->set_visible( i_bShow ); mxPageMarginTxt1->set_visible( i_bShow ); mxPageMarginEdt->set_visible( i_bShow ); mxPageMarginTxt2->set_visible( i_bShow ); mxSheetMarginTxt1->set_visible( i_bShow ); mxSheetMarginEdt->set_visible( i_bShow ); mxSheetMarginTxt2->set_visible( i_bShow ); } namespace { void setHelpId( weld::Widget* i_pWindow, const Sequence< OUString >& i_rHelpIds, sal_Int32 i_nIndex ) { if( i_nIndex >= 0 && i_nIndex < i_rHelpIds.getLength() ) i_pWindow->set_help_id( i_rHelpIds.getConstArray()[i_nIndex] ); } void setHelpText( weld::Widget* i_pWindow, const Sequence< OUString >& i_rHelpTexts, sal_Int32 i_nIndex ) { // without a help text set and the correct smartID, // help texts will be retrieved from the online help system if( i_nIndex >= 0 && i_nIndex < i_rHelpTexts.getLength() ) i_pWindow->set_tooltip_text(i_rHelpTexts.getConstArray()[i_nIndex]); } } void PrintDialog::setupOptionalUI() { const Sequence< PropertyValue >& rOptions( maPController->getUIOptions() ); for( const auto& rOption : rOptions ) { if (rOption.Name == "OptionsUIFile") { OUString sOptionsUIFile; rOption.Value >>= sOptionsUIFile; mxCustomOptionsUIBuilder = Application::CreateBuilder(mxCustom.get(), sOptionsUIFile); std::unique_ptr xWindow = mxCustomOptionsUIBuilder->weld_container(u"box"_ustr); xWindow->set_help_id(u"vcl/ui/printdialog/PrintDialog"_ustr); xWindow->show(); continue; } Sequence< beans::PropertyValue > aOptProp; rOption.Value >>= aOptProp; // extract ui element OUString aCtrlType; OUString aID; OUString aText; OUString aPropertyName; Sequence< OUString > aChoices; Sequence< sal_Bool > aChoicesDisabled; Sequence< OUString > aHelpTexts; Sequence< OUString > aIDs; Sequence< OUString > aHelpIds; sal_Int64 nMinValue = 0, nMaxValue = 0; OUString aGroupingHint; for (const beans::PropertyValue& rEntry : aOptProp) { if ( rEntry.Name == "ID" ) { rEntry.Value >>= aIDs; aID = aIDs[0]; } if ( rEntry.Name == "Text" ) { rEntry.Value >>= aText; } else if ( rEntry.Name == "ControlType" ) { rEntry.Value >>= aCtrlType; } else if ( rEntry.Name == "Choices" ) { rEntry.Value >>= aChoices; } else if ( rEntry.Name == "ChoicesDisabled" ) { rEntry.Value >>= aChoicesDisabled; } else if ( rEntry.Name == "Property" ) { PropertyValue aVal; rEntry.Value >>= aVal; aPropertyName = aVal.Name; } else if ( rEntry.Name == "Enabled" ) { } else if ( rEntry.Name == "GroupingHint" ) { rEntry.Value >>= aGroupingHint; } else if ( rEntry.Name == "DependsOnName" ) { } else if ( rEntry.Name == "DependsOnEntry" ) { } else if ( rEntry.Name == "AttachToDependency" ) { } else if ( rEntry.Name == "MinValue" ) { rEntry.Value >>= nMinValue; } else if ( rEntry.Name == "MaxValue" ) { rEntry.Value >>= nMaxValue; } else if ( rEntry.Name == "HelpText" ) { if( ! (rEntry.Value >>= aHelpTexts) ) { OUString aHelpText; if( rEntry.Value >>= aHelpText ) { aHelpTexts.realloc( 1 ); *aHelpTexts.getArray() = aHelpText; } } } else if ( rEntry.Name == "HelpId" ) { if( ! (rEntry.Value >>= aHelpIds ) ) { OUString aHelpId; if( rEntry.Value >>= aHelpId ) { aHelpIds.realloc( 1 ); *aHelpIds.getArray() = aHelpId; } } } else if ( rEntry.Name == "HintNoLayoutPage" ) { bool bHasLayoutFrame = false; rEntry.Value >>= bHasLayoutFrame; mbShowLayoutFrame = !bHasLayoutFrame; } } if (aCtrlType == "Group") { aID = "custom"; weld::Container* pPage = mxTabCtrl->get_page(aID); if (!pPage) continue; mxTabCtrl->set_tab_label_text(aID, aText); // set help id if (aHelpIds.hasElements()) pPage->set_help_id(aHelpIds[0]); // set help text if (aHelpTexts.hasElements()) pPage->set_tooltip_text(aHelpTexts[0]); pPage->show(); } else if (aCtrlType == "Subgroup" && !aID.isEmpty()) { std::unique_ptr xWidget; // since 'New Print Dialog Design' fromwhich in calc is not a frame anymore if (aID == "fromwhich") { std::unique_ptr xLabel = m_xBuilder->weld_label(aID); xLabel->set_label(aText); xWidget = std::move(xLabel); } else { std::unique_ptr xFrame = m_xBuilder->weld_frame(aID); if (!xFrame && mxCustomOptionsUIBuilder) xFrame = mxCustomOptionsUIBuilder->weld_frame(aID); if (xFrame) { xFrame->set_label(aText); xWidget = std::move(xFrame); } } if (!xWidget) continue; // set help id setHelpId(xWidget.get(), aHelpIds, 0); // set help text setHelpText(xWidget.get(), aHelpTexts, 0); xWidget->show(); } // EVIL else if( aCtrlType == "Bool" && aGroupingHint == "LayoutPage" && aPropertyName == "PrintProspect" ) { mxBrochureBtn->set_label(aText); mxBrochureBtn->show(); bool bVal = false; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal ) pVal->Value >>= bVal; mxBrochureBtn->set_active( bVal ); mxBrochureBtn->set_sensitive( maPController->isUIOptionEnabled( aPropertyName ) && pVal != nullptr ); maPropertyToWindowMap[aPropertyName].emplace_back(mxBrochureBtn.get()); maControlToPropertyMap[mxBrochureBtn.get()] = aPropertyName; // set help id setHelpId( mxBrochureBtn.get(), aHelpIds, 0 ); // set help text setHelpText( mxBrochureBtn.get(), aHelpTexts, 0 ); } else if (aCtrlType == "Bool") { // add a check box std::unique_ptr xNewBox = m_xBuilder->weld_check_button(aID); if (!xNewBox && mxCustomOptionsUIBuilder) xNewBox = mxCustomOptionsUIBuilder->weld_check_button(aID); if (!xNewBox) continue; xNewBox->set_label( aText ); xNewBox->show(); bool bVal = false; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal ) pVal->Value >>= bVal; xNewBox->set_active( bVal ); xNewBox->connect_toggled( LINK( this, PrintDialog, UIOption_CheckHdl ) ); maExtraControls.emplace_back(std::move(xNewBox)); weld::Widget* pWidget = maExtraControls.back().get(); maPropertyToWindowMap[aPropertyName].emplace_back(pWidget); maControlToPropertyMap[pWidget] = aPropertyName; // set help id setHelpId(pWidget, aHelpIds, 0); // set help text setHelpText(pWidget, aHelpTexts, 0); } else if (aCtrlType == "Radio") { sal_Int32 nCurHelpText = 0; // iterate options sal_Int32 nSelectVal = 0; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal && pVal->Value.hasValue() ) pVal->Value >>= nSelectVal; for( sal_Int32 m = 0; m < aChoices.getLength(); m++ ) { aID = aIDs[m]; std::unique_ptr xBtn = m_xBuilder->weld_radio_button(aID); if (!xBtn && mxCustomOptionsUIBuilder) xBtn = mxCustomOptionsUIBuilder->weld_radio_button(aID); if (!xBtn) continue; xBtn->set_label( aChoices[m] ); xBtn->set_active( m == nSelectVal ); xBtn->connect_toggled( LINK( this, PrintDialog, UIOption_RadioHdl ) ); if( aChoicesDisabled.getLength() > m && aChoicesDisabled[m] ) xBtn->set_sensitive( false ); xBtn->show(); maExtraControls.emplace_back(std::move(xBtn)); weld::Widget* pWidget = maExtraControls.back().get(); maPropertyToWindowMap[ aPropertyName ].emplace_back(pWidget); maControlToPropertyMap[pWidget] = aPropertyName; maControlToNumValMap[pWidget] = m; // set help id setHelpId( pWidget, aHelpIds, nCurHelpText ); // set help text setHelpText( pWidget, aHelpTexts, nCurHelpText ); nCurHelpText++; } } else if ( aCtrlType == "List" ) { std::unique_ptr xList = m_xBuilder->weld_combo_box(aID); if (!xList && mxCustomOptionsUIBuilder) xList = mxCustomOptionsUIBuilder->weld_combo_box(aID); if (!xList) continue; // iterate options for (const auto& rChoice : aChoices) xList->append_text(rChoice); sal_Int32 nSelectVal = 0; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal && pVal->Value.hasValue() ) pVal->Value >>= nSelectVal; xList->set_active(nSelectVal); xList->connect_changed( LINK( this, PrintDialog, UIOption_SelectHdl ) ); xList->show(); maExtraControls.emplace_back(std::move(xList)); weld::Widget* pWidget = maExtraControls.back().get(); maPropertyToWindowMap[ aPropertyName ].emplace_back(pWidget); maControlToPropertyMap[pWidget] = aPropertyName; // set help id setHelpId( pWidget, aHelpIds, 0 ); // set help text setHelpText( pWidget, aHelpTexts, 0 ); } else if ( aCtrlType == "Range" ) { std::unique_ptr xField = m_xBuilder->weld_spin_button(aID); if (!xField && mxCustomOptionsUIBuilder) xField = mxCustomOptionsUIBuilder->weld_spin_button(aID); if (!xField) continue; // set min/max and current value if(nMinValue != nMaxValue) xField->set_range(nMinValue, nMaxValue); sal_Int64 nCurVal = 0; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal && pVal->Value.hasValue() ) pVal->Value >>= nCurVal; xField->set_value( nCurVal ); xField->connect_value_changed( LINK( this, PrintDialog, UIOption_SpinModifyHdl ) ); xField->show(); maExtraControls.emplace_back(std::move(xField)); weld::Widget* pWidget = maExtraControls.back().get(); maPropertyToWindowMap[ aPropertyName ].emplace_back(pWidget); maControlToPropertyMap[pWidget] = aPropertyName; // set help id setHelpId( pWidget, aHelpIds, 0 ); // set help text setHelpText( pWidget, aHelpTexts, 0 ); } else if (aCtrlType == "Edit") { std::unique_ptr xField = m_xBuilder->weld_entry(aID); if (!xField && mxCustomOptionsUIBuilder) xField = mxCustomOptionsUIBuilder->weld_entry(aID); if (!xField) continue; OUString aCurVal; PropertyValue* pVal = maPController->getValue( aPropertyName ); if( pVal && pVal->Value.hasValue() ) pVal->Value >>= aCurVal; xField->set_text( aCurVal ); xField->connect_changed( LINK( this, PrintDialog, UIOption_EntryModifyHdl ) ); xField->show(); maExtraControls.emplace_back(std::move(xField)); weld::Widget* pWidget = maExtraControls.back().get(); maPropertyToWindowMap[ aPropertyName ].emplace_back(pWidget); maControlToPropertyMap[pWidget] = aPropertyName; // set help id setHelpId( pWidget, aHelpIds, 0 ); // set help text setHelpText( pWidget, aHelpTexts, 0 ); } else { SAL_WARN( "vcl", "Unsupported UI option: \"" << aCtrlType << '"'); } } // #i106506# if no brochure button, then the singular Pages radio button // makes no sense, so replace it by a FixedText label if (!mxBrochureBtn->get_visible() && mxPagesBtn->get_visible()) { mxPagesBoxTitleTxt->set_label(mxPagesBtn->get_label()); mxPagesBoxTitleTxt->show(); mxPagesBtn->hide(); mxNupPagesBox->set_accessible_relation_labeled_by(mxPagesBoxTitleTxt.get()); } // update enable states checkOptionalControlDependencies(); // print range not shown (currently math only) -> hide spacer line and reverse order if (!mxPageRangeEdit->get_visible()) { mxReverseOrderBox->hide(); } if (!mxCustomOptionsUIBuilder) mxTabCtrl->remove_page(mxTabCtrl->get_page_ident(1)); } void PrintDialog::makeEnabled( weld::Widget* i_pWindow ) { auto it = maControlToPropertyMap.find( i_pWindow ); if( it != maControlToPropertyMap.end() ) { OUString aDependency( maPController->makeEnabled( it->second ) ); if( !aDependency.isEmpty() ) updateWindowFromProperty( aDependency ); } } void PrintDialog::updateWindowFromProperty( const OUString& i_rProperty ) { beans::PropertyValue* pValue = maPController->getValue( i_rProperty ); auto it = maPropertyToWindowMap.find( i_rProperty ); if( !(pValue && it != maPropertyToWindowMap.end()) ) return; const auto& rWindows( it->second ); if( rWindows.empty() ) return; bool bVal = false; sal_Int32 nVal = -1; if( pValue->Value >>= bVal ) { // we should have a CheckBox for this one weld::CheckButton* pBox = dynamic_cast(rWindows.front()); if( pBox ) { pBox->set_active( bVal ); } else if ( i_rProperty == "PrintProspect" ) { // EVIL special case if( bVal ) mxBrochureBtn->set_active(true); else mxPagesBtn->set_active(true); } else { SAL_WARN( "vcl", "missing a checkbox" ); } } else if( pValue->Value >>= nVal ) { // this could be a ListBox or a RadioButtonGroup weld::ComboBox* pList = dynamic_cast(rWindows.front()); if( pList ) { pList->set_active( static_cast< sal_uInt16 >(nVal) ); } else if( nVal >= 0 && o3tl::make_unsigned(nVal) < rWindows.size() ) { weld::RadioButton* pBtn = dynamic_cast(rWindows[nVal]); SAL_WARN_IF( !pBtn, "vcl", "unexpected control for property" ); if( pBtn ) pBtn->set_active(true); } } } bool PrintDialog::isPrintToFile() const { return ( mxPrinters->get_active() == 0 ); } bool PrintDialog::isCollate() const { return mxCopyCountField->get_value() > 1 && mxCollateBox->get_active(); } bool PrintDialog::isSingleJobs() const { return mxSingleJobsBox->get_active(); } bool PrintDialog::hasPreview() const { return mxPreviewBox->get_active(); } PropertyValue* PrintDialog::getValueForWindow( weld::Widget* i_pWindow ) const { PropertyValue* pVal = nullptr; auto it = maControlToPropertyMap.find( i_pWindow ); if( it != maControlToPropertyMap.end() ) { pVal = maPController->getValue( it->second ); SAL_WARN_IF( !pVal, "vcl", "property value not found" ); } else { OSL_FAIL( "changed control not in property map" ); } return pVal; } IMPL_LINK(PrintDialog, ToggleHdl, weld::Toggleable&, rButton, void) { if (&rButton == mxPreviewBox.get()) { maUpdatePreviewIdle.Start(); } else if( &rButton == mxBorderCB.get() ) { updateNup(); } else if (&rButton == mxSingleJobsBox.get()) { maPController->setValue( u"SinglePrintJobs"_ustr, Any( isSingleJobs() ) ); checkControlDependencies(); } else if( &rButton == mxCollateBox.get() ) { maPController->setValue( u"Collate"_ustr, Any( isCollate() ) ); checkControlDependencies(); } else if( &rButton == mxReverseOrderBox.get() ) { bool bChecked = mxReverseOrderBox->get_active(); maPController->setReversePrint( bChecked ); maPController->setValue( u"PrintReverse"_ustr, Any( bChecked ) ); maUpdatePreviewIdle.Start(); } else if (&rButton == mxBrochureBtn.get()) { PropertyValue* pVal = getValueForWindow(mxBrochureBtn.get()); if( pVal ) { bool bVal = mxBrochureBtn->get_active(); pVal->Value <<= bVal; checkOptionalControlDependencies(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } if (mxBrochureBtn->get_active()) { mxOrientationBox->set_sensitive( false ); mxOrientationBox->set_active( ORIENTATION_LANDSCAPE ); mxNupPagesBox->set_active( 0 ); updateNupFromPages(); showAdvancedControls( false ); enableNupControls( false ); } else { mxOrientationBox->set_sensitive( true ); mxOrientationBox->set_active( ORIENTATION_AUTOMATIC ); enableNupControls( true ); updateNupFromPages(); } } } IMPL_LINK(PrintDialog, ClickHdl, weld::Button&, rButton, void) { if (&rButton == mxOKButton.get() || &rButton == mxCancelButton.get()) { storeToSettings(); m_xDialog->response(&rButton == mxOKButton.get() ? RET_OK : RET_CANCEL); } else if( &rButton == mxForwardBtn.get() ) { previewForward(); } else if( &rButton == mxBackwardBtn.get() ) { previewBackward(); } else if( &rButton == mxFirstBtn.get() ) { previewFirst(); } else if( &rButton == mxLastBtn.get() ) { previewLast(); } else { if( &rButton == mxSetupButton.get() ) { maPController->setupPrinter(m_xDialog.get()); if ( !isPrintToFile() ) { VclPtr aPrt( maPController->getPrinter() ); mePaper = aPrt->GetPaper(); for (int nPaper = 0; nPaper < aPrt->GetPaperInfoCount(); nPaper++ ) { PaperInfo aInfo = aPrt->GetPaperInfo( nPaper ); aInfo.doSloppyFit(true); Paper ePaper = aInfo.getPaper(); if ( mePaper == ePaper ) { mxPaperSizeBox->set_active( nPaper ); break; } } } updateOrientationBox( false ); updatePageSize(mxOrientationBox->get_active()); setupPaperSidesBox(); // tdf#63905 don't use cache: page size may change maUpdatePreviewNoCacheIdle.Start(); } checkControlDependencies(); } } IMPL_LINK( PrintDialog, SelectHdl, weld::ComboBox&, rBox, void ) { if (&rBox == mxPrinters.get()) { if ( !isPrintToFile() ) { OUString aNewPrinter(rBox.get_active_text()); // set new printer maPController->setPrinter( VclPtrInstance( aNewPrinter ) ); maPController->resetPrinterOptions( false ); // invalidate page cache and start fresh maPController->invalidatePageCache(); maFirstPageSize = Size(); updateOrientationBox(); // update text fields mxOKButton->set_label(maPrintText); updatePrinterText(); updateNup(false); setPaperSizes(); maUpdatePreviewIdle.Start(); } else // print to file { // use the default printer or FIXME: the last used one? maPController->setPrinter( VclPtrInstance( Printer::GetDefaultPrinterName() ) ); mxOKButton->set_label(maPrintToFileText); maPController->resetPrinterOptions( true ); setPaperSizes(); updateOrientationBox(); maUpdatePreviewIdle.Start(); } updatePageSize(mxOrientationBox->get_active()); setupPaperSidesBox(); } else if ( &rBox == mxPaperSidesBox.get() ) { DuplexMode eDuplex = static_cast(mxPaperSidesBox->get_active() + 1); maPController->getPrinter()->SetDuplexMode( eDuplex ); } else if( &rBox == mxOrientationBox.get() ) { int nOrientation = mxOrientationBox->get_active(); if ( nOrientation != ORIENTATION_AUTOMATIC ) setPaperOrientation( static_cast( nOrientation - 1 ), true ); updatePageSize(nOrientation); updateNup( false ); } else if ( &rBox == mxNupOrderBox.get() ) { updateNup(); } else if( &rBox == mxNupPagesBox.get() ) { if( !mxPagesBtn->get_active() ) mxPagesBtn->set_active(true); updatePageSize(mxOrientationBox->get_active()); updateNupFromPages( false ); } else if ( &rBox == mxPaperSizeBox.get() ) { VclPtr aPrt( maPController->getPrinter() ); PaperInfo aInfo = aPrt->GetPaperInfo( rBox.get_active() ); aInfo.doSloppyFit(true); mePaper = aInfo.getPaper(); if ( mePaper == PAPER_USER ) aPrt->SetPaperSizeUser( Size( aInfo.getWidth(), aInfo.getHeight() ) ); else aPrt->SetPaper( mePaper ); maPController->setPaperSizeFromUser( Size( aInfo.getWidth(), aInfo.getHeight() ) ); updatePageSize(mxOrientationBox->get_active()); int nOrientation = mxOrientationBox->get_active(); if (nOrientation != ORIENTATION_AUTOMATIC) setPaperOrientation(static_cast(nOrientation - 1), true); maUpdatePreviewNoCacheIdle.Start(); } } IMPL_LINK_NOARG(PrintDialog, MetricSpinModifyHdl, weld::MetricSpinButton&, void) { checkControlDependencies(); updateNupFromPages(); } IMPL_LINK_NOARG(PrintDialog, FocusOutHdl, weld::Widget&, void) { ActivateHdl(*mxPageEdit); } IMPL_LINK_NOARG(PrintDialog, ActivateHdl, weld::Entry&, bool) { sal_Int32 nPage = mxPageEdit->get_text().toInt32(); if (nPage < 1) { nPage = 1; mxPageEdit->set_text(u"1"_ustr); } else if (nPage > mnCachedPages) { nPage = mnCachedPages; mxPageEdit->set_text(OUString::number(mnCachedPages)); } int nNewCurPage = nPage - 1; if (nNewCurPage != mnCurPage) { mnCurPage = nNewCurPage; maUpdatePreviewIdle.Start(); } return true; } IMPL_LINK( PrintDialog, SpinModifyHdl, weld::SpinButton&, rEdit, void ) { checkControlDependencies(); if (&rEdit == mxNupRowsEdt.get() || &rEdit == mxNupColEdt.get()) { updateNupFromPages(); } else if( &rEdit == mxCopyCountField.get() ) { maPController->setValue( u"CopyCount"_ustr, Any( sal_Int32(mxCopyCountField->get_value()) ) ); maPController->setValue( u"Collate"_ustr, Any( isCollate() ) ); } } IMPL_LINK( PrintDialog, UIOption_CheckHdl, weld::Toggleable&, i_rBox, void ) { PropertyValue* pVal = getValueForWindow( &i_rBox ); if( pVal ) { makeEnabled( &i_rBox ); bool bVal = i_rBox.get_active(); pVal->Value <<= bVal; checkOptionalControlDependencies(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } } IMPL_LINK( PrintDialog, UIOption_RadioHdl, weld::Toggleable&, i_rBtn, void ) { // this handler gets called for all radiobuttons that get unchecked, too // however we only want one notification for the new value (that is for // the button that gets checked) if( !i_rBtn.get_active() ) return; PropertyValue* pVal = getValueForWindow( &i_rBtn ); auto it = maControlToNumValMap.find( &i_rBtn ); if( !(pVal && it != maControlToNumValMap.end()) ) return; makeEnabled( &i_rBtn ); sal_Int32 nVal = it->second; pVal->Value <<= nVal; updateOrientationBox(); checkOptionalControlDependencies(); // tdf#41205 give focus to the page range edit if the corresponding radio button was selected if (pVal->Name == "PrintContent" && mxPageRangesRadioButton->get_active()) mxPageRangeEdit->grab_focus(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } IMPL_LINK( PrintDialog, UIOption_SelectHdl, weld::ComboBox&, i_rBox, void ) { PropertyValue* pVal = getValueForWindow( &i_rBox ); if( !pVal ) return; makeEnabled( &i_rBox ); sal_Int32 nVal( i_rBox.get_active() ); pVal->Value <<= nVal; //If we are in impress we start in print slides mode and get a //maFirstPageSize for slides which are usually landscape mode, if we //change to notes which are usually in portrait mode, and then visit //n-up print, we will assume notes are in landscape unless we throw //away maFirstPageSize when we change page content type if (pVal->Name == "PageContentType") { maFirstPageSize = Size(); css::uno::Sequence aChoicesDisabled{ false, // Original size false, // Fit to printable page (nVal == 2) /*Notes*/ ? true : false, // disable/enable Multiple sheets of paper (nVal == 2) /*Notes*/ ? true : false // disable/enable Tile sheet of paper }; maPController->setUIChoicesDisabled(u"PageOptions"_ustr, aChoicesDisabled); } checkOptionalControlDependencies(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } IMPL_LINK( PrintDialog, UIOption_SpinModifyHdl, weld::SpinButton&, i_rBox, void ) { PropertyValue* pVal = getValueForWindow( &i_rBox ); if( pVal ) { makeEnabled( &i_rBox ); sal_Int64 nVal = i_rBox.get_value(); pVal->Value <<= nVal; checkOptionalControlDependencies(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } } IMPL_LINK( PrintDialog, UIOption_EntryModifyHdl, weld::Entry&, i_rBox, void ) { PropertyValue* pVal = getValueForWindow( &i_rBox ); if( pVal && mxPageRangesRadioButton->get_active() ) { makeEnabled( &i_rBox ); OUString aVal( i_rBox.get_text() ); pVal->Value <<= aVal; checkOptionalControlDependencies(); // update preview and page settings maUpdatePreviewNoCacheIdle.Start(); } } void PrintDialog::previewForward() { sal_Int32 nValue = mxPageEdit->get_text().toInt32() + 1; if (nValue <= mnCachedPages) { mxPageEdit->set_text(OUString::number(nValue)); ActivateHdl(*mxPageEdit); } } void PrintDialog::previewBackward() { sal_Int32 nValue = mxPageEdit->get_text().toInt32() - 1; if (nValue >= 1) { mxPageEdit->set_text(OUString::number(nValue)); ActivateHdl(*mxPageEdit); } } void PrintDialog::previewFirst() { mxPageEdit->set_text(u"1"_ustr); ActivateHdl(*mxPageEdit); } void PrintDialog::previewLast() { mxPageEdit->set_text(OUString::number(mnCachedPages)); ActivateHdl(*mxPageEdit); } static OUString getNewLabel(const OUString& aLabel, int i_nCurr, int i_nMax) { OUString aNewText( aLabel.replaceFirst( "%p", OUString::number( i_nCurr ) ) ); aNewText = aNewText.replaceFirst( "%n", OUString::number( i_nMax ) ); return aNewText; } // PrintProgressDialog PrintProgressDialog::PrintProgressDialog(weld::Window* i_pParent, int i_nMax) : GenericDialogController(i_pParent, u"vcl/ui/printprogressdialog.ui"_ustr, u"PrintProgressDialog"_ustr) , mbCanceled(false) , mnCur(0) , mnMax(i_nMax) , mxText(m_xBuilder->weld_label(u"label"_ustr)) , mxProgress(m_xBuilder->weld_progress_bar(u"progressbar"_ustr)) , mxButton(m_xBuilder->weld_button(u"cancel"_ustr)) { if( mnMax < 1 ) mnMax = 1; maStr = mxText->get_label(); //just multiply largest value by 10 and take the width of that string as //the max size we will want mxText->set_label(getNewLabel(maStr, mnMax * 10, mnMax * 10)); mxText->set_size_request(mxText->get_preferred_size().Width(), -1); //Pick a useful max width mxProgress->set_size_request(mxProgress->get_approximate_digit_width() * 25, -1); mxButton->connect_clicked( LINK( this, PrintProgressDialog, ClickHdl ) ); // after this patch f7157f04fab298423e2c4f6a7e5f8e361164b15f, we have seen the calc Max string (sometimes) look above // now init to the right start values mxText->set_label(getNewLabel(maStr, mnCur, mnMax)); } PrintProgressDialog::~PrintProgressDialog() { } IMPL_LINK_NOARG(PrintProgressDialog, ClickHdl, weld::Button&, void) { mbCanceled = true; } void PrintProgressDialog::setProgress( int i_nCurrent ) { mnCur = i_nCurrent; if( mnMax < 1 ) mnMax = 1; mxText->set_label(getNewLabel(maStr, mnCur, mnMax)); // here view the dialog, with the right label mxProgress->set_percentage(mnCur*100/mnMax); } void PrintProgressDialog::tick() { if( mnCur < mnMax ) setProgress( ++mnCur ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */