You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
906 lines
25 KiB
906 lines
25 KiB
/*****************************************************************************
|
|
* input.c
|
|
*****************************************************************************
|
|
* Copyright (C) 2007-2008 the VideoLAN team
|
|
*
|
|
* Authors: Antoine Cellerier <dionoea at videolan tod org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* Preamble
|
|
*****************************************************************************/
|
|
#ifndef _GNU_SOURCE
|
|
# define _GNU_SOURCE
|
|
#endif
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_es.h>
|
|
#include <vlc_meta.h>
|
|
#include <vlc_url.h>
|
|
#include <vlc_playlist.h>
|
|
#include <vlc_player.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "../vlc.h"
|
|
#include "input.h"
|
|
#include "../libs.h"
|
|
#include "../extension.h"
|
|
|
|
vout_thread_t *vlclua_get_vout_internal(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
return vlc_player_vout_Hold(player);
|
|
}
|
|
|
|
audio_output_t *vlclua_get_aout_internal(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
return vlc_player_aout_Hold(player);
|
|
}
|
|
|
|
static input_item_t* vlclua_input_item_get_internal( lua_State *L );
|
|
|
|
static int vlclua_input_item_info( lua_State *L )
|
|
{
|
|
input_item_t *p_item = vlclua_input_item_get_internal( L );
|
|
int i_cat = 0;
|
|
|
|
info_category_t *p_category;
|
|
vlc_list_foreach( p_category, &p_item->categories, node )
|
|
i_cat++;
|
|
|
|
lua_createtable( L, 0, i_cat );
|
|
|
|
vlc_list_foreach( p_category, &p_item->categories, node )
|
|
{
|
|
info_t *p_info;
|
|
|
|
if (info_category_IsHidden(p_category))
|
|
continue;
|
|
|
|
lua_pushstring( L, p_category->psz_name );
|
|
lua_newtable( L );
|
|
info_foreach(p_info, &p_category->infos)
|
|
{
|
|
lua_pushstring( L, p_info->psz_name );
|
|
lua_pushstring( L, p_info->psz_value );
|
|
lua_settable( L, -3 );
|
|
}
|
|
lua_settable( L, -3 );
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_is_playing( lua_State *L )
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
bool started = vlc_player_IsStarted(player);
|
|
vlc_player_Unlock(player);
|
|
lua_pushboolean(L, started);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_title_index(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
ssize_t idx = vlc_player_GetSelectedTitleIdx(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushinteger(L, idx);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_titles_count(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
struct vlc_player_title_list *titles = vlc_player_GetTitleList(player);
|
|
size_t count = titles ? vlc_player_title_list_GetCount(titles) : 0;
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushinteger(L, count);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_title_next(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectNextTitle(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_title_prev(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectPrevTitle(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_title_goto(lua_State *L)
|
|
{
|
|
int idx = luaL_checkinteger(L, 1);
|
|
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectTitleIdx(player, idx);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_get_chapter_index(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
ssize_t idx = vlc_player_GetSelectedChapterIdx(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushinteger(L, idx);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_chapters_count(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
const struct vlc_player_title *current_title =
|
|
vlc_player_GetSelectedTitle(player);
|
|
|
|
size_t count = current_title ? current_title->chapter_count : 0;
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushinteger(L, count);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_chapter_next(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectNextChapter(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_chapter_prev(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectPrevChapter(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_chapter_goto(lua_State *L)
|
|
{
|
|
int idx = luaL_checkinteger(L, 1);
|
|
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SelectChapterIdx(player, idx);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_get_time(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_tick_t time = vlc_player_GetTime(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushinteger(L, US_FROM_VLC_TICK(time));
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_position(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
float pos = vlc_player_GetPosition(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushnumber(L, pos);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_rate(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
float rate = vlc_player_GetRate(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
lua_pushnumber(L, rate);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_set_rate(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
float rate = luaL_checknumber(L, 1);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_ChangeRate(player, rate);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_increment_rate(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_IncrementRate(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_decrement_rate(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_DecrementRate(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_get_tracks_(lua_State *L,
|
|
enum es_format_category_e cat)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
|
|
size_t count = vlc_player_GetTrackCount(player, cat);
|
|
lua_createtable(L, count, 0);
|
|
|
|
for (size_t i = 0; i < count; ++i)
|
|
{
|
|
const struct vlc_player_track *track =
|
|
vlc_player_GetTrackAt(player, cat, i);
|
|
if (!track) {
|
|
continue;
|
|
}
|
|
|
|
lua_newtable(L);
|
|
|
|
lua_pushinteger(L, vlc_es_id_GetInputId(track->es_id));
|
|
lua_setfield(L, -2, "id");
|
|
|
|
lua_pushstring(L, track->name);
|
|
lua_setfield(L, -2, "name");
|
|
|
|
lua_pushboolean(L, track->selected);
|
|
lua_setfield(L, -2, "selected");
|
|
|
|
lua_rawseti(L, -2, i + 1);
|
|
}
|
|
|
|
vlc_player_Unlock(player);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_get_video_tracks(lua_State *L)
|
|
{
|
|
return vlclua_player_get_tracks_(L, VIDEO_ES);
|
|
}
|
|
|
|
static int vlclua_player_get_audio_tracks(lua_State *L)
|
|
{
|
|
return vlclua_player_get_tracks_(L, AUDIO_ES);
|
|
}
|
|
|
|
static int vlclua_player_get_spu_tracks(lua_State *L)
|
|
{
|
|
return vlclua_player_get_tracks_(L, SPU_ES);
|
|
}
|
|
|
|
static const struct vlc_player_track *
|
|
FindTrack(vlc_player_t *player, enum es_format_category_e cat, int id)
|
|
{
|
|
size_t count = vlc_player_GetTrackCount(player, cat);
|
|
for (size_t i = 0; i < count; ++i)
|
|
{
|
|
const struct vlc_player_track *track =
|
|
vlc_player_GetTrackAt(player, cat, i);
|
|
if (id == vlc_es_id_GetInputId(track->es_id))
|
|
return track;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static int vlclua_player_toggle_track_(lua_State *L,
|
|
enum es_format_category_e cat,
|
|
int id)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
|
|
if (id != -1) {
|
|
const struct vlc_player_track *track = FindTrack(player, cat, id);
|
|
|
|
if (track != NULL) {
|
|
if (track->selected)
|
|
vlc_player_UnselectTrack(player, track);
|
|
else
|
|
vlc_player_SelectTrack(player, track,
|
|
VLC_PLAYER_SELECT_EXCLUSIVE);
|
|
}
|
|
} else
|
|
vlc_player_UnselectTrackCategory(player, cat);
|
|
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_toggle_video_track(lua_State *L)
|
|
{
|
|
int id = luaL_checkinteger(L, 1);
|
|
return vlclua_player_toggle_track_(L, VIDEO_ES, id);
|
|
}
|
|
|
|
static int vlclua_player_toggle_audio_track(lua_State *L)
|
|
{
|
|
int id = luaL_checkinteger(L, 1);
|
|
return vlclua_player_toggle_track_(L, AUDIO_ES, id);
|
|
}
|
|
|
|
static int vlclua_player_toggle_spu_track(lua_State *L)
|
|
{
|
|
int id = luaL_checkinteger(L, 1);
|
|
return vlclua_player_toggle_track_(L, SPU_ES, id);
|
|
}
|
|
|
|
static int vlclua_player_next_video_frame(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_NextVideoFrame(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_seek_by_pos_(lua_State *L,
|
|
enum vlc_player_whence whence)
|
|
{
|
|
float position = luaL_checknumber(L, 1);
|
|
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SeekByPos(player, position, VLC_PLAYER_SEEK_PRECISE, whence);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_seek_by_pos_absolute(lua_State *L)
|
|
{
|
|
return vlclua_player_seek_by_pos_(L, VLC_PLAYER_WHENCE_ABSOLUTE);
|
|
}
|
|
|
|
static int vlclua_player_seek_by_pos_relative(lua_State *L)
|
|
{
|
|
return vlclua_player_seek_by_pos_(L, VLC_PLAYER_WHENCE_RELATIVE);
|
|
}
|
|
|
|
static int vlclua_player_seek_by_time_(lua_State *L,
|
|
enum vlc_player_whence whence)
|
|
{
|
|
int usec = luaL_checkinteger(L, 1);
|
|
vlc_tick_t time = VLC_TICK_FROM_US(usec);
|
|
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SeekByTime(player, time, VLC_PLAYER_SEEK_PRECISE, whence);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_seek_by_time_absolute(lua_State *L)
|
|
{
|
|
return vlclua_player_seek_by_time_(L, VLC_PLAYER_WHENCE_ABSOLUTE);
|
|
}
|
|
|
|
static int vlclua_player_seek_by_time_relative(lua_State *L)
|
|
{
|
|
return vlclua_player_seek_by_time_(L, VLC_PLAYER_WHENCE_RELATIVE);
|
|
}
|
|
|
|
static int vlclua_player_get_audio_delay(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_tick_t delay = vlc_player_GetAudioDelay(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
double delay_sec = secf_from_vlc_tick(delay);
|
|
|
|
lua_pushnumber(L, delay_sec);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_set_audio_delay(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
double delay_sec = luaL_checknumber(L, 1);
|
|
vlc_tick_t delay = vlc_tick_from_sec(delay_sec);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SetAudioDelay(player, delay, VLC_PLAYER_WHENCE_ABSOLUTE);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_player_get_subtitle_delay(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_tick_t delay = vlc_player_GetSubtitleDelay(player);
|
|
vlc_player_Unlock(player);
|
|
|
|
double delay_sec = secf_from_vlc_tick(delay);
|
|
|
|
lua_pushnumber(L, delay_sec);
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_player_set_subtitle_delay(lua_State *L)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
double delay_sec = luaL_checknumber(L, 1);
|
|
vlc_tick_t delay = vlc_tick_from_sec(delay_sec);
|
|
|
|
vlc_player_Lock(player);
|
|
vlc_player_SetSubtitleDelay(player, delay, VLC_PLAYER_WHENCE_ABSOLUTE);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vlclua_input_metas_internal( lua_State *L, input_item_t *p_item )
|
|
{
|
|
if( !p_item )
|
|
{
|
|
lua_pushnil( L );
|
|
return 1;
|
|
}
|
|
|
|
lua_newtable( L );
|
|
const char *psz_meta;
|
|
|
|
char* psz_uri = input_item_GetURI( p_item );
|
|
char* psz_filename = psz_uri ? strrchr( psz_uri, '/' ) : NULL;
|
|
|
|
if( psz_filename && psz_filename[1] == '\0' )
|
|
{
|
|
/* trailing slash, get the preceeding data */
|
|
psz_filename[0] = '\0';
|
|
psz_filename = strrchr( psz_uri, '/' );
|
|
}
|
|
|
|
if( psz_filename )
|
|
{
|
|
/* url decode, without leading slash */
|
|
psz_filename = vlc_uri_decode( psz_filename + 1 );
|
|
}
|
|
|
|
lua_pushstring( L, psz_filename );
|
|
lua_setfield( L, -2, "filename" );
|
|
|
|
free( psz_uri );
|
|
|
|
#define PUSH_META( n, m ) \
|
|
psz_meta = vlc_meta_Get( p_item->p_meta, vlc_meta_ ## n ); \
|
|
lua_pushstring( L, psz_meta ); \
|
|
lua_setfield( L, -2, m )
|
|
|
|
vlc_mutex_lock(&p_item->lock);
|
|
|
|
if (p_item->p_meta)
|
|
{
|
|
PUSH_META( Title, "title" );
|
|
PUSH_META( Artist, "artist" );
|
|
PUSH_META( Genre, "genre" );
|
|
PUSH_META( Copyright, "copyright" );
|
|
PUSH_META( Album, "album" );
|
|
PUSH_META( TrackNumber, "track_number" );
|
|
PUSH_META( Description, "description" );
|
|
PUSH_META( Rating, "rating" );
|
|
PUSH_META( Date, "date" );
|
|
PUSH_META( Setting, "setting" );
|
|
PUSH_META( URL, "url" );
|
|
PUSH_META( Language, "language" );
|
|
PUSH_META( NowPlaying, "now_playing" );
|
|
PUSH_META( Publisher, "publisher" );
|
|
PUSH_META( EncodedBy, "encoded_by" );
|
|
PUSH_META( ArtworkURL, "artwork_url" );
|
|
PUSH_META( TrackID, "track_id" );
|
|
PUSH_META( TrackTotal, "track_total" );
|
|
PUSH_META( Director, "director" );
|
|
PUSH_META( Season, "season" );
|
|
PUSH_META( Episode, "episode" );
|
|
PUSH_META( ShowName, "show_name" );
|
|
PUSH_META( Actors, "actors" );
|
|
|
|
#undef PUSH_META
|
|
|
|
char **names = vlc_meta_CopyExtraNames(p_item->p_meta);
|
|
for(int i = 0; names[i]; i++)
|
|
{
|
|
const char *meta = vlc_meta_GetExtra(p_item->p_meta, names[i]);
|
|
lua_pushstring( L, meta );
|
|
lua_setfield( L, -2, names[i] );
|
|
free(names[i]);
|
|
}
|
|
free(names);
|
|
}
|
|
vlc_mutex_unlock(&p_item->lock);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_stats( lua_State *L )
|
|
{
|
|
input_item_t *p_item = vlclua_input_item_get_internal( L );
|
|
lua_newtable( L );
|
|
if( p_item == NULL )
|
|
return 1;
|
|
|
|
vlc_mutex_lock( &p_item->lock );
|
|
input_stats_t *p_stats = p_item->p_stats;
|
|
if( p_stats != NULL )
|
|
{
|
|
#define STATS_INT( n ) lua_pushinteger( L, p_item->p_stats->i_ ## n ); \
|
|
lua_setfield( L, -2, #n );
|
|
#define STATS_FLOAT( n ) lua_pushnumber( L, p_item->p_stats->f_ ## n ); \
|
|
lua_setfield( L, -2, #n );
|
|
STATS_INT( read_packets )
|
|
STATS_INT( read_bytes )
|
|
STATS_FLOAT( input_bitrate )
|
|
STATS_INT( demux_read_packets )
|
|
STATS_INT( demux_read_bytes )
|
|
STATS_FLOAT( demux_bitrate )
|
|
STATS_INT( demux_corrupted )
|
|
STATS_INT( demux_discontinuity )
|
|
STATS_INT( decoded_audio )
|
|
STATS_INT( decoded_video )
|
|
STATS_INT( displayed_pictures )
|
|
STATS_INT( late_pictures )
|
|
STATS_INT( lost_pictures )
|
|
STATS_INT( played_abuffers )
|
|
STATS_INT( lost_abuffers )
|
|
#undef STATS_INT
|
|
#undef STATS_FLOAT
|
|
}
|
|
vlc_mutex_unlock( &p_item->lock );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_add_subtitle(lua_State *L, bool b_path)
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
if (!lua_isstring(L, 1))
|
|
return luaL_error( L, "vlc.player.add_subtitle() usage: (path, autoselect=false)" );
|
|
|
|
bool autoselect = false;
|
|
if (lua_gettop(L) >= 2)
|
|
autoselect = lua_toboolean(L, 2);
|
|
|
|
const char *sub = luaL_checkstring(L, 1);
|
|
char *mrl;
|
|
if (b_path)
|
|
mrl = vlc_path2uri(sub, NULL);
|
|
|
|
const char *uri = b_path ? mrl : sub;
|
|
vlc_player_Lock(player);
|
|
vlc_player_AddAssociatedMedia(player, SPU_ES, uri, autoselect, true, false);
|
|
vlc_player_Unlock(player);
|
|
if (b_path)
|
|
free(mrl);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_add_subtitle_path( lua_State *L )
|
|
{
|
|
return vlclua_input_add_subtitle( L, true );
|
|
}
|
|
|
|
static int vlclua_input_add_subtitle_mrl( lua_State *L )
|
|
{
|
|
return vlclua_input_add_subtitle( L, false );
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Input items
|
|
*****************************************************************************/
|
|
|
|
static input_item_t* vlclua_input_item_get_internal( lua_State *L )
|
|
{
|
|
input_item_t **pp_item = luaL_checkudata( L, 1, "input_item" );
|
|
input_item_t *p_item = *pp_item;
|
|
|
|
if( !p_item )
|
|
luaL_error( L, "script went completely foobar" );
|
|
|
|
return p_item;
|
|
}
|
|
|
|
/* Garbage collection of an input_item_t */
|
|
static int vlclua_input_item_delete( lua_State *L )
|
|
{
|
|
input_item_t **pp_item = luaL_checkudata( L, 1, "input_item" );
|
|
input_item_t *p_item = *pp_item;
|
|
|
|
if( !p_item )
|
|
return luaL_error( L, "script went completely foobar" );
|
|
|
|
*pp_item = NULL;
|
|
input_item_Release( p_item );
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_get_current( lua_State *L )
|
|
{
|
|
vlc_player_t *player = vlclua_get_player_internal(L);
|
|
|
|
vlc_player_Lock(player);
|
|
input_item_t *media = vlc_player_GetCurrentMedia(player);
|
|
if (media)
|
|
vlclua_input_item_get(L, media);
|
|
else
|
|
lua_pushnil(L);
|
|
vlc_player_Unlock(player);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_metas( lua_State *L )
|
|
{
|
|
vlclua_input_metas_internal( L, vlclua_input_item_get_internal( L ) );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_is_preparsed( lua_State *L )
|
|
{
|
|
lua_pushboolean( L, input_item_IsPreparsed( vlclua_input_item_get_internal( L ) ) );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_uri( lua_State *L )
|
|
{
|
|
char *uri = input_item_GetURI( vlclua_input_item_get_internal( L ) );
|
|
lua_pushstring( L, uri );
|
|
free( uri );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_name( lua_State *L )
|
|
{
|
|
char *name = input_item_GetName( vlclua_input_item_get_internal( L ) );
|
|
lua_pushstring( L, name );
|
|
free( name );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_duration( lua_State *L )
|
|
{
|
|
vlc_tick_t duration = input_item_GetDuration( vlclua_input_item_get_internal( L ) );
|
|
lua_pushnumber( L, secf_from_vlc_tick(duration) );
|
|
return 1;
|
|
}
|
|
|
|
static int vlclua_input_item_set_meta( lua_State *L )
|
|
{
|
|
input_item_t *p_item = vlclua_input_item_get_internal( L );
|
|
lua_settop( L, 1 + 2 ); // two arguments
|
|
const char *psz_name = luaL_checkstring( L, 2 ),
|
|
*psz_value = luaL_checkstring( L, 3 );
|
|
|
|
#define META_TYPE( n, s ) { s, vlc_meta_ ## n },
|
|
static const struct
|
|
{
|
|
const char psz_name[15];
|
|
unsigned char type;
|
|
} pp_meta_types[] = {
|
|
META_TYPE( Title, "title" )
|
|
META_TYPE( Artist, "artist" )
|
|
META_TYPE( Genre, "genre" )
|
|
META_TYPE( Copyright, "copyright" )
|
|
META_TYPE( Album, "album" )
|
|
META_TYPE( TrackNumber, "track_number" )
|
|
META_TYPE( Description, "description" )
|
|
META_TYPE( Rating, "rating" )
|
|
META_TYPE( Date, "date" )
|
|
META_TYPE( Setting, "setting" )
|
|
META_TYPE( URL, "url" )
|
|
META_TYPE( Language, "language" )
|
|
META_TYPE( NowPlaying, "now_playing" )
|
|
META_TYPE( ESNowPlaying, "now_playing" )
|
|
META_TYPE( Publisher, "publisher" )
|
|
META_TYPE( EncodedBy, "encoded_by" )
|
|
META_TYPE( ArtworkURL, "artwork_url" )
|
|
META_TYPE( TrackID, "track_id" )
|
|
META_TYPE( TrackTotal, "track_total" )
|
|
META_TYPE( Director, "director" )
|
|
META_TYPE( Season, "season" )
|
|
META_TYPE( Episode, "episode" )
|
|
META_TYPE( ShowName, "show_name" )
|
|
META_TYPE( Actors, "actors" )
|
|
META_TYPE( AlbumArtist, "album_artist" )
|
|
META_TYPE( DiscNumber, "disc_number" )
|
|
META_TYPE( DiscTotal, "disc_total" )
|
|
};
|
|
#undef META_TYPE
|
|
|
|
static_assert( sizeof(pp_meta_types)
|
|
== VLC_META_TYPE_COUNT * sizeof(pp_meta_types[0]),
|
|
"Inconsistent meta data types" );
|
|
vlc_meta_type_t type = vlc_meta_Title;
|
|
for( unsigned i = 0; i < VLC_META_TYPE_COUNT; i++ )
|
|
{
|
|
if( !strcasecmp( pp_meta_types[i].psz_name, psz_name ) )
|
|
{
|
|
type = pp_meta_types[i].type;
|
|
input_item_SetMeta( p_item, type, psz_value );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
vlc_meta_AddExtra( p_item->p_meta, psz_name, psz_value );
|
|
return 1;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Lua bindings
|
|
*****************************************************************************/
|
|
static const luaL_Reg vlclua_input_reg[] = {
|
|
{ "is_playing", vlclua_input_is_playing },
|
|
{ "item", vlclua_input_item_get_current },
|
|
{ "add_subtitle", vlclua_input_add_subtitle_path },
|
|
{ "add_subtitle_mrl", vlclua_input_add_subtitle_mrl },
|
|
{ "get_title_index", vlclua_player_get_title_index },
|
|
{ "get_titles_count", vlclua_player_get_titles_count },
|
|
{ "title_next", vlclua_player_title_next },
|
|
{ "title_prev", vlclua_player_title_prev },
|
|
{ "title_goto", vlclua_player_title_goto },
|
|
{ "get_chapter_index", vlclua_player_get_chapter_index },
|
|
{ "get_chapters_count", vlclua_player_get_chapters_count },
|
|
{ "chapter_next", vlclua_player_chapter_next },
|
|
{ "chapter_prev", vlclua_player_chapter_prev },
|
|
{ "chapter_goto", vlclua_player_chapter_goto },
|
|
{ "get_time", vlclua_player_get_time },
|
|
{ "get_position", vlclua_player_get_position },
|
|
{ "get_rate", vlclua_player_get_rate },
|
|
{ "set_rate", vlclua_player_set_rate },
|
|
{ "increment_rate", vlclua_player_increment_rate },
|
|
{ "decrement_rate", vlclua_player_decrement_rate },
|
|
{ "get_video_tracks", vlclua_player_get_video_tracks },
|
|
{ "get_audio_tracks", vlclua_player_get_audio_tracks },
|
|
{ "get_spu_tracks", vlclua_player_get_spu_tracks },
|
|
{ "toggle_video_track", vlclua_player_toggle_video_track },
|
|
{ "toggle_audio_track", vlclua_player_toggle_audio_track },
|
|
{ "toggle_spu_track", vlclua_player_toggle_spu_track },
|
|
{ "next_video_frame", vlclua_player_next_video_frame },
|
|
{ "seek_by_pos_absolute", vlclua_player_seek_by_pos_absolute },
|
|
{ "seek_by_pos_relative", vlclua_player_seek_by_pos_relative },
|
|
{ "seek_by_time_absolute", vlclua_player_seek_by_time_absolute },
|
|
{ "seek_by_time_relative", vlclua_player_seek_by_time_relative },
|
|
{ "get_audio_delay", vlclua_player_get_audio_delay },
|
|
{ "set_audio_delay", vlclua_player_set_audio_delay },
|
|
{ "get_subtitle_delay", vlclua_player_get_subtitle_delay },
|
|
{ "set_subtitle_delay", vlclua_player_set_subtitle_delay },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
void luaopen_input( lua_State *L )
|
|
{
|
|
lua_newtable( L );
|
|
luaL_register( L, NULL, vlclua_input_reg );
|
|
lua_setfield( L, -2, "player" );
|
|
}
|
|
|
|
static const luaL_Reg vlclua_input_item_reg[] = {
|
|
{ "is_preparsed", vlclua_input_item_is_preparsed },
|
|
{ "metas", vlclua_input_item_metas },
|
|
{ "set_meta", vlclua_input_item_set_meta },
|
|
{ "uri", vlclua_input_item_uri },
|
|
{ "name", vlclua_input_item_name },
|
|
{ "duration", vlclua_input_item_duration },
|
|
{ "stats", vlclua_input_item_stats },
|
|
{ "info", vlclua_input_item_info },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
int vlclua_input_item_get( lua_State *L, input_item_t *p_item )
|
|
{
|
|
input_item_Hold( p_item );
|
|
input_item_t **pp = lua_newuserdata( L, sizeof( input_item_t* ) );
|
|
*pp = p_item;
|
|
|
|
if( luaL_newmetatable( L, "input_item" ) )
|
|
{
|
|
lua_newtable( L );
|
|
luaL_register( L, NULL, vlclua_input_item_reg );
|
|
lua_setfield( L, -2, "__index" );
|
|
lua_pushcfunction( L, vlclua_input_item_delete );
|
|
lua_setfield( L, -2, "__gc" );
|
|
}
|
|
|
|
lua_setmetatable(L, -2);
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
void luaopen_input_item( lua_State *L, input_item_t *item )
|
|
{
|
|
assert(item);
|
|
vlclua_input_item_get( L, item );
|
|
lua_setfield( L, -2, "item" );
|
|
}
|
|
|