/* -*- 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 "SlideRenderer.hxx" #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; namespace sd::presenter { //===== SlideRenderer ========================================================== SlideRenderer::SlideRenderer () { } SlideRenderer::~SlideRenderer() { } //----- XInitialization ------------------------------------------------------- void SAL_CALL SlideRenderer::initialize (const Sequence& rArguments) { ThrowIfDisposed(); if (rArguments.hasElements()) { throw RuntimeException(u"SlideRenderer: invalid number of arguments"_ustr, static_cast(this)); } } OUString SlideRenderer::getImplementationName() { return u"com.sun.star.comp.Draw.SlideRenderer"_ustr; } sal_Bool SlideRenderer::supportsService(OUString const & ServiceName) { return cppu::supportsService(this, ServiceName); } css::uno::Sequence SlideRenderer::getSupportedServiceNames() { return {u"com.sun.star.drawing.SlideRenderer"_ustr}; } //----- XSlideRenderer -------------------------------------------------------- Reference SlideRenderer::createPreview ( const Reference& rxSlide, const awt::Size& rMaximalSize, sal_Int16 nSuperSampleFactor) { ThrowIfDisposed(); SolarMutexGuard aGuard; return VCLUnoHelper::CreateBitmap( CreatePreview(rxSlide, rMaximalSize, nSuperSampleFactor)); } Reference SlideRenderer::createPreviewForCanvas ( const Reference& rxSlide, const awt::Size& rMaximalSize, sal_Int16 nSuperSampleFactor, const Reference& rxCanvas) { ThrowIfDisposed(); SolarMutexGuard aGuard; cppcanvas::CanvasSharedPtr pCanvas ( cppcanvas::VCLFactory::createCanvas(rxCanvas)); if (pCanvas) return cppcanvas::VCLFactory::createBitmap( pCanvas, CreatePreview(rxSlide, rMaximalSize, nSuperSampleFactor))->getUNOBitmap(); else return nullptr; } awt::Size SAL_CALL SlideRenderer::calculatePreviewSize ( double nSlideAspectRatio, const awt::Size& rMaximalSize) { if (rMaximalSize.Width <= 0 || rMaximalSize.Height <= 0 || nSlideAspectRatio <= 0) { return awt::Size(0,0); } const double nWindowAspectRatio (double(rMaximalSize.Width) / double(rMaximalSize.Height)); if (nSlideAspectRatio < nWindowAspectRatio) return awt::Size( sal::static_int_cast(rMaximalSize.Height * nSlideAspectRatio), rMaximalSize.Height); else return awt::Size( rMaximalSize.Width, sal::static_int_cast(rMaximalSize.Width / nSlideAspectRatio)); } BitmapEx SlideRenderer::CreatePreview ( const Reference& rxSlide, const awt::Size& rMaximalSize, sal_Int16 nSuperSampleFactor) { const SdPage* pPage = SdPage::getImplementation(rxSlide); if (pPage == nullptr) throw lang::IllegalArgumentException(u"SlideRenderer::createPreview() called with invalid slide"_ustr, static_cast(this), 0); // Determine the size of the current slide and its aspect ratio. Size aPageSize = pPage->GetSize(); if (aPageSize.Height() <= 0) throw lang::IllegalArgumentException(u"SlideRenderer::createPreview() called with invalid size"_ustr, static_cast(this), 1); // Compare with the aspect ratio of the window (which rMaximalSize // assumed to be) and calculate the size of the preview so that it // a) will have the aspect ratio of the page and // b) will be as large as possible. awt::Size aPreviewSize (calculatePreviewSize( double(aPageSize.Width()) / double(aPageSize.Height()), rMaximalSize)); if (aPreviewSize.Width <= 0 || aPreviewSize.Height <= 0) return BitmapEx(); // Make sure that the super sample factor has a sane value. sal_Int16 nFactor (nSuperSampleFactor); if (nFactor < 1) nFactor = 1; else if (nFactor > 10) nFactor = 10; // Create the preview. When the super sample factor n is greater than 1 // then a preview is created in size (n*width, n*height) and then scaled // down to (width, height). This is a poor mans antialiasing for the // time being. When we have true antialiasing support this workaround // can be removed. const Image aPreview = maPreviewRenderer.RenderPage ( pPage, Size(aPreviewSize.Width*nFactor, aPreviewSize.Height*nFactor), true); if (nFactor == 1) return aPreview.GetBitmapEx(); else { BitmapEx aScaledPreview = aPreview.GetBitmapEx(); aScaledPreview.Scale( Size(aPreviewSize.Width,aPreviewSize.Height), BmpScaleFlag::BestQuality); return aScaledPreview; } } void SlideRenderer::ThrowIfDisposed() { if (m_bDisposed) { throw lang::DisposedException (u"SlideRenderer object has already been disposed"_ustr, static_cast(this)); } } } // end of namespace ::sd::presenter extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface* com_sun_star_comp_Draw_SlideRenderer_get_implementation(css::uno::XComponentContext*, css::uno::Sequence const &) { return cppu::acquire(new sd::presenter::SlideRenderer); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */