2 * Copyright (c) 2016 Samsung Electronics Co., Ltd.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
19 #include <tizen-video-player.h>
22 #include <dali/public-api/common/stage.h>
23 #include <dali/devel-api/threading/mutex.h>
24 #include <dali/integration-api/debug.h>
28 // The plugin factories
29 extern "C" DALI_EXPORT_API Dali::VideoPlayerPlugin* CreateVideoPlayerPlugin( void )
31 return new Dali::Plugin::TizenVideoPlayer;
34 extern "C" DALI_EXPORT_API void DestroyVideoPlayerPlugin( Dali::VideoPlayerPlugin* plugin )
51 const int TIMER_INTERVAL( 20 );
53 static void MediaPacketVideoDecodedCb( media_packet_h packet, void* user_data )
55 TizenVideoPlayer* player = static_cast< TizenVideoPlayer* >( user_data );
59 DALI_LOG_ERROR( "Decoded callback got Null pointer as user_data.\n" );
63 player->PushPacket( packet );
67 static void EmitPlaybackFinishedSignal( void* user_data )
69 TizenVideoPlayer* player = static_cast< TizenVideoPlayer* >( user_data );
70 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal.\n" );
74 DALI_LOG_ERROR( "Decoded callback got Null pointer as user_data.\n" );
78 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal.\n" );
80 if( !player->mFinishedSignal.Empty() )
82 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal.3\n" );
83 player->mFinishedSignal.Emit();
89 void LogPlayerError( int error )
91 if( error != PLAYER_ERROR_NONE )
95 case PLAYER_ERROR_OUT_OF_MEMORY:
97 DALI_LOG_ERROR( "Player error: Out of memory\n" );
100 case PLAYER_ERROR_INVALID_PARAMETER:
102 DALI_LOG_ERROR( "Player error: Invalid parameter\n" );
105 case PLAYER_ERROR_NO_SUCH_FILE:
107 DALI_LOG_ERROR( "Player error: No such file\n" );
110 case PLAYER_ERROR_INVALID_OPERATION:
112 DALI_LOG_ERROR( "Player error: Invalid operation\n" );
115 case PLAYER_ERROR_FILE_NO_SPACE_ON_DEVICE:
117 DALI_LOG_ERROR( "Player error: No space on device\n" );
120 case PLAYER_ERROR_FEATURE_NOT_SUPPORTED_ON_DEVICE:
122 DALI_LOG_ERROR( "Player error: Not supported feature on device\n" );
125 case PLAYER_ERROR_SEEK_FAILED:
127 DALI_LOG_ERROR( "Player error: Seek failed\n" );
130 case PLAYER_ERROR_INVALID_STATE:
132 DALI_LOG_ERROR( "Player error: Invalid state\n" );
135 case PLAYER_ERROR_NOT_SUPPORTED_FILE:
137 DALI_LOG_ERROR( "Player error: Not supported file\n" );
140 case PLAYER_ERROR_INVALID_URI:
142 DALI_LOG_ERROR( "Player error: Invalid uri\n" );
145 case PLAYER_ERROR_SOUND_POLICY:
147 DALI_LOG_ERROR( "Player error: Sound policy\n" );
150 case PLAYER_ERROR_CONNECTION_FAILED:
152 DALI_LOG_ERROR( "Player error: Connection failed\n" );
155 case PLAYER_ERROR_VIDEO_CAPTURE_FAILED:
157 DALI_LOG_ERROR( "Player error: Video capture failed\n" );
160 case PLAYER_ERROR_DRM_EXPIRED:
162 DALI_LOG_ERROR( "Player error: DRM expired\n" );
165 case PLAYER_ERROR_DRM_NO_LICENSE:
167 DALI_LOG_ERROR( "Player error: No license\n" );
170 case PLAYER_ERROR_DRM_FUTURE_USE:
172 DALI_LOG_ERROR( "Player error: License for future use\n" );
175 case PLAYER_ERROR_DRM_NOT_PERMITTED:
177 DALI_LOG_ERROR( "Player error: Format not permitted\n" );
180 case PLAYER_ERROR_RESOURCE_LIMIT:
182 DALI_LOG_ERROR( "Player error: Resource limit\n" );
185 case PLAYER_ERROR_PERMISSION_DENIED:
187 DALI_LOG_ERROR( "Player error: Permission denied\n" );
190 case PLAYER_ERROR_SERVICE_DISCONNECTED:
192 DALI_LOG_ERROR( "Player error: Service disconnected\n" );
195 case PLAYER_ERROR_BUFFER_SPACE:
197 DALI_LOG_ERROR( "Player error: Buffer space\n" );
204 } // unnamed namespace
206 TizenVideoPlayer::TizenVideoPlayer()
208 mPlayerState( PLAYER_STATE_NONE ),
211 mBackgroundColor( Dali::Stage::DEFAULT_BACKGROUND_COLOR ),
212 mTargetType( NativeImage )
216 TizenVideoPlayer::~TizenVideoPlayer()
220 void TizenVideoPlayer::GetPlayerState( player_state_e* state )
222 if( mPlayer != NULL && player_get_state( mPlayer, state ) != PLAYER_ERROR_NONE )
224 DALI_LOG_ERROR( "player_get_state error: Invalid parameter\n" );
225 *state = PLAYER_STATE_NONE;
229 void TizenVideoPlayer::SetUrl( const std::string& url )
235 GetPlayerState( &mPlayerState );
237 if( mPlayerState != PLAYER_STATE_NONE && mPlayerState != PLAYER_STATE_IDLE )
240 int error = player_unprepare( mPlayer );
241 LogPlayerError( error );
244 if( mPlayerState == PLAYER_STATE_IDLE )
246 int error = player_set_uri( mPlayer, mUrl.c_str() );
247 LogPlayerError( error );
249 error = player_prepare( mPlayer );
250 LogPlayerError( error );
255 std::string TizenVideoPlayer::GetUrl()
260 void TizenVideoPlayer::SetRenderingTarget( Any target )
263 if( mPlayerState != PLAYER_STATE_NONE )
265 GetPlayerState( &mPlayerState );
267 if( mPlayerState != PLAYER_STATE_IDLE )
270 error = player_unprepare( mPlayer );
271 LogPlayerError( error );
274 error = player_destroy( mPlayer );
275 LogPlayerError( error );
276 mPlayerState = PLAYER_STATE_NONE;
281 mNativeImageSourcePtr = NULL;
283 if( target.GetType() == typeid( Dali::NativeImageSourcePtr ) )
285 if( mTargetType == TizenVideoPlayer::WindowSurface )
287 Stage::GetCurrent().SetBackgroundColor( mBackgroundColor );
289 mTargetType = TizenVideoPlayer::NativeImage;
291 Dali::NativeImageSourcePtr nativeImageSourcePtr = AnyCast< Dali::NativeImageSourcePtr >( target );
293 InitializeTextureStreamMode( nativeImageSourcePtr );
295 else if( target.GetType() == typeid( Ecore_Wl_Window* ) )
297 mTargetType = TizenVideoPlayer::WindowSurface;
298 mBackgroundColor = Stage::GetCurrent().GetBackgroundColor();
299 Stage::GetCurrent().SetBackgroundColor( Color::TRANSPARENT );
301 Ecore_Wl_Window* nativeWindow = Dali::AnyCast< Ecore_Wl_Window* >( target );
302 InitializeUnderlayMode( nativeWindow );
306 DALI_LOG_ERROR( "Video rendering target is unknown\n" );
310 void TizenVideoPlayer::SetLooping( bool looping )
312 GetPlayerState( &mPlayerState );
314 if( mPlayerState != PLAYER_STATE_NONE )
316 int error = player_set_looping( mPlayer, looping );
317 LogPlayerError( error );
321 bool TizenVideoPlayer::IsLooping()
323 GetPlayerState( &mPlayerState );
325 bool looping = false;
326 if( mPlayerState != PLAYER_STATE_NONE )
328 int error = player_is_looping( mPlayer, &looping );
329 LogPlayerError( error );
335 void TizenVideoPlayer::Play()
337 GetPlayerState( &mPlayerState );
339 if( mPlayerState == PLAYER_STATE_READY || mPlayerState == PLAYER_STATE_PAUSED )
341 if( mNativeImageSourcePtr != NULL && mTimer )
346 int error = player_start( mPlayer );
347 LogPlayerError( error );
351 void TizenVideoPlayer::Pause()
353 GetPlayerState( &mPlayerState );
355 if( mPlayerState == PLAYER_STATE_PLAYING )
357 int error = player_pause( mPlayer );
358 LogPlayerError( error );
360 if( mNativeImageSourcePtr != NULL && mTimer )
368 void TizenVideoPlayer::Stop()
370 GetPlayerState( &mPlayerState );
372 if( mPlayerState == PLAYER_STATE_PLAYING || mPlayerState == PLAYER_STATE_PAUSED )
374 int error = player_stop( mPlayer );
375 LogPlayerError( error );
378 if( mNativeImageSourcePtr != NULL && mTimer )
385 void TizenVideoPlayer::SetMute( bool muted )
387 GetPlayerState( &mPlayerState );
389 if( mPlayerState == PLAYER_STATE_IDLE ||
390 mPlayerState == PLAYER_STATE_READY ||
391 mPlayerState == PLAYER_STATE_PLAYING ||
392 mPlayerState == PLAYER_STATE_PAUSED
395 int error = player_set_mute( mPlayer, muted );
396 LogPlayerError( error );
400 bool TizenVideoPlayer::IsMuted()
402 GetPlayerState( &mPlayerState );
405 if( mPlayerState == PLAYER_STATE_IDLE ||
406 mPlayerState == PLAYER_STATE_READY ||
407 mPlayerState == PLAYER_STATE_PLAYING ||
408 mPlayerState == PLAYER_STATE_PAUSED
411 int error = player_is_muted( mPlayer, &muted );
412 LogPlayerError( error );
418 void TizenVideoPlayer::SetVolume( float left, float right )
420 GetPlayerState( &mPlayerState );
422 int error = player_set_volume( mPlayer, left, right );
423 LogPlayerError( error );
426 void TizenVideoPlayer::GetVolume( float& left, float& right )
428 GetPlayerState( &mPlayerState );
430 int error = player_get_volume( mPlayer, &left, &right );
431 LogPlayerError( error );
434 void TizenVideoPlayer::SetPlayPosition( int millisecond )
438 GetPlayerState( &mPlayerState );
440 if( mPlayerState == PLAYER_STATE_READY ||
441 mPlayerState == PLAYER_STATE_PLAYING ||
442 mPlayerState == PLAYER_STATE_PAUSED
445 error = player_set_play_position( mPlayer, millisecond, true, NULL, NULL );
446 LogPlayerError( error );
450 int TizenVideoPlayer::GetPlayPosition()
455 GetPlayerState( &mPlayerState );
457 if( mPlayerState == PLAYER_STATE_IDLE ||
458 mPlayerState == PLAYER_STATE_READY ||
459 mPlayerState == PLAYER_STATE_PLAYING ||
460 mPlayerState == PLAYER_STATE_PAUSED
463 error = player_get_play_position( mPlayer, &millisecond );
464 LogPlayerError( error );
470 void TizenVideoPlayer::SetDisplayRotation( Dali::VideoPlayerPlugin::DisplayRotation rotation )
472 if( mNativeImageSourcePtr != NULL )
474 DALI_LOG_ERROR( "SetDisplayRotation is only for window rendering target.\n" );
479 if( mPlayerState != PLAYER_STATE_NONE )
481 error = player_set_display_rotation( mPlayer, static_cast< player_display_rotation_e >( rotation ) );
482 LogPlayerError( error );
486 Dali::VideoPlayerPlugin::DisplayRotation TizenVideoPlayer::GetDisplayRotation()
488 if( mNativeImageSourcePtr != NULL )
490 DALI_LOG_ERROR( "GetDisplayRotation is only for window rendering target.\n" );
491 return Dali::VideoPlayerPlugin::ROTATION_NONE;
495 player_display_rotation_e rotation = PLAYER_DISPLAY_ROTATION_NONE;
496 if( mPlayerState != PLAYER_STATE_NONE )
498 error = player_get_display_rotation( mPlayer, &rotation );
499 LogPlayerError( error );
501 return static_cast< Dali::VideoPlayerPlugin::DisplayRotation >( rotation );
504 Dali::VideoPlayerPlugin::VideoPlayerSignalType& TizenVideoPlayer::FinishedSignal()
506 return mFinishedSignal;
509 void TizenVideoPlayer::InitializeTextureStreamMode( Dali::NativeImageSourcePtr nativeImageSourcePtr )
513 mNativeImageSourcePtr = nativeImageSourcePtr;
515 if( mPlayerState == PLAYER_STATE_NONE )
517 error = player_create( &mPlayer );
518 LogPlayerError( error );
521 GetPlayerState( &mPlayerState );
523 if( mPlayerState == PLAYER_STATE_IDLE )
525 error = player_set_completed_cb( mPlayer, EmitPlaybackFinishedSignal, this );
526 LogPlayerError( error );
528 error = player_set_media_packet_video_frame_decoded_cb( mPlayer, MediaPacketVideoDecodedCb, this );
529 LogPlayerError( error );
531 error = player_set_sound_type( mPlayer, SOUND_TYPE_MEDIA );
532 LogPlayerError( error );
534 error = player_set_display_mode( mPlayer, PLAYER_DISPLAY_MODE_FULL_SCREEN );
535 LogPlayerError( error );
537 error = player_set_display( mPlayer, PLAYER_DISPLAY_TYPE_NONE, NULL );
538 LogPlayerError( error );
540 error = player_set_display_visible( mPlayer, true );
541 LogPlayerError( error );
543 mTimer = Dali::Timer::New( TIMER_INTERVAL );
544 mTimer.TickSignal().Connect( this, &TizenVideoPlayer::Update );
548 void TizenVideoPlayer::InitializeUnderlayMode( Ecore_Wl_Window* ecoreWlWindow )
551 if( mPlayerState == PLAYER_STATE_NONE )
553 error = player_create( &mPlayer );
554 LogPlayerError( error );
557 GetPlayerState( &mPlayerState );
559 if( mPlayerState == PLAYER_STATE_IDLE )
561 error = player_set_completed_cb( mPlayer, EmitPlaybackFinishedSignal, this );
562 LogPlayerError( error );
564 error = player_set_sound_type( mPlayer, SOUND_TYPE_MEDIA );
565 LogPlayerError( error );
567 error = player_set_display_mode( mPlayer, PLAYER_DISPLAY_MODE_FULL_SCREEN );
568 LogPlayerError( error );
571 ecore_wl_screen_size_get( &width, &height );
572 error = player_set_ecore_wl_display( mPlayer, PLAYER_DISPLAY_TYPE_OVERLAY, ecoreWlWindow, 0, 0, width, height );
573 LogPlayerError( error );
575 error = player_set_display_visible( mPlayer, true );
576 LogPlayerError( error );
580 bool TizenVideoPlayer::Update()
582 Dali::Mutex::ScopedLock lock( mPacketMutex );
586 if( mPacket != NULL )
588 error = media_packet_destroy( mPacket );
589 if( error != MEDIA_PACKET_ERROR_NONE )
591 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
596 if( !mPacketVector.Empty() )
598 mPacket = static_cast< media_packet_h >( mPacketVector[0] );
599 mPacketVector.Remove( mPacketVector.Begin() );
602 if( mPacket == NULL )
607 error = media_packet_get_tbm_surface( mPacket, &mTbmSurface );
608 if( error != MEDIA_PACKET_ERROR_NONE )
610 media_packet_destroy( mPacket );
612 DALI_LOG_ERROR( " error: %d\n", error );
616 Any source( mTbmSurface );
617 mNativeImageSourcePtr->SetSource( source );
618 Dali::Stage::GetCurrent().KeepRendering( 0.0f );
623 void TizenVideoPlayer::DestroyPackets()
626 if( mPacket != NULL )
628 error = media_packet_destroy( mPacket );
629 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
633 for(unsigned int i = 0; i < mPacketVector.Size(); ++i)
635 mPacket = static_cast< media_packet_h >( mPacketVector[i] );
636 error = media_packet_destroy( mPacket );
637 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
640 mPacketVector.Clear();
643 void TizenVideoPlayer::PushPacket( media_packet_h packet )
645 Dali::Mutex::ScopedLock lock( mPacketMutex );
646 mPacketVector.PushBack( packet );
649 } // namespace Plugin