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 ),
210 mBackgroundColor( Dali::Stage::DEFAULT_BACKGROUND_COLOR ),
211 mTargetType( NativeImage )
215 TizenVideoPlayer::~TizenVideoPlayer()
219 void TizenVideoPlayer::GetPlayerState( player_state_e* state )
221 if( mPlayer != NULL && player_get_state( mPlayer, state ) != PLAYER_ERROR_NONE )
223 DALI_LOG_ERROR( "player_get_state error: Invalid parameter\n" );
224 *state = PLAYER_STATE_NONE;
228 void TizenVideoPlayer::SetUrl( const std::string& url )
234 GetPlayerState( &mPlayerState );
236 if( mPlayerState != PLAYER_STATE_NONE && mPlayerState != PLAYER_STATE_IDLE )
239 int error = player_unprepare( mPlayer );
240 LogPlayerError( error );
243 if( mPlayerState == PLAYER_STATE_IDLE )
245 int error = player_set_uri( mPlayer, mUrl.c_str() );
246 LogPlayerError( error );
248 error = player_prepare( mPlayer );
249 LogPlayerError( error );
254 std::string TizenVideoPlayer::GetUrl()
259 void TizenVideoPlayer::SetRenderingTarget( Any target )
262 if( mPlayerState != PLAYER_STATE_NONE )
264 GetPlayerState( &mPlayerState );
266 if( mPlayerState != PLAYER_STATE_IDLE )
269 error = player_unprepare( mPlayer );
270 LogPlayerError( error );
273 error = player_destroy( mPlayer );
274 LogPlayerError( error );
275 mPlayerState = PLAYER_STATE_NONE;
280 mNativeImageSourcePtr = NULL;
282 if( target.GetType() == typeid( Dali::NativeImageSourcePtr ) )
284 if( mTargetType == TizenVideoPlayer::WindowSurface )
286 Stage::GetCurrent().SetBackgroundColor( mBackgroundColor );
288 mTargetType = TizenVideoPlayer::NativeImage;
290 Dali::NativeImageSourcePtr nativeImageSourcePtr = AnyCast< Dali::NativeImageSourcePtr >( target );
292 InitializeTextureStreamMode( nativeImageSourcePtr );
294 else if( target.GetType() == typeid( Ecore_Wl_Window* ) )
296 mTargetType = TizenVideoPlayer::WindowSurface;
297 mBackgroundColor = Stage::GetCurrent().GetBackgroundColor();
298 Stage::GetCurrent().SetBackgroundColor( Color::TRANSPARENT );
300 Ecore_Wl_Window* nativeWindow = Dali::AnyCast< Ecore_Wl_Window* >( target );
301 InitializeUnderlayMode( nativeWindow );
305 DALI_LOG_ERROR( "Video rendering target is unknown\n" );
309 void TizenVideoPlayer::SetLooping( bool looping )
311 GetPlayerState( &mPlayerState );
313 if( mPlayerState != PLAYER_STATE_NONE )
315 int error = player_set_looping( mPlayer, looping );
316 LogPlayerError( error );
320 bool TizenVideoPlayer::IsLooping()
322 GetPlayerState( &mPlayerState );
324 bool looping = false;
325 if( mPlayerState != PLAYER_STATE_NONE )
327 int error = player_is_looping( mPlayer, &looping );
328 LogPlayerError( error );
334 void TizenVideoPlayer::Play()
336 GetPlayerState( &mPlayerState );
338 if( mPlayerState == PLAYER_STATE_READY || mPlayerState == PLAYER_STATE_PAUSED )
340 if( mNativeImageSourcePtr != NULL && mTimer )
345 int error = player_start( mPlayer );
346 LogPlayerError( error );
350 void TizenVideoPlayer::Pause()
352 GetPlayerState( &mPlayerState );
354 if( mPlayerState == PLAYER_STATE_PLAYING )
356 int error = player_pause( mPlayer );
357 LogPlayerError( error );
359 if( mNativeImageSourcePtr != NULL && mTimer )
367 void TizenVideoPlayer::Stop()
369 GetPlayerState( &mPlayerState );
371 if( mPlayerState == PLAYER_STATE_PLAYING || mPlayerState == PLAYER_STATE_PAUSED )
373 int error = player_stop( mPlayer );
374 LogPlayerError( error );
377 if( mNativeImageSourcePtr != NULL && mTimer )
384 void TizenVideoPlayer::SetMute( bool muted )
386 GetPlayerState( &mPlayerState );
388 if( mPlayerState == PLAYER_STATE_IDLE ||
389 mPlayerState == PLAYER_STATE_READY ||
390 mPlayerState == PLAYER_STATE_PLAYING ||
391 mPlayerState == PLAYER_STATE_PAUSED
394 int error = player_set_mute( mPlayer, muted );
395 LogPlayerError( error );
399 bool TizenVideoPlayer::IsMuted()
401 GetPlayerState( &mPlayerState );
404 if( mPlayerState == PLAYER_STATE_IDLE ||
405 mPlayerState == PLAYER_STATE_READY ||
406 mPlayerState == PLAYER_STATE_PLAYING ||
407 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