/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * 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 "pdffilter.hxx" #include "pdfexport.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star::io; PDFFilter::PDFFilter( const Reference< XComponentContext > &rxContext ) : mxContext( rxContext ) { } PDFFilter::~PDFFilter() { } bool PDFFilter::implExport( const Sequence< PropertyValue >& rDescriptor ) { Reference< XOutputStream > xOStm; Sequence< PropertyValue > aFilterData; OUString aFilterOptions; sal_Int32 nLength = rDescriptor.getLength(); const PropertyValue* pValue = rDescriptor.getConstArray(); bool bIsRedactMode = false; bool bRet = false; Reference< task::XStatusIndicator > xStatusIndicator; Reference< task::XInteractionHandler > xIH; for (sal_Int32 i = 0; i < nLength; ++i) { if ( pValue[ i ].Name == "OutputStream" ) pValue[ i ].Value >>= xOStm; else if ( pValue[ i ].Name == "FilterData" ) pValue[ i ].Value >>= aFilterData; else if ( pValue[ i ].Name == "FilterOptions" ) pValue[ i ].Value >>= aFilterOptions; else if ( pValue[ i ].Name == "StatusIndicator" ) pValue[ i ].Value >>= xStatusIndicator; else if ( pValue[i].Name == "InteractionHandler" ) pValue[i].Value >>= xIH; } for (sal_Int32 i = 0 ; i < nLength; ++i) { if ( pValue[i].Name == "IsRedactMode") pValue[i].Value >>= bIsRedactMode; } if (!aFilterData.hasElements() && aFilterOptions.startsWith("{")) { try { // Allow setting filter data keys from the cmdline. std::vector aData = comphelper::JsonToPropertyValues(aFilterOptions.toUtf8()); aFilterData = comphelper::containerToSequence(aData); } catch (const boost::property_tree::json_parser::json_parser_error& e) { // This wasn't a valid json; maybe came from import filter (tdf#150846) SAL_WARN("filter.pdf", "error parsing FilterOptions: " << e.message()); } } /* we don't get FilterData if we are exporting directly to pdf, but we have to use the last user settings (especially for the CompressMode) */ if ( !aFilterData.hasElements() ) { FilterConfigItem aCfgItem( u"Office.Common/Filter/PDF/Export/" ); aCfgItem.ReadBool( "UseLosslessCompression", false ); aCfgItem.ReadInt32( "Quality", 90 ); aCfgItem.ReadBool( "ReduceImageResolution", false ); aCfgItem.ReadInt32( "MaxImageResolution", 300 ); aCfgItem.ReadBool( "UseTaggedPDF", false ); aCfgItem.ReadInt32( "SelectPdfVersion", 0 ); aCfgItem.ReadBool("PDFUACompliance", false); aCfgItem.ReadBool( "ExportNotes", false ); aCfgItem.ReadBool( "ExportNotesInMargin", false ); aCfgItem.ReadBool( "ExportPlaceholders", false ); aCfgItem.ReadBool( "ExportNotesPages", false ); aCfgItem.ReadBool( "ExportOnlyNotesPages", false ); aCfgItem.ReadBool( "UseTransitionEffects", true ); aCfgItem.ReadBool( "IsSkipEmptyPages", false ); aCfgItem.ReadBool( "ExportFormFields", true ); aCfgItem.ReadInt32( "FormsType", 0 ); aCfgItem.ReadBool( "HideViewerToolbar", false ); aCfgItem.ReadBool( "HideViewerMenubar", false ); aCfgItem.ReadBool( "HideViewerWindowControls", false ); aCfgItem.ReadBool( "ResizeWindowToInitialPage", false ); aCfgItem.ReadBool( "CenterWindow", false ); aCfgItem.ReadBool( "OpenInFullScreenMode", false ); aCfgItem.ReadBool( "DisplayPDFDocumentTitle", true ); aCfgItem.ReadInt32( "InitialView", 0 ); aCfgItem.ReadInt32( "Magnification", 0 ); aCfgItem.ReadInt32( "Zoom", 100 ); aCfgItem.ReadInt32( "PageLayout", 0 ); aCfgItem.ReadBool( "FirstPageOnLeft", false ); aCfgItem.ReadInt32( "InitialPage", 1 ); aCfgItem.ReadBool( "IsAddStream", false ); // the encryption is not available when exporting directly, since the encryption is off by default and the selection // (encrypt or not) is not persistent; it's available through macro though, // provided the correct property values are set, see help // now, the relative link stuff aCfgItem.ReadBool( "ExportLinksRelativeFsys", false ); aCfgItem.ReadInt32("PDFViewSelection", 0 ); aCfgItem.ReadBool( "ConvertOOoTargetToPDFTarget", false ); aCfgItem.ReadBool( "ExportBookmarksToPDFDestination", false ); aCfgItem.ReadBool( "ExportBookmarks", true ); aCfgItem.ReadBool( "ExportHiddenSlides", false ); aCfgItem.ReadBool( "SinglePageSheets", false ); aCfgItem.ReadInt32( "OpenBookmarkLevels", -1 ); aCfgItem.ReadBool( "IsRedactMode", false); aFilterData = aCfgItem.GetFilterData(); } if (bIsRedactMode) { bool bFound = false; for (PropertyValue& rProp : asNonConstRange(aFilterData)) { if (rProp.Name == "IsRedactMode") { rProp.Value <<= bIsRedactMode; bFound = true; break; } } if (!bFound) { sal_Int32 nNewSize = aFilterData.getLength() + 1; aFilterData.realloc( nNewSize ); auto pFilterData = aFilterData.getArray(); pFilterData[nNewSize - 1].Name = "IsRedactMode"; pFilterData[nNewSize - 1].Value <<= bIsRedactMode; } } if( mxSrcDoc.is() && xOStm.is() ) { PDFExport aExport( mxSrcDoc, xStatusIndicator, xIH, mxContext ); ::utl::TempFileNamed aTempFile; aTempFile.EnableKillingFile(); bRet = aExport.Export( aTempFile.GetURL(), aFilterData ); if( bRet ) { std::unique_ptr pIStm(::utl::UcbStreamHelper::CreateStream( aTempFile.GetURL(), StreamMode::READ )); if( pIStm ) { SvOutputStream aOStm( xOStm ); aOStm.WriteStream( *pIStm ); bRet = ( aOStm.Tell() && ( aOStm.GetError() == ERRCODE_NONE ) ); } } } return bRet; } namespace { class FocusWindowWaitCursor { private: VclPtr m_pFocusWindow; public: FocusWindowWaitCursor() : m_pFocusWindow( Application::GetFocusWindow() ) { if( m_pFocusWindow ) { m_pFocusWindow->AddEventListener( LINK( this, FocusWindowWaitCursor, DestroyedLink ) ); m_pFocusWindow->EnterWait(); } } ~FocusWindowWaitCursor() { if( m_pFocusWindow ) { m_pFocusWindow->LeaveWait(); m_pFocusWindow->RemoveEventListener( LINK( this, FocusWindowWaitCursor, DestroyedLink ) ); } } DECL_LINK( DestroyedLink, VclWindowEvent&, void ); }; } IMPL_LINK( FocusWindowWaitCursor, DestroyedLink, VclWindowEvent&, rEvent, void ) { if( rEvent.GetId() == VclEventId::ObjectDying ) m_pFocusWindow = nullptr; } sal_Bool SAL_CALL PDFFilter::filter( const Sequence< PropertyValue >& rDescriptor ) { FocusWindowWaitCursor aCur; SalLayoutGlyphsCache::self()->SetCacheGlyphsWhenDoingFallbackFonts(true); const bool bRet = implExport( rDescriptor ); SalLayoutGlyphsCache::self()->SetCacheGlyphsWhenDoingFallbackFonts(false); return bRet; } void SAL_CALL PDFFilter::cancel( ) { } void SAL_CALL PDFFilter::setSourceDocument( const Reference< XComponent >& xDoc ) { mxSrcDoc = xDoc; } void SAL_CALL PDFFilter::initialize( const css::uno::Sequence< css::uno::Any >& ) { } OUString SAL_CALL PDFFilter::getImplementationName() { return "com.sun.star.comp.PDF.PDFFilter"; } sal_Bool SAL_CALL PDFFilter::supportsService( const OUString& rServiceName ) { return cppu::supportsService( this, rServiceName ); } css::uno::Sequence< OUString > SAL_CALL PDFFilter::getSupportedServiceNames( ) { return { "com.sun.star.document.PDFFilter" }; } extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface* filter_PDFFilter_get_implementation( css::uno::XComponentContext* context, css::uno::Sequence const&) { return cppu::acquire(new PDFFilter(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */