-static void PreparseEnqueueItemSub( playlist_t *p_playlist,
- playlist_item_t *p_item )
-{
- int i;
- if( p_item->i_children == -1 )
- {
- vlc_gc_incref( p_item );
- INSERT_ELEM( p_playlist->p_preparse->pp_waiting,
- p_playlist->p_preparse->i_waiting,
- p_playlist->p_preparse->i_waiting,
- p_item->p_input );
- }
- else
- {
- for( i = 0; i < p_item->i_children; i++)
- {
- PreparseEnqueueItemSub( p_playlist, p_item->pp_children[i] );
- }
- }
-}
-
-/*****************************************************************************
- * Playback logic
- *****************************************************************************/
-
-/**
- * Synchronise the current index of the playlist
- * to match the index of the current item.
- *
- * \param p_playlist the playlist structure
- * \param p_cur the current playlist item
- * \return nothing
- */
-static void ResyncCurrentIndex( playlist_t *p_playlist, playlist_item_t *p_cur )
-{
- PL_DEBUG( "resyncing on %s", PLI_NAME( p_cur ) );
- /* Simply resync index */
- int i;
- p_playlist->i_current_index = -1;
- for( i = 0 ; i< p_playlist->current.i_size; i++ )
- {
- if( ARRAY_VAL( p_playlist->current, i ) == p_cur )
- {
- p_playlist->i_current_index = i;
- break;
- }
- }
- PL_DEBUG( "%s is at %i", PLI_NAME( p_cur ), p_playlist->i_current_index );
-}
-
-void ResetCurrentlyPlaying( playlist_t *p_playlist, vlc_bool_t b_random,
- playlist_item_t *p_cur )
-{
- playlist_item_t *p_next = NULL;
- stats_TimerStart( p_playlist, "Items array build",
- STATS_TIMER_PLAYLIST_BUILD );
- PL_DEBUG( "rebuilding array of current - root %s",
- PLI_NAME( p_playlist->status.p_node ) );
- ARRAY_RESET( p_playlist->current );
- p_playlist->i_current_index = -1;
- while( 1 )
- {
- /** FIXME: this is *slow* */
- p_next = playlist_GetNextLeaf( p_playlist,
- p_playlist->status.p_node,
- p_next, VLC_TRUE, VLC_FALSE );
- if( p_next )
- {
- if( p_next == p_cur )
- p_playlist->i_current_index = p_playlist->current.i_size;
- ARRAY_APPEND( p_playlist->current, p_next);
- }
- else break;
- }
- PL_DEBUG("rebuild done - %i items, index %i", p_playlist->current.i_size,
- p_playlist->i_current_index);
- if( b_random )
- {
- /* Shuffle the array */
- srand( (unsigned int)mdate() );
- int j;
- for( j = p_playlist->current.i_size - 1; j > 0; j-- )
- {
- int i = rand() % (j+1); /* between 0 and j */
- playlist_item_t *p_tmp;
- /* swap the two items */
- p_tmp = ARRAY_VAL(p_playlist->current, i);
- ARRAY_VAL(p_playlist->current,i) = ARRAY_VAL(p_playlist->current,j);
- ARRAY_VAL(p_playlist->current,j) = p_tmp;
- }
- }
- p_playlist->b_reset_currently_playing = VLC_FALSE;
- stats_TimerStop( p_playlist, STATS_TIMER_PLAYLIST_BUILD );
-}
-
-/**
- * Compute the next playlist item depending on
- * the playlist course mode (forward, backward, random, view,...).
- *
- * \param p_playlist the playlist object
- * \return nothing
- */
-playlist_item_t * playlist_NextItem( playlist_t *p_playlist )
-{
- playlist_item_t *p_new = NULL;
- int i_skip = 0, i;
-
- vlc_bool_t b_loop = var_GetBool( p_playlist, "loop" );
- vlc_bool_t b_random = var_GetBool( p_playlist, "random" );
- vlc_bool_t b_repeat = var_GetBool( p_playlist, "repeat" );
- vlc_bool_t b_playstop = var_GetBool( p_playlist, "play-and-stop" );
-
- /* Handle quickly a few special cases */
- /* No items to play */
- if( p_playlist->items.i_size == 0 )
- {
- msg_Info( p_playlist, "playlist is empty" );
- return NULL;
- }
-
- /* Repeat and play/stop */
- if( !p_playlist->request.b_request && b_repeat == VLC_TRUE &&
- p_playlist->status.p_item )
- {
- msg_Dbg( p_playlist,"repeating item" );
- return p_playlist->status.p_item;
- }
- if( !p_playlist->request.b_request && b_playstop == VLC_TRUE )
- {
- msg_Dbg( p_playlist,"stopping (play and stop)" );
- return NULL;
- }
-
- if( !p_playlist->request.b_request && p_playlist->status.p_item )
- {
- playlist_item_t *p_parent = p_playlist->status.p_item;
- while( p_parent )
- {
- if( p_parent->i_flags & PLAYLIST_SKIP_FLAG )
- {
- msg_Dbg( p_playlist, "blocking item, stopping") ;
- return NULL;
- }
- p_parent = p_parent->p_parent;
- }
- }
-
- /* Start the real work */
- if( p_playlist->request.b_request )
- {
- p_new = p_playlist->request.p_item;
- i_skip = p_playlist->request.i_skip;
- PL_DEBUG( "processing request item %s node %s skip %i",
- PLI_NAME( p_playlist->request.p_item ),
- PLI_NAME( p_playlist->request.p_node ), i_skip );
-
- if( p_playlist->request.p_node &&
- p_playlist->request.p_node != p_playlist->status.p_node )
- {
- p_playlist->status.p_node = p_playlist->request.p_node;
- p_playlist->b_reset_currently_playing = VLC_TRUE;
- }
-
- /* If we are asked for a node, don't take it */
- if( i_skip == 0 && ( p_new == NULL || p_new->i_children != -1 ) )
- i_skip++;
-
- if( p_playlist->b_reset_currently_playing )
- /* A bit too bad to reset twice ... */
- ResetCurrentlyPlaying( p_playlist, b_random, p_new );
- else if( p_new )
- ResyncCurrentIndex( p_playlist, p_new );
- else
- p_playlist->i_current_index = -1;
-
- if( p_playlist->current.i_size && (i_skip > 0) )
- {
- if( p_playlist->i_current_index < -1 )
- p_playlist->i_current_index = -1;
- for( i = i_skip; i > 0 ; i-- )
- {
- p_playlist->i_current_index++;
- if( p_playlist->i_current_index >= p_playlist->current.i_size )
- {
- PL_DEBUG( "looping - restarting at beginning of node" );
- p_playlist->i_current_index = 0;
- }
- }
- p_new = ARRAY_VAL( p_playlist->current,
- p_playlist->i_current_index );
- }
- else if( p_playlist->current.i_size && (i_skip < 0) )
- {
- for( i = i_skip; i < 0 ; i++ )
- {
- p_playlist->i_current_index--;
- if( p_playlist->i_current_index <= -1 )
- {
- PL_DEBUG( "looping - restarting at end of node" );
- p_playlist->i_current_index = p_playlist->current.i_size-1;
- }
- }
- p_new = ARRAY_VAL( p_playlist->current,
- p_playlist->i_current_index );
- }
- /* Clear the request */
- p_playlist->request.b_request = VLC_FALSE;
- }
- /* "Automatic" item change ( next ) */
- else
- {
- PL_DEBUG( "changing item without a request (current %i/%i)",
- p_playlist->i_current_index, p_playlist->current.i_size );
- /* Cant go to next from current item */
- if( p_playlist->status.p_item &&
- p_playlist->status.p_item->i_flags & PLAYLIST_SKIP_FLAG )
- return NULL;
-
- if( p_playlist->b_reset_currently_playing )
- ResetCurrentlyPlaying( p_playlist, b_random,
- p_playlist->status.p_item );
-
- p_playlist->i_current_index++;
- if( p_playlist->i_current_index == p_playlist->current.i_size )
- {
- if( !b_loop || p_playlist->current.i_size == 0 ) return NULL;
- p_playlist->i_current_index = 0;
- }
- PL_DEBUG( "using item %i", p_playlist->i_current_index );
- if ( p_playlist->current.i_size == 0 ) return NULL;
-
- p_new = ARRAY_VAL( p_playlist->current, p_playlist->i_current_index );
- /* The new item can't be autoselected */
- if( p_new != NULL && p_new->i_flags & PLAYLIST_SKIP_FLAG )
- return NULL;
- }
- return p_new;
-}
-
-/**
- * Start the input for an item
- *
- * \param p_playlist the playlist objetc
- * \param p_item the item to play
- * \return nothing
- */
-int playlist_PlayItem( playlist_t *p_playlist, playlist_item_t *p_item )