blob: 2d4c572314e88b6adb9bc1e39a67ae78a867d818 [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_slideshow.hxx"
// must be first
#include <canvas/debug.hxx>
#include <canvas/verbosetrace.hxx>
#include "eventqueue.hxx"
#include "animationaudionode.hxx"
#include "delayevent.hxx"
#include "tools.hxx"
#include "nodetools.hxx"
#include "boost/bind.hpp"
using namespace com::sun::star;
namespace slideshow {
namespace internal {
AnimationAudioNode::AnimationAudioNode(
const uno::Reference< animations::XAnimationNode >& xNode,
const BaseContainerNodeSharedPtr& rParent,
const NodeContext& rContext )
: BaseNode( xNode, rParent, rContext ),
mxAudioNode( xNode, uno::UNO_QUERY_THROW ),
maSoundURL(),
mpPlayer()
{
mxAudioNode->getSource() >>= maSoundURL;
OSL_ENSURE( maSoundURL.getLength(),
"could not extract sound source URL/empty URL string" );
ENSURE_OR_THROW( getContext().mxComponentContext.is(),
"Invalid component context" );
}
void AnimationAudioNode::dispose()
{
resetPlayer();
mxAudioNode.clear();
BaseNode::dispose();
}
void AnimationAudioNode::activate_st()
{
createPlayer();
AnimationEventHandlerSharedPtr aHandler(
boost::dynamic_pointer_cast<AnimationEventHandler>( getSelf() ) );
OSL_ENSURE( aHandler,
"could not cast self to AnimationEventHandler?" );
getContext().mrEventMultiplexer.addCommandStopAudioHandler( aHandler );
if (mpPlayer && mpPlayer->startPlayback())
{
// TODO(F2): Handle end time attribute, too
if( getXAnimationNode()->getDuration().hasValue() )
{
scheduleDeactivationEvent();
}
else
{
// no node duration. Take inherent media time, then
scheduleDeactivationEvent(
makeDelay( boost::bind( &AnimationNode::deactivate, getSelf() ),
mpPlayer->getDuration(),
"AnimationAudioNode::deactivate with delay") );
}
}
else
{
// deactivate ASAP:
scheduleDeactivationEvent(
makeEvent( boost::bind( &AnimationNode::deactivate, getSelf() ),
"AnimationAudioNode::deactivate without delay") );
}
}
// TODO(F2): generate deactivation event, when sound
// is over
void AnimationAudioNode::deactivate_st( NodeState /*eDestState*/ )
{
AnimationEventHandlerSharedPtr aHandler(
boost::dynamic_pointer_cast<AnimationEventHandler>( getSelf() ) );
OSL_ENSURE( aHandler,
"could not cas self to AnimationEventHandler?" );
getContext().mrEventMultiplexer.removeCommandStopAudioHandler( aHandler );
// force-end sound
if (mpPlayer)
{
mpPlayer->stopPlayback();
resetPlayer();
}
// notify _after_ state change:
getContext().mrEventQueue.addEvent(
makeEvent( boost::bind( &EventMultiplexer::notifyAudioStopped,
boost::ref(getContext().mrEventMultiplexer),
getSelf() ),
"AnimationAudioNode::notifyAudioStopped") );
}
bool AnimationAudioNode::hasPendingAnimation() const
{
// force slide to use the animation framework
// (otherwise, a single sound on the slide would
// not be played).
return true;
}
void AnimationAudioNode::createPlayer() const
{
if (mpPlayer)
return;
try
{
mpPlayer = SoundPlayer::create( getContext().mrEventMultiplexer,
maSoundURL,
getContext().mxComponentContext );
}
catch( lang::NoSupportException& )
{
// catch possible exceptions from SoundPlayer,
// since being not able to playback the sound
// is not a hard error here (remainder of the
// animations should still work).
}
}
void AnimationAudioNode::resetPlayer() const
{
if (mpPlayer)
{
mpPlayer->stopPlayback();
mpPlayer->dispose();
mpPlayer.reset();
}
}
bool AnimationAudioNode::handleAnimationEvent(
const AnimationNodeSharedPtr& /*rNode*/ )
{
// TODO(F2): for now we support only STOPAUDIO events.
deactivate();
return true;
}
} // namespace internal
} // namespace presentation