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.
167 lines
4.9 KiB
167 lines
4.9 KiB
/*****************************************************************************
|
|
* xmlparser.cpp
|
|
*****************************************************************************
|
|
* Copyright (C) 2004 the VideoLAN team
|
|
* $Id$
|
|
*
|
|
* Authors: Cyril Deguet <asmax@via.ecp.fr>
|
|
*
|
|
* 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.
|
|
*****************************************************************************/
|
|
|
|
#include "xmlparser.hpp"
|
|
#include "../src/os_factory.hpp"
|
|
#include <vlc_url.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <vlc_fs.h>
|
|
|
|
XMLParser::XMLParser( intf_thread_t *pIntf, const std::string &rFileName )
|
|
: SkinObject( pIntf ), m_pXML( NULL ), m_pReader( NULL ), m_pStream( NULL )
|
|
{
|
|
m_pXML = xml_Create( pIntf );
|
|
if( !m_pXML )
|
|
{
|
|
msg_Err( getIntf(), "cannot initialize xml" );
|
|
return;
|
|
}
|
|
|
|
LoadCatalog();
|
|
|
|
char *psz_uri = vlc_path2uri( rFileName.c_str(), NULL );
|
|
m_pStream = stream_UrlNew( pIntf, psz_uri );
|
|
free( psz_uri );
|
|
if( !m_pStream )
|
|
{
|
|
msg_Err( getIntf(), "failed to open %s for reading",
|
|
rFileName.c_str() );
|
|
return;
|
|
}
|
|
|
|
m_pReader = xml_ReaderCreate( m_pXML, m_pStream );
|
|
if( !m_pReader )
|
|
{
|
|
msg_Err( getIntf(), "failed to open %s for parsing",
|
|
rFileName.c_str() );
|
|
return;
|
|
}
|
|
|
|
xml_ReaderUseDTD( m_pReader );
|
|
}
|
|
|
|
|
|
XMLParser::~XMLParser()
|
|
{
|
|
if( m_pReader ) xml_ReaderDelete( m_pReader );
|
|
if( m_pXML ) xml_Delete( m_pXML );
|
|
if( m_pStream ) stream_Delete( m_pStream );
|
|
}
|
|
|
|
|
|
void XMLParser::LoadCatalog()
|
|
{
|
|
// Get the resource path and look for the DTD
|
|
OSFactory *pOSFactory = OSFactory::instance( getIntf() );
|
|
const std::list<std::string> &resPath = pOSFactory->getResourcePath();
|
|
const std::string &sep = pOSFactory->getDirSeparator();
|
|
std::list<std::string>::const_iterator it;
|
|
|
|
struct stat statBuf;
|
|
|
|
// Try to load the catalog first (needed at least on win32 where
|
|
// we don't have a default catalog)
|
|
for( it = resPath.begin(); it != resPath.end(); ++it )
|
|
{
|
|
std::string catalog_path = (*it) + sep + "skin.catalog";
|
|
if( !vlc_stat( catalog_path.c_str(), &statBuf ) )
|
|
{
|
|
msg_Dbg( getIntf(), "Using catalog %s", catalog_path.c_str() );
|
|
xml_CatalogLoad( m_pXML, catalog_path.c_str() );
|
|
break;
|
|
}
|
|
}
|
|
if( it == resPath.end() )
|
|
{
|
|
// Ok, try the default one
|
|
xml_CatalogLoad( m_pXML, NULL );
|
|
}
|
|
|
|
for( it = resPath.begin(); it != resPath.end(); ++it )
|
|
{
|
|
std::string path = (*it) + sep + "skin.dtd";
|
|
if( !vlc_stat( path.c_str(), &statBuf ) )
|
|
{
|
|
// DTD found
|
|
msg_Dbg( getIntf(), "using DTD %s", path.c_str() );
|
|
|
|
// Add an entry in the default catalog
|
|
xml_CatalogAdd( m_pXML, "public",
|
|
"-//VideoLAN//DTD VLC Skins V"
|
|
SKINS_DTD_VERSION "//EN", path.c_str() );
|
|
break;
|
|
}
|
|
}
|
|
if( it == resPath.end() )
|
|
{
|
|
msg_Err( getIntf(), "cannot find the skins DTD");
|
|
}
|
|
}
|
|
|
|
bool XMLParser::parse()
|
|
{
|
|
const char *node;
|
|
int type;
|
|
|
|
if( !m_pReader ) return false;
|
|
|
|
m_errors = false;
|
|
|
|
while( (type = xml_ReaderNextNode( m_pReader, &node )) > 0 )
|
|
{
|
|
if( m_errors ) return false;
|
|
|
|
switch( type )
|
|
{
|
|
case XML_READER_STARTELEM:
|
|
{
|
|
// Read the attributes
|
|
AttrList_t attributes;
|
|
const char *name, *value;
|
|
while( (name = xml_ReaderNextAttr( m_pReader, &value )) != NULL )
|
|
attributes[strdup(name)] = strdup(value);
|
|
|
|
handleBeginElement( node, attributes );
|
|
|
|
std::map<const char*, const char*, ltstr> ::iterator it =
|
|
attributes.begin();
|
|
while( it != attributes.end() )
|
|
{
|
|
free( (char *)it->first );
|
|
free( (char *)it->second );
|
|
++it;
|
|
}
|
|
break;
|
|
}
|
|
|
|
// End element
|
|
case XML_READER_ENDELEM:
|
|
{
|
|
handleEndElement( node );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return (type == 0 && !m_errors );
|
|
}
|
|
|