⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xspf.c

📁 VLC Player Source Code
💻 C
📖 第 1 页 / 共 3 页
字号:
/******************************************************************************* * xspf.c : XSPF playlist import functions ******************************************************************************* * Copyright (C) 2006 the VideoLAN team * $Id: 991291522db032594a5c4b9e7eba2fc915147bd3 $ * * Authors: Daniel Stränger <vlc at schmaller dot de> *          Yoann Peronneau <yoann@videolan.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. ******************************************************************************//** * \file modules/demux/playlist/xspf.c * \brief XSPF playlist import functions */#ifdef HAVE_CONFIG_H# include "config.h"#endif#include <vlc_common.h>#include <vlc_demux.h>#include <vlc_xml.h>#include <vlc_strings.h>#include <vlc_url.h>#include "xspf.h"#include "playlist.h"struct demux_sys_t{    input_item_t **pp_tracklist;    int i_tracklist_entries;    int i_identifier;    char * psz_base;};static int Control( demux_t *, int, va_list );static int Demux( demux_t * );/** * \brief XSPF submodule initialization function */int Import_xspf( vlc_object_t *p_this ){    DEMUX_BY_EXTENSION_OR_FORCED_MSG( ".xspf", "xspf-open",                                      "using XSPF playlist reader" );    return VLC_SUCCESS;}void Close_xspf( vlc_object_t *p_this ){    demux_t *p_demux = (demux_t *)p_this;    int i;    for(i = 0; i < p_demux->p_sys->i_tracklist_entries; i++)    {        if(p_demux->p_sys->pp_tracklist[i])            vlc_gc_decref( p_demux->p_sys->pp_tracklist[i] );    }    FREENULL( p_demux->p_sys->pp_tracklist );    FREENULL( p_demux->p_sys->psz_base );    free( p_demux->p_sys );}/** * \brief demuxer function for XSPF parsing */int Demux( demux_t *p_demux ){    int i_ret = 1;    xml_t *p_xml = NULL;    xml_reader_t *p_xml_reader = NULL;    char *psz_name = NULL;    INIT_PLAYLIST_STUFF;    p_demux->p_sys->pp_tracklist = NULL;    p_demux->p_sys->i_tracklist_entries = 0;    p_demux->p_sys->i_identifier = 0;    p_demux->p_sys->psz_base = NULL;    /* create new xml parser from stream */    p_xml = xml_Create( p_demux );    if( !p_xml )        i_ret = -1;    else    {        p_xml_reader = xml_ReaderCreate( p_xml, p_demux->s );        if( !p_xml_reader )            i_ret = -1;    }    /* locating the root node */    if( i_ret == 1 )    {        do        {            if( xml_ReaderRead( p_xml_reader ) != 1 )            {                msg_Err( p_demux, "can't read xml stream" );                i_ret = -1;            }        } while( i_ret == VLC_SUCCESS &&                 xml_ReaderNodeType( p_xml_reader ) != XML_READER_STARTELEM );    }    /* checking root node name */    if( i_ret == 1 )    {        psz_name = xml_ReaderName( p_xml_reader );        if( !psz_name || strcmp( psz_name, "playlist" ) )        {            msg_Err( p_demux, "invalid root node name: %s", psz_name );            i_ret = -1;        }        FREE_NAME();    }    if( i_ret == 1 )        i_ret = parse_playlist_node( p_demux, p_current_input,                                     p_xml_reader, "playlist" ) ? 0 : -1;    int i;    for( i = 0 ; i < p_demux->p_sys->i_tracklist_entries ; i++ )    {        input_item_t *p_new_input = p_demux->p_sys->pp_tracklist[i];        if( p_new_input )        {            input_item_AddSubItem( p_current_input, p_new_input );        }    }    HANDLE_PLAY_AND_RELEASE;    if( p_xml_reader )        xml_ReaderDelete( p_xml, p_xml_reader );    if( p_xml )        xml_Delete( p_xml );    return i_ret; /* Needed for correct operation of go back */}/** \brief dummy function for demux callback interface */static int Control( demux_t *p_demux, int i_query, va_list args ){    VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args);    return VLC_EGENERIC;}/** * \brief parse the root node of a XSPF playlist * \param p_demux demuxer instance * \param p_input_item current input item * \param p_xml_reader xml reader instance * \param psz_element name of element to parse */static bool parse_playlist_node COMPLEX_INTERFACE{    char *psz_name=NULL;    char *psz_value=NULL;    bool b_version_found = false;    int i_node;    xml_elem_hnd_t *p_handler=NULL;    xml_elem_hnd_t pl_elements[] =        { {"title",        SIMPLE_CONTENT,  {.smpl = set_item_info} },          {"creator",      SIMPLE_CONTENT,  {.smpl = set_item_info} },          {"annotation",   SIMPLE_CONTENT,  {.smpl = set_item_info} },          {"info",         SIMPLE_CONTENT,  {NULL} },          {"location",     SIMPLE_CONTENT,  {NULL} },          {"identifier",   SIMPLE_CONTENT,  {NULL} },          {"image",        SIMPLE_CONTENT,  {.smpl = set_item_info} },          {"date",         SIMPLE_CONTENT,  {NULL} },          {"license",      SIMPLE_CONTENT,  {NULL} },          {"attribution",  COMPLEX_CONTENT, {.cmplx = skip_element} },          {"link",         SIMPLE_CONTENT,  {NULL} },          {"meta",         SIMPLE_CONTENT,  {NULL} },          {"extension",    COMPLEX_CONTENT, {.cmplx = parse_extension_node} },          {"trackList",    COMPLEX_CONTENT, {.cmplx = parse_tracklist_node} },          {NULL,           UNKNOWN_CONTENT, {NULL} }        };    /* read all playlist attributes */    while( xml_ReaderNextAttr( p_xml_reader ) == VLC_SUCCESS )    {        psz_name = xml_ReaderName( p_xml_reader );        psz_value = xml_ReaderValue( p_xml_reader );        if( !psz_name || !psz_value )        {            msg_Err( p_demux, "invalid xml stream @ <playlist>" );            FREE_ATT();            return false;        }        /* attribute: version */        if( !strcmp( psz_name, "version" ) )        {            b_version_found = true;            if( strcmp( psz_value, "0" ) && strcmp( psz_value, "1" ) )                msg_Warn( p_demux, "unsupported XSPF version" );        }        /* attribute: xmlns */        else if( !strcmp( psz_name, "xmlns" ) )            ;        else if( !strcmp( psz_name, "xml:base" ) )        {            p_demux->p_sys->psz_base = decode_URI_duplicate( psz_value );        }        /* unknown attribute */        else            msg_Warn( p_demux, "invalid <playlist> attribute:\"%s\"", psz_name);        FREE_ATT();    }    /* attribute version is mandatory !!! */    if( !b_version_found )        msg_Warn( p_demux, "<playlist> requires \"version\" attribute" );    /* parse the child elements - we only take care of <trackList> */    while( xml_ReaderRead( p_xml_reader ) == 1 )    {        i_node = xml_ReaderNodeType( p_xml_reader );        switch( i_node )        {            case XML_READER_NONE:                break;            case XML_READER_STARTELEM:                /*  element start tag  */                psz_name = xml_ReaderName( p_xml_reader );                if( !psz_name || !*psz_name )                {                    msg_Err( p_demux, "invalid xml stream" );                    FREE_ATT();                    return false;                }                /* choose handler */                for( p_handler = pl_elements;                     p_handler->name && strcmp( psz_name, p_handler->name );                     p_handler++ );                if( !p_handler->name )                {                    msg_Err( p_demux, "unexpected element <%s>", psz_name );                    FREE_ATT();                    return false;                }                FREE_NAME();                /* complex content is parsed in a separate function */                if( p_handler->type == COMPLEX_CONTENT )                {                    if( p_handler->pf_handler.cmplx( p_demux,                                                     p_input_item,                                                     p_xml_reader,                                                     p_handler->name ) )                    {                        p_handler = NULL;                        FREE_ATT();                    }                    else                    {                        FREE_ATT();                        return false;                    }                }                break;            case XML_READER_TEXT:                /* simple element content */                FREE_ATT();                psz_value = xml_ReaderValue( p_xml_reader );                if( !psz_value )                {                    msg_Err( p_demux, "invalid xml stream" );                    FREE_ATT();                    return false;                }                break;            case XML_READER_ENDELEM:                /* element end tag */                psz_name = xml_ReaderName( p_xml_reader );                if( !psz_name )                {                    msg_Err( p_demux, "invalid xml stream" );                    FREE_ATT();                    return false;                }                /* leave if the current parent node <playlist> is terminated */                if( !strcmp( psz_name, psz_element ) )                {                    FREE_ATT();                    return true;                }                /* there MUST have been a start tag for that element name */                if( !p_handler || !p_handler->name                    || strcmp( p_handler->name, psz_name ))                {                    msg_Err( p_demux, "there's no open element left for <%s>",                             psz_name );                    FREE_ATT();                    return false;                }                if( p_handler->pf_handler.smpl )                {                    p_handler->pf_handler.smpl( p_input_item, p_handler->name,                                                psz_value );                }                FREE_ATT();                p_handler = NULL;                break;            default:                /* unknown/unexpected xml node */                msg_Err( p_demux, "unexpected xml node %i", i_node );                FREE_ATT();

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -