📄 xspf.c
字号:
/******************************************************************************* * 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 + -