1
0
Fork 0
libreoffice/slideshow/source/engine/shapes/appletshape.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

299 lines
11 KiB
C++

/* -*- 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 "appletshape.hxx"
#include "externalshapebase.hxx"
#include "viewappletshape.hxx"
#include <tools.hxx>
#include <o3tl/safeint.hxx>
#include <osl/diagnose.h>
#include <algorithm>
#include <utility>
using namespace ::com::sun::star;
namespace slideshow::internal
{
namespace {
/** Represents an applet shape.
This implementation offers support for applet shapes (both
Java applets, and Netscape plugins). Such shapes need
special treatment.
*/
class AppletShape : public ExternalShapeBase
{
public:
/** Create a shape for the given XShape for an applet object
@param xShape
The XShape to represent.
@param nPrio
Externally-determined shape priority (used e.g. for
paint ordering). This number _must be_ unique!
@param rServiceName
Service name to use, when creating the actual viewer
component
@param pPropCopyTable
Table of plain ASCII property names, to copy from
xShape to applet.
@param nNumPropEntries
Number of property table entries (in pPropCopyTable)
*/
AppletShape( const css::uno::Reference< css::drawing::XShape >& xShape,
double nPrio,
OUString aServiceName,
const char** pPropCopyTable,
std::size_t nNumPropEntries,
const SlideShowContext& rContext ); // throw ShapeLoadFailedException;
private:
// View layer methods
virtual void addViewLayer( const ViewLayerSharedPtr& rNewLayer,
bool bRedrawLayer ) override;
virtual bool removeViewLayer( const ViewLayerSharedPtr& rNewLayer ) override;
virtual void clearAllViewLayers() override;
// ExternalShapeBase methods
virtual bool implRender( const ::basegfx::B2DRange& rCurrBounds ) const override;
virtual void implViewChanged( const UnoViewSharedPtr& rView ) override;
virtual void implViewsChanged() override;
virtual bool implStartIntrinsicAnimation() override;
virtual bool implEndIntrinsicAnimation() override;
virtual void implPauseIntrinsicAnimation() override;
virtual bool implIsIntrinsicAnimationPlaying() const override;
virtual void implSetIntrinsicAnimationTime(double) override;
const OUString maServiceName;
const char** mpPropCopyTable;
const std::size_t mnNumPropEntries;
/// the list of active view shapes (one for each registered view layer)
typedef ::std::vector< ViewAppletShapeSharedPtr > ViewAppletShapeVector;
ViewAppletShapeVector maViewAppletShapes;
bool mbIsPlaying;
};
}
AppletShape::AppletShape( const uno::Reference< drawing::XShape >& xShape,
double nPrio,
OUString aServiceName,
const char** pPropCopyTable,
std::size_t nNumPropEntries,
const SlideShowContext& rContext ) :
ExternalShapeBase( xShape, nPrio, rContext ),
maServiceName(std::move( aServiceName )),
mpPropCopyTable( pPropCopyTable ),
mnNumPropEntries( nNumPropEntries ),
maViewAppletShapes(),
mbIsPlaying(false)
{
}
void AppletShape::implViewChanged( const UnoViewSharedPtr& rView )
{
const ::basegfx::B2DRectangle aBounds = getBounds();
// determine ViewAppletShape that needs update
for( const auto& pViewAppletShape : maViewAppletShapes )
{
if( pViewAppletShape->getViewLayer()->isOnView( rView ) )
pViewAppletShape->resize( aBounds );
}
}
void AppletShape::implViewsChanged()
{
// resize all ViewShapes
const ::basegfx::B2DRectangle aBounds = getBounds();
for( const auto& pViewAppletShape : maViewAppletShapes )
pViewAppletShape->resize( aBounds );
}
void AppletShape::addViewLayer( const ViewLayerSharedPtr& rNewLayer,
bool bRedrawLayer )
{
try
{
maViewAppletShapes.push_back(
std::make_shared<ViewAppletShape>( rNewLayer,
getXShape(),
maServiceName,
mpPropCopyTable,
mnNumPropEntries,
mxComponentContext ));
// push new size to view shape
maViewAppletShapes.back()->resize( getBounds() );
// render the Shape on the newly added ViewLayer
if( bRedrawLayer )
maViewAppletShapes.back()->render( getBounds() );
}
catch(uno::Exception&)
{
// ignore failed shapes - slideshow should run with
// the remaining content
}
}
bool AppletShape::removeViewLayer( const ViewLayerSharedPtr& rLayer )
{
const ViewAppletShapeVector::iterator aEnd( maViewAppletShapes.end() );
OSL_ENSURE( ::std::count_if(maViewAppletShapes.begin(),
aEnd,
[&rLayer]
( const ViewAppletShapeSharedPtr& pShape )
{ return rLayer == pShape->getViewLayer(); } ) < 2,
"AppletShape::removeViewLayer(): Duplicate ViewLayer entries!" );
// TODO : needed for the moment since ANDROID doesn't know size_t return from std::erase_if
#if defined ANDROID
ViewAppletShapeVector::iterator aIter;
if( (aIter=::std::remove_if( maViewAppletShapes.begin(),
aEnd,
[&rLayer]
( const ViewAppletShapeSharedPtr& pShape )
{ return rLayer == pShape->getViewLayer(); } ) ) == aEnd )
{
// view layer seemingly was not added, failed
return false;
}
// actually erase from container
maViewAppletShapes.erase( aIter, aEnd );
return true;
#else
size_t nb = std::erase_if(maViewAppletShapes,
[&rLayer]
( const ViewAppletShapeSharedPtr& pShape )
{ return rLayer == pShape->getViewLayer(); } );
// if nb == 0, it means view layer seemingly was not added, failed
return (nb != 0);
#endif
}
void AppletShape::clearAllViewLayers()
{
maViewAppletShapes.clear();
}
bool AppletShape::implRender( const ::basegfx::B2DRange& rCurrBounds ) const
{
// redraw all view shapes, by calling their update() method
if( o3tl::make_unsigned(::std::count_if( maViewAppletShapes.begin(),
maViewAppletShapes.end(),
[&rCurrBounds]
( const ViewAppletShapeSharedPtr& pShape )
{ return pShape->render( rCurrBounds ); } ))
!= maViewAppletShapes.size() )
{
// at least one of the ViewShape::update() calls did return
// false - update failed on at least one ViewLayer
return false;
}
return true;
}
bool AppletShape::implStartIntrinsicAnimation()
{
const ::basegfx::B2DRectangle aBounds = getBounds();
for( const auto& pViewAppletShape : maViewAppletShapes )
pViewAppletShape->startApplet( aBounds );
mbIsPlaying = true;
return true;
}
bool AppletShape::implEndIntrinsicAnimation()
{
for( const auto& pViewAppletShape : maViewAppletShapes )
pViewAppletShape->endApplet();
mbIsPlaying = false;
return true;
}
void AppletShape::implPauseIntrinsicAnimation()
{
// TODO(F1): any way of temporarily disabling/deactivating
// applets?
}
bool AppletShape::implIsIntrinsicAnimationPlaying() const
{
return mbIsPlaying;
}
void AppletShape::implSetIntrinsicAnimationTime(double)
{
// No way of doing this, or?
}
std::shared_ptr<Shape> createAppletShape(
const uno::Reference< drawing::XShape >& xShape,
double nPrio,
const OUString& rServiceName,
const char** pPropCopyTable,
std::size_t nNumPropEntries,
const SlideShowContext& rContext )
{
return std::make_shared<AppletShape>(xShape,
nPrio,
rServiceName,
pPropCopyTable,
nNumPropEntries,
rContext);
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */