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." );
63 player->PushPacket( packet );
67 static void EmitPlaybackFinishedSignal( void* user_data )
69 TizenVideoPlayer* player = static_cast< TizenVideoPlayer* >( user_data );
70 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal.0" );
74 DALI_LOG_ERROR( "Decoded callback got Null pointer as user_data." );
78 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal." );
80 if( !player->mFinishedSignal.Empty() )
82 DALI_LOG_ERROR( "EmitPlaybackFinishedSignal.3" );
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()
207 : mPlayerState( PLAYER_STATE_NONE ),
209 mBackgroundColor( Dali::Stage::DEFAULT_BACKGROUND_COLOR ),
210 mTargetType( NativeImage )
214 TizenVideoPlayer::~TizenVideoPlayer()
218 void TizenVideoPlayer::GetPlayerState( player_state_e* state )
220 if( player_get_state( mPlayer, state ) != PLAYER_ERROR_NONE )
222 DALI_LOG_ERROR( "player_get_state error: Invalid parameter\n" );
223 *state = PLAYER_STATE_NONE;
227 void TizenVideoPlayer::SetUrl( const std::string& url )
233 GetPlayerState( &mPlayerState );
235 if( mPlayerState != PLAYER_STATE_NONE && mPlayerState != PLAYER_STATE_IDLE )
238 int error = player_unprepare( mPlayer );
239 LogPlayerError( error );
242 if( mPlayerState == PLAYER_STATE_IDLE )
244 int error = player_set_uri( mPlayer, mUrl.c_str() );
245 LogPlayerError( error );
247 error = player_prepare( mPlayer );
248 LogPlayerError( error );
253 std::string TizenVideoPlayer::GetUrl()
258 void TizenVideoPlayer::SetRenderingTarget( Any target )
261 if( mPlayerState != PLAYER_STATE_NONE )
263 GetPlayerState( &mPlayerState );
265 if( mPlayerState != PLAYER_STATE_IDLE )
268 error = player_unprepare( mPlayer );
269 LogPlayerError( error );
272 error = player_destroy( mPlayer );
273 LogPlayerError( error );
274 mPlayerState = PLAYER_STATE_NONE;
279 mNativeImageSourcePtr = NULL;
281 if( target.GetType() == typeid( Dali::NativeImageSourcePtr ) )
283 if( mTargetType == TizenVideoPlayer::WindowSurface )
285 Stage::GetCurrent().SetBackgroundColor( mBackgroundColor );
287 mTargetType = TizenVideoPlayer::NativeImage;
289 Dali::NativeImageSourcePtr nativeImageSourcePtr = AnyCast< Dali::NativeImageSourcePtr >( target );
291 InitializeTextureStreamMode( nativeImageSourcePtr );
293 else if( target.GetType() == typeid( Ecore_Wl_Window* ) )
295 mTargetType = TizenVideoPlayer::WindowSurface;
296 mBackgroundColor = Stage::GetCurrent().GetBackgroundColor();
297 Stage::GetCurrent().SetBackgroundColor( Color::TRANSPARENT );
299 Ecore_Wl_Window* nativeWindow = Dali::AnyCast< Ecore_Wl_Window* >( target );
300 InitializeUnderlayMode( nativeWindow );
304 DALI_LOG_ERROR( "Video rendering target is unknown" );
308 void TizenVideoPlayer::SetLooping( bool looping )
310 GetPlayerState( &mPlayerState );
312 if( mPlayerState != PLAYER_STATE_NONE )
314 int error = player_set_looping( mPlayer, looping );
315 LogPlayerError( error );
319 bool TizenVideoPlayer::IsLooping()
321 GetPlayerState( &mPlayerState );
323 bool looping = false;
324 if( mPlayerState != PLAYER_STATE_NONE )
326 int error = player_is_looping( mPlayer, &looping );
327 LogPlayerError( error );
333 void TizenVideoPlayer::Play()
335 GetPlayerState( &mPlayerState );
337 if( mPlayerState == PLAYER_STATE_READY || mPlayerState == PLAYER_STATE_PAUSED )
339 if( mNativeImageSourcePtr != NULL && mTimer )
344 int error = player_start( mPlayer );
345 LogPlayerError( error );
349 void TizenVideoPlayer::Pause()
351 GetPlayerState( &mPlayerState );
353 if( mPlayerState == PLAYER_STATE_PLAYING )
355 int error = player_pause( mPlayer );
356 LogPlayerError( error );
358 if( mNativeImageSourcePtr != NULL && mTimer )
366 void TizenVideoPlayer::Stop()
368 GetPlayerState( &mPlayerState );
370 if( mPlayerState == PLAYER_STATE_PLAYING || mPlayerState == PLAYER_STATE_PAUSED )
372 int error = player_stop( mPlayer );
373 LogPlayerError( error );
376 if( mNativeImageSourcePtr != NULL && mTimer )
383 void TizenVideoPlayer::SetMute( bool muted )
385 GetPlayerState( &mPlayerState );
387 if( mPlayerState == PLAYER_STATE_IDLE ||
388 mPlayerState == PLAYER_STATE_READY ||
389 mPlayerState == PLAYER_STATE_PLAYING ||
390 mPlayerState == PLAYER_STATE_PAUSED
393 int error = player_set_mute( mPlayer, muted );
394 LogPlayerError( error );
398 bool TizenVideoPlayer::IsMuted()
400 GetPlayerState( &mPlayerState );
403 if( mPlayerState == PLAYER_STATE_IDLE ||
404 mPlayerState == PLAYER_STATE_READY ||
405 mPlayerState == PLAYER_STATE_PLAYING ||
406 mPlayerState == PLAYER_STATE_PAUSED
410 int error = player_is_muted( mPlayer, &muted );
411 LogPlayerError( error );
417 void TizenVideoPlayer::SetVolume( float left, float right )
419 GetPlayerState( &mPlayerState );
421 int error = player_set_volume( mPlayer, left, right );
422 LogPlayerError( error );
425 void TizenVideoPlayer::GetVolume( float& left, float& right )
427 GetPlayerState( &mPlayerState );
429 int error = player_get_volume( mPlayer, &left, &right );
430 LogPlayerError( error );
433 void TizenVideoPlayer::SetPlayPosition( int millisecond )
437 GetPlayerState( &mPlayerState );
439 if( mPlayerState == PLAYER_STATE_READY ||
440 mPlayerState == PLAYER_STATE_PLAYING ||
441 mPlayerState == PLAYER_STATE_PAUSED
444 error = player_set_play_position( mPlayer, millisecond, true, NULL, NULL );
445 LogPlayerError( error );
449 int TizenVideoPlayer::GetPlayPosition()
454 GetPlayerState( &mPlayerState );
456 if( mPlayerState == PLAYER_STATE_IDLE ||
457 mPlayerState == PLAYER_STATE_READY ||
458 mPlayerState == PLAYER_STATE_PLAYING ||
459 mPlayerState == PLAYER_STATE_PAUSED
462 error = player_get_play_position( mPlayer, &millisecond );
463 LogPlayerError( error );
469 void TizenVideoPlayer::SetDisplayRotation( Dali::VideoPlayerPlugin::DisplayRotation rotation )
471 if( mNativeImageSourcePtr != NULL )
473 DALI_LOG_ERROR( "SetDisplayRotation is only for window rendering target.\n" );
478 if( mPlayerState != PLAYER_STATE_NONE )
480 error = player_set_display_rotation( mPlayer, static_cast< player_display_rotation_e >( rotation ) );
481 LogPlayerError( error );
485 Dali::VideoPlayerPlugin::DisplayRotation TizenVideoPlayer::GetDisplayRotation()
487 if( mNativeImageSourcePtr != NULL )
489 DALI_LOG_ERROR( "GetDisplayRotation is only for window rendering target.\n" );
490 return Dali::VideoPlayerPlugin::ROTATION_NONE;
494 player_display_rotation_e rotation = PLAYER_DISPLAY_ROTATION_NONE;
495 if( mPlayerState != PLAYER_STATE_NONE )
497 error = player_get_display_rotation( mPlayer, &rotation );
498 LogPlayerError( error );
500 return static_cast< Dali::VideoPlayerPlugin::DisplayRotation >( rotation );
503 Dali::VideoPlayerPlugin::VideoPlayerSignalType& TizenVideoPlayer::FinishedSignal()
505 return mFinishedSignal;
508 void TizenVideoPlayer::InitializeTextureStreamMode( Dali::NativeImageSourcePtr nativeImageSourcePtr )
512 mNativeImageSourcePtr = nativeImageSourcePtr;
514 if( mPlayerState == PLAYER_STATE_NONE )
516 error = player_create( &mPlayer );
517 LogPlayerError( error );
520 GetPlayerState( &mPlayerState );
522 if( mPlayerState == PLAYER_STATE_IDLE )
524 error = player_set_completed_cb( mPlayer, EmitPlaybackFinishedSignal, this );
525 LogPlayerError( error );
527 error = player_set_media_packet_video_frame_decoded_cb( mPlayer, MediaPacketVideoDecodedCb, this );
528 LogPlayerError( error );
530 error = player_set_sound_type( mPlayer, SOUND_TYPE_MEDIA );
531 LogPlayerError( error );
533 error = player_set_display_mode( mPlayer, PLAYER_DISPLAY_MODE_FULL_SCREEN );
534 LogPlayerError( error );
536 error = player_set_display( mPlayer, PLAYER_DISPLAY_TYPE_NONE, NULL );
537 LogPlayerError( error );
539 error = player_set_display_visible( mPlayer, true );
540 LogPlayerError( error );
542 mTimer = Dali::Timer::New( TIMER_INTERVAL );
543 mTimer.TickSignal().Connect( this, &TizenVideoPlayer::Update );
547 void TizenVideoPlayer::InitializeUnderlayMode( Ecore_Wl_Window* ecoreWlWindow )
550 if( mPlayerState == PLAYER_STATE_NONE )
552 error = player_create( &mPlayer );
553 LogPlayerError( error );
556 GetPlayerState( &mPlayerState );
558 if( mPlayerState == PLAYER_STATE_IDLE )
560 error = player_set_completed_cb( mPlayer, EmitPlaybackFinishedSignal, this );
561 LogPlayerError( error );
563 error = player_set_sound_type( mPlayer, SOUND_TYPE_MEDIA );
564 LogPlayerError( error );
566 error = player_set_display_mode( mPlayer, PLAYER_DISPLAY_MODE_FULL_SCREEN );
567 LogPlayerError( error );
570 ecore_wl_screen_size_get( &width, &height );
571 error = player_set_ecore_wl_display( mPlayer, PLAYER_DISPLAY_TYPE_OVERLAY, ecoreWlWindow, 0, 0, width, height );
572 LogPlayerError( error );
574 error = player_set_display_visible( mPlayer, true );
575 LogPlayerError( error );
579 bool TizenVideoPlayer::Update()
581 Dali::Mutex::ScopedLock lock( mPacketMutex );
585 if( mPacket != NULL )
587 error = media_packet_destroy( mPacket );
588 if( error != MEDIA_PACKET_ERROR_NONE )
590 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
595 if( !mPacketVector.Empty() )
597 mPacket = static_cast< media_packet_h >( mPacketVector[0] );
598 mPacketVector.Remove( mPacketVector.Begin() );
601 if( mPacket == NULL )
606 error = media_packet_get_tbm_surface( mPacket, &mTbmSurface );
607 if( error != MEDIA_PACKET_ERROR_NONE )
609 media_packet_destroy( mPacket );
611 DALI_LOG_ERROR( " error: %d\n", error );
615 Any source( mTbmSurface );
616 mNativeImageSourcePtr->SetSource( source );
617 Dali::Stage::GetCurrent().KeepRendering( 0.0f );
622 void TizenVideoPlayer::DestroyPackets()
625 if( mPacket != NULL )
627 error = media_packet_destroy( mPacket );
628 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
632 for(unsigned int i = 0; i < mPacketVector.Size(); ++i)
634 mPacket = static_cast< media_packet_h >( mPacketVector[i] );
635 error = media_packet_destroy( mPacket );
636 DALI_LOG_ERROR( "Media packet destroy error: %d\n", error );
639 mPacketVector.Clear();
642 void TizenVideoPlayer::PushPacket( media_packet_h packet )
644 Dali::Mutex::ScopedLock lock( mPacketMutex );
645 mPacketVector.PushBack( packet );
648 } // namespace Plugin