Blame libdjvu/DjVuMessage.cpp

Packit df99a1
//C-  -*- C++ -*-
Packit df99a1
//C- -------------------------------------------------------------------
Packit df99a1
//C- DjVuLibre-3.5
Packit df99a1
//C- Copyright (c) 2002  Leon Bottou and Yann Le Cun.
Packit df99a1
//C- Copyright (c) 2001  AT&T
Packit df99a1
//C-
Packit df99a1
//C- This software is subject to, and may be distributed under, the
Packit df99a1
//C- GNU General Public License, either Version 2 of the license,
Packit df99a1
//C- or (at your option) any later version. The license should have
Packit df99a1
//C- accompanied the software or you may obtain a copy of the license
Packit df99a1
//C- from the Free Software Foundation at http://www.fsf.org .
Packit df99a1
//C-
Packit df99a1
//C- This program is distributed in the hope that it will be useful,
Packit df99a1
//C- but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit df99a1
//C- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit df99a1
//C- GNU General Public License for more details.
Packit df99a1
//C- 
Packit df99a1
//C- DjVuLibre-3.5 is derived from the DjVu(r) Reference Library from
Packit df99a1
//C- Lizardtech Software.  Lizardtech Software has authorized us to
Packit df99a1
//C- replace the original DjVu(r) Reference Library notice by the following
Packit df99a1
//C- text (see doc/lizard2002.djvu and doc/lizardtech2007.djvu):
Packit df99a1
//C-
Packit df99a1
//C-  ------------------------------------------------------------------
Packit df99a1
//C- | DjVu (r) Reference Library (v. 3.5)
Packit df99a1
//C- | Copyright (c) 1999-2001 LizardTech, Inc. All Rights Reserved.
Packit df99a1
//C- | The DjVu Reference Library is protected by U.S. Pat. No.
Packit df99a1
//C- | 6,058,214 and patents pending.
Packit df99a1
//C- |
Packit df99a1
//C- | This software is subject to, and may be distributed under, the
Packit df99a1
//C- | GNU General Public License, either Version 2 of the license,
Packit df99a1
//C- | or (at your option) any later version. The license should have
Packit df99a1
//C- | accompanied the software or you may obtain a copy of the license
Packit df99a1
//C- | from the Free Software Foundation at http://www.fsf.org .
Packit df99a1
//C- |
Packit df99a1
//C- | The computer code originally released by LizardTech under this
Packit df99a1
//C- | license and unmodified by other parties is deemed "the LIZARDTECH
Packit df99a1
//C- | ORIGINAL CODE."  Subject to any third party intellectual property
Packit df99a1
//C- | claims, LizardTech grants recipient a worldwide, royalty-free, 
Packit df99a1
//C- | non-exclusive license to make, use, sell, or otherwise dispose of 
Packit df99a1
//C- | the LIZARDTECH ORIGINAL CODE or of programs derived from the 
Packit df99a1
//C- | LIZARDTECH ORIGINAL CODE in compliance with the terms of the GNU 
Packit df99a1
//C- | General Public License.   This grant only confers the right to 
Packit df99a1
//C- | infringe patent claims underlying the LIZARDTECH ORIGINAL CODE to 
Packit df99a1
//C- | the extent such infringement is reasonably necessary to enable 
Packit df99a1
//C- | recipient to make, have made, practice, sell, or otherwise dispose 
Packit df99a1
//C- | of the LIZARDTECH ORIGINAL CODE (or portions thereof) and not to 
Packit df99a1
//C- | any greater extent that may be necessary to utilize further 
Packit df99a1
//C- | modifications or combinations.
Packit df99a1
//C- |
Packit df99a1
//C- | The LIZARDTECH ORIGINAL CODE is provided "AS IS" WITHOUT WARRANTY
Packit df99a1
//C- | OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
Packit df99a1
//C- | TO ANY WARRANTY OF NON-INFRINGEMENT, OR ANY IMPLIED WARRANTY OF
Packit df99a1
//C- | MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
Packit df99a1
//C- +------------------------------------------------------------------
Packit df99a1
Packit df99a1
#ifdef HAVE_CONFIG_H
Packit df99a1
# include "config.h"
Packit df99a1
#endif
Packit df99a1
#if NEED_GNUG_PRAGMAS
Packit df99a1
# pragma implementation
Packit df99a1
#endif
Packit df99a1
Packit df99a1
// From: Leon Bottou, 1/31/2002
Packit df99a1
// All these XML messages are Lizardtech innovations.
Packit df99a1
Packit df99a1
#include "DjVuMessage.h"
Packit df99a1
#include "GOS.h"
Packit df99a1
#include "XMLTags.h"
Packit df99a1
#include "ByteStream.h"
Packit df99a1
#include "GURL.h"
Packit df99a1
#include "debug.h"
Packit df99a1
#include <ctype.h>
Packit df99a1
#include <string.h>
Packit df99a1
#include <stddef.h>
Packit df99a1
#include <stdlib.h>
Packit df99a1
#ifdef _WIN32
Packit df99a1
# include <tchar.h>
Packit df99a1
# include <windows.h>
Packit df99a1
# include <winreg.h>
Packit df99a1
#endif
Packit df99a1
#ifdef UNIX
Packit df99a1
# include <unistd.h>
Packit df99a1
# include <pwd.h>
Packit df99a1
# include <sys/types.h>
Packit df99a1
#endif
Packit df99a1
#include <locale.h>
Packit df99a1
#ifndef LC_MESSAGES
Packit df99a1
# define LC_MESSAGES LC_ALL
Packit df99a1
#endif
Packit df99a1
Packit df99a1
Packit df99a1
#ifdef HAVE_NAMESPACES
Packit df99a1
namespace DJVU {
Packit df99a1
# ifdef NOT_DEFINED // Just to fool emacs c++ mode
Packit df99a1
}
Packit df99a1
#endif
Packit df99a1
#endif
Packit df99a1
Packit df99a1
GUTF8String &
Packit df99a1
DjVuMessage::programname(void)
Packit df99a1
{
Packit df99a1
  static GUTF8String xprogramname;
Packit df99a1
  use_language();
Packit df99a1
  return xprogramname;
Packit df99a1
}
Packit df99a1
Packit df99a1
static const char namestring[]="name";
Packit df99a1
static const char srcstring[]="src";
Packit df99a1
Packit df99a1
static const char *failed_to_parse_XML=ERR_MSG("DjVuMessage.failed_to_parse_XML");
Packit df99a1
static const char bodystring[]="BODY";
Packit df99a1
static const char languagestring[]="LANGUAGE";
Packit df99a1
static const char headstring[]="HEAD";
Packit df99a1
static const char includestring[]="INCLUDE";
Packit df99a1
static const char messagestring[]="MESSAGE";
Packit df99a1
static const char localestring[]="locale";
Packit df99a1
Packit df99a1
Packit df99a1
// directory names for searching messages
Packit df99a1
#ifdef AUTOCONF
Packit df99a1
static const char DjVuDataDir[] = DIR_DATADIR "/djvu/osi";
Packit df99a1
#endif /* AUTOCONF */
Packit df99a1
static const char ModuleDjVuDir[] ="share/djvu/osi";
Packit df99a1
static const char ProfilesDjVuDir[] ="profiles";
Packit df99a1
static const char LocalDjVuDir[] =".DjVu";      // relative to ${HOME}
Packit df99a1
#ifdef LT_DEFAULT_PREFIX
Packit df99a1
static const char DjVuPrefixDir[] = LT_DEFAULT_PREFIX "/profiles";
Packit df99a1
#endif
Packit df99a1
#ifndef NDEBUG
Packit df99a1
static const char DebugModuleDjVuDir[] ="../TOPDIR/SRCDIR/profiles";
Packit df99a1
#endif
Packit df99a1
#ifdef _WIN32
Packit df99a1
static const char RootDjVuDir[] ="C:/Program Files/LizardTech/Profiles";
Packit df99a1
static const TCHAR registrypath[]= TEXT("Software\\LizardTech\\DjVu\\Profile Path");
Packit df99a1
#else
Packit df99a1
static const char RootDjVuDir[] ="/etc/DjVu/";  // global last resort
Packit df99a1
#endif
Packit df99a1
Packit df99a1
static const char DjVuEnv[] = "DJVU_CONFIG_DIR";
Packit df99a1
Packit df99a1
//  The name of the message file
Packit df99a1
static const char MessageFile[]="messages.xml";
Packit df99a1
static const char LanguageFile[]="languages.xml";
Packit df99a1
Packit df99a1
#ifdef _WIN32
Packit df99a1
static GURL
Packit df99a1
RegOpenReadConfig ( HKEY hParentKey )
Packit df99a1
{
Packit df99a1
  GURL retval;
Packit df99a1
  LPCTSTR path = registrypath;
Packit df99a1
  HKEY hKey = 0;
Packit df99a1
  if (RegOpenKeyEx(hParentKey, path, 0,
Packit df99a1
		   KEY_READ, &hKey) == ERROR_SUCCESS )
Packit df99a1
  {
Packit df99a1
    CHAR path[1024];
Packit df99a1
    CHAR *szPathValue = path;
Packit df99a1
    LPCSTR lpszEntry = "";
Packit df99a1
    DWORD dwCount = (sizeof(path)/sizeof(CHAR))-1;
Packit df99a1
    DWORD dwType;
Packit df99a1
    LONG lResult = RegQueryValueExA(hKey, lpszEntry, NULL,
Packit df99a1
			&dwType, (LPBYTE)szPathValue, &dwCount);
Packit df99a1
    RegCloseKey(hKey);
Packit df99a1
    if ((lResult == ERROR_SUCCESS))
Packit df99a1
    {
Packit df99a1
      szPathValue[dwCount] = 0;
Packit df99a1
      retval=GURL::Filename::Native(path);
Packit df99a1
    }
Packit df99a1
  } 
Packit df99a1
  return retval;
Packit df99a1
}
Packit df99a1
Packit df99a1
static GURL
Packit df99a1
GetModulePath( void )
Packit df99a1
{
Packit df99a1
  const GUTF8String cwd(GOS::cwd());
Packit df99a1
  CHAR path[1024];
Packit df99a1
  DWORD dwCount = (sizeof(path)/sizeof(CHAR))-1;
Packit df99a1
  GetModuleFileNameA(0, path, dwCount);
Packit df99a1
  GURL retval=GURL::Filename::Native(path).base();
Packit df99a1
  GOS::cwd(cwd);
Packit df99a1
  return retval;
Packit df99a1
}
Packit df99a1
#elif defined(UNIX)
Packit df99a1
Packit df99a1
static GList<GURL>
Packit df99a1
parsePATH(void)
Packit df99a1
{
Packit df99a1
  GList<GURL> retval;
Packit df99a1
  const char *path=getenv("PATH");
Packit df99a1
  if(path)
Packit df99a1
  {
Packit df99a1
    GNativeString p(path);
Packit df99a1
    int from=0;
Packit df99a1
    for(int to;(to=p.search(':',from))>0;from=to+1)
Packit df99a1
    {
Packit df99a1
      if(to > from)
Packit df99a1
      {
Packit df99a1
        retval.append(GURL::Filename::Native(p.substr(from,to-from)));
Packit df99a1
      }
Packit df99a1
    }
Packit df99a1
    if((from+1)<(int)p.length())
Packit df99a1
    {
Packit df99a1
      retval.append(GURL::Filename::Native(p.substr(from,-1)));
Packit df99a1
    }
Packit df99a1
  }
Packit df99a1
  return retval;
Packit df99a1
}
Packit df99a1
Packit df99a1
static GURL
Packit df99a1
GetModulePath( void )
Packit df99a1
{
Packit df99a1
 GURL retval;
Packit df99a1
 GUTF8String &xprogramname=DjVuMessage::programname();
Packit df99a1
 if(xprogramname.length())
Packit df99a1
 {
Packit df99a1
   if(xprogramname[1]=='/'
Packit df99a1
     ||!xprogramname.cmp("../",3)
Packit df99a1
     ||!xprogramname.cmp("./",2))
Packit df99a1
   {
Packit df99a1
     retval=GURL::Filename::UTF8(xprogramname);
Packit df99a1
   }
Packit df99a1
   if(retval.is_empty() || !retval.is_file())
Packit df99a1
   {
Packit df99a1
     GList<GURL> paths(parsePATH());
Packit df99a1
     GMap<GUTF8String,void *> pathMAP;
Packit df99a1
     for(GPosition pos=paths;pos;++pos)
Packit df99a1
     {
Packit df99a1
       retval=GURL::UTF8(xprogramname,paths[pos]);
Packit df99a1
       const GUTF8String path(retval.get_string());
Packit df99a1
       if(!pathMAP.contains(path))
Packit df99a1
       {
Packit df99a1
         if(retval.is_file())
Packit df99a1
           break;
Packit df99a1
         pathMAP[path]=0;
Packit df99a1
       }
Packit df99a1
     }
Packit df99a1
   }
Packit df99a1
   if (! retval.is_empty() )
Packit df99a1
     retval = retval.follow_symlinks();
Packit df99a1
   if (! retval.is_empty() )
Packit df99a1
     retval = retval.base();
Packit df99a1
 }
Packit df99a1
 return retval;
Packit df99a1
}
Packit df99a1
#endif
Packit df99a1
Packit df99a1
static void
Packit df99a1
appendPath(const GURL &url, 
Packit df99a1
           GMap<GUTF8String,void *> &map,
Packit df99a1
           GList<GURL> &list)
Packit df99a1
{
Packit df99a1
  if( !url.is_empty() && !map.contains(url.get_string()) )
Packit df99a1
    {
Packit df99a1
      map[url.get_string()]=0;
Packit df99a1
      list.append(url);
Packit df99a1
    }
Packit df99a1
}
Packit df99a1
Packit df99a1
GList<GURL>
Packit df99a1
DjVuMessage::GetProfilePaths(void)
Packit df99a1
{
Packit df99a1
  static bool first=true;
Packit df99a1
  static GList<GURL> realpaths;
Packit df99a1
  if(first)
Packit df99a1
  {
Packit df99a1
    first=false;
Packit df99a1
    GMap<GUTF8String,void *> pathsmap;
Packit df99a1
    GList<GURL> paths;
Packit df99a1
    GURL path;
Packit df99a1
    const GUTF8String envp(GOS::getenv(DjVuEnv));
Packit df99a1
    if(envp.length())
Packit df99a1
      appendPath(GURL::Filename::UTF8(envp),pathsmap,paths);
Packit df99a1
#if defined(_WIN32) || defined(UNIX)
Packit df99a1
    GURL mpath(GetModulePath());
Packit df99a1
    if(!mpath.is_empty() && mpath.is_dir())
Packit df99a1
    {
Packit df99a1
#if defined(UNIX) && !defined(AUTOCONF) && !defined(NDEBUG)
Packit df99a1
      appendPath(GURL::UTF8(DebugModuleDjVuDir,mpath),pathsmap,paths);
Packit df99a1
#endif
Packit df99a1
      appendPath(mpath,pathsmap,paths);
Packit df99a1
      appendPath(GURL::UTF8(ModuleDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      appendPath(GURL::UTF8(ProfilesDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      mpath=mpath.base();
Packit df99a1
      appendPath(GURL::UTF8(ModuleDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      appendPath(GURL::UTF8(ProfilesDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      mpath=mpath.base();
Packit df99a1
      appendPath(GURL::UTF8(ModuleDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      appendPath(GURL::UTF8(ProfilesDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      mpath=mpath.base();
Packit df99a1
      appendPath(GURL::UTF8(ModuleDjVuDir,mpath),pathsmap,paths);
Packit df99a1
      appendPath(GURL::UTF8(ProfilesDjVuDir,mpath),pathsmap,paths);
Packit df99a1
    }
Packit df99a1
#endif
Packit df99a1
#if defined(AUTOCONF)
Packit df99a1
    GURL dpath = GURL::Filename::UTF8(DjVuDataDir);
Packit df99a1
    appendPath(dpath,pathsmap,paths);
Packit df99a1
#endif
Packit df99a1
#ifdef _WIN32
Packit df99a1
    appendPath(RegOpenReadConfig(HKEY_CURRENT_USER),pathsmap,paths);
Packit df99a1
    appendPath(RegOpenReadConfig(HKEY_LOCAL_MACHINE),pathsmap,paths);
Packit df99a1
#else
Packit df99a1
    GUTF8String home=GOS::getenv("HOME");
Packit df99a1
# if HAVE_GETPWUID
Packit df99a1
    if (! home.length()) {
Packit df99a1
      struct passwd *pw=0;
Packit df99a1
      if ((pw = getpwuid(getuid())))
Packit df99a1
        home=GNativeString(pw->pw_dir);
Packit df99a1
    }
Packit df99a1
# endif
Packit df99a1
    if (home.length()) {
Packit df99a1
      GURL hpath = GURL::UTF8(LocalDjVuDir,GURL::Filename::UTF8(home));
Packit df99a1
      appendPath(hpath,pathsmap,paths);
Packit df99a1
    }
Packit df99a1
#endif
Packit df99a1
#ifdef LT_DEFAULT_PREFIX
Packit df99a1
    appendPath(GURL::Filename::UTF8(DjVuPrefixDir),pathsmap,paths);
Packit df99a1
#endif
Packit df99a1
    appendPath(GURL::Filename::UTF8(RootDjVuDir),pathsmap,paths);
Packit df99a1
    pathsmap.empty();
Packit df99a1
Packit df99a1
    GPosition pos;
Packit df99a1
    GList< GMap<GUTF8String,GP<lt_XMLTags> > > localemaps;
Packit df99a1
    for(pos=paths;pos;++pos)
Packit df99a1
    {
Packit df99a1
      path=GURL::UTF8(LanguageFile,paths[pos]);
Packit df99a1
      if(path.is_file())
Packit df99a1
      {
Packit df99a1
        const GP<lt_XMLTags> xml(lt_XMLTags::create(ByteStream::create(path,"rb")));
Packit df99a1
        const GPList<lt_XMLTags> Body(xml->get_Tags(bodystring));
Packit df99a1
        GPosition pos=Body;
Packit df99a1
        if(!pos || (pos != Body.lastpos()))
Packit df99a1
        {
Packit df99a1
          G_THROW( ERR_MSG("XMLAnno.extra_body") );
Packit df99a1
        }
Packit df99a1
        const GP<lt_XMLTags> GBody(Body[pos]);
Packit df99a1
        if(!GBody)
Packit df99a1
        {
Packit df99a1
          G_THROW( ERR_MSG("XMLAnno.no_body") );
Packit df99a1
        }
Packit df99a1
        GMap<GUTF8String,GP<lt_XMLTags> > localemap;
Packit df99a1
        lt_XMLTags::get_Maps(languagestring,localestring,Body,localemap);
Packit df99a1
        localemaps.append(localemap);
Packit df99a1
      }
Packit df99a1
    } 
Packit df99a1
    GList<GURL> localepaths;
Packit df99a1
Packit df99a1
    // Need to do it the right way!
Packit df99a1
    GUTF8String defaultlocale = getenv("LANGUAGE");
Packit df99a1
    if (! defaultlocale) 
Packit df99a1
      {
Packit df99a1
        const GUTF8String oldlocale(setlocale(LC_MESSAGES,0));
Packit df99a1
        defaultlocale = setlocale(LC_MESSAGES,"");
Packit df99a1
        setlocale(LC_MESSAGES,(const char *)oldlocale);
Packit df99a1
      }
Packit df99a1
    // Unfathomable search.
Packit df99a1
    for(int loop=0; loop<2; loop++)
Packit df99a1
    {
Packit df99a1
      static const char sepchars[]=" _.@";
Packit df99a1
      const char *p=sepchars+sizeof(sepchars)-1;
Packit df99a1
      do
Packit df99a1
      {
Packit df99a1
        int sepcharpos=p[0]?defaultlocale.search(p[0]):defaultlocale.length();
Packit df99a1
        if(sepcharpos > 0)
Packit df99a1
        {
Packit df99a1
          const GUTF8String sublocale(defaultlocale,sepcharpos);
Packit df99a1
          const GUTF8String downcasesublocale("downcase^"+sublocale.downcase());
Packit df99a1
          for(pos=localemaps;pos;++pos) 
Packit df99a1
          {
Packit df99a1
            const GMap<GUTF8String,GP<lt_XMLTags> > &localemap=localemaps[pos];
Packit df99a1
            GPosition pos=localemap.contains(sublocale);
Packit df99a1
            if(!pos)
Packit df99a1
              pos=localemap.contains(downcasesublocale);
Packit df99a1
            if(pos)
Packit df99a1
            {
Packit df99a1
              const GMap<GUTF8String,GUTF8String>&args
Packit df99a1
                = localemap[pos]->get_args();
Packit df99a1
              pos = args.contains(srcstring);
Packit df99a1
              if (pos)
Packit df99a1
              {
Packit df99a1
                const GUTF8String src(args[pos]);
Packit df99a1
                for(pos=paths;pos;++pos)
Packit df99a1
                {
Packit df99a1
                  path=GURL::UTF8(src,paths[pos]);
Packit df99a1
                  if(path.is_dir())
Packit df99a1
                    localepaths.append(path);
Packit df99a1
                }
Packit df99a1
              }
Packit df99a1
              // We don't need to check anymore language files.
Packit df99a1
              p=sepchars;
Packit df99a1
              break;
Packit df99a1
            }
Packit df99a1
          }
Packit df99a1
          if(!pos)
Packit df99a1
            {
Packit df99a1
            for(pos=paths;pos;++pos)
Packit df99a1
              {
Packit df99a1
              path=GURL::UTF8(sublocale,paths[pos]);
Packit df99a1
              if(path.is_dir())
Packit df99a1
                localepaths.append(path);
Packit df99a1
            }
Packit df99a1
          }
Packit df99a1
        }
Packit df99a1
      } while(p-- != sepchars);
Packit df99a1
      if((GPosition) localepaths)
Packit df99a1
        break;
Packit df99a1
      defaultlocale="C";
Packit df99a1
    }
Packit df99a1
    for(pos=localepaths;pos;++pos)
Packit df99a1
      appendPath(localepaths[pos],pathsmap,realpaths);
Packit df99a1
    for(pos=paths;pos;++pos)
Packit df99a1
      appendPath(paths[pos],pathsmap,realpaths);
Packit df99a1
  }
Packit df99a1
  return realpaths;
Packit df99a1
}
Packit df99a1
Packit df99a1
static GUTF8String
Packit df99a1
getbodies(
Packit df99a1
  GList<GURL> &paths,
Packit df99a1
  const GUTF8String &MessageFileName,
Packit df99a1
  GPList<lt_XMLTags> &body, 
Packit df99a1
  GMap<GUTF8String, void *> & map )
Packit df99a1
{
Packit df99a1
  GUTF8String errors;
Packit df99a1
  bool isdone=false;
Packit df99a1
  GPosition firstpathpos=paths;
Packit df99a1
  for(GPosition pathpos=firstpathpos;!isdone && pathpos;++pathpos)
Packit df99a1
  {
Packit df99a1
    const GURL::UTF8 url(MessageFileName,paths[pathpos]);
Packit df99a1
    if(url.is_file())
Packit df99a1
    {
Packit df99a1
      map[MessageFileName]=0;
Packit df99a1
      GP<lt_XMLTags> gtags;
Packit df99a1
      {
Packit df99a1
        GP<ByteStream> bs=ByteStream::create(url,"rb");
Packit df99a1
        G_TRY
Packit df99a1
        {
Packit df99a1
          gtags=lt_XMLTags::create(bs);
Packit df99a1
        }
Packit df99a1
        G_CATCH(ex)
Packit df99a1
        {
Packit df99a1
          GUTF8String mesg(failed_to_parse_XML+("\t"+url.get_string()));
Packit df99a1
          if(errors.length())
Packit df99a1
          {
Packit df99a1
            errors+="\n"+mesg;
Packit df99a1
          }else
Packit df99a1
          {
Packit df99a1
            errors=mesg;
Packit df99a1
          }
Packit df99a1
          errors+="\n"+GUTF8String(ex.get_cause());
Packit df99a1
        }
Packit df99a1
        G_ENDCATCH;
Packit df99a1
      }
Packit df99a1
      if(gtags)
Packit df99a1
      {
Packit df99a1
        lt_XMLTags &tags=*gtags;
Packit df99a1
        GPList<lt_XMLTags> Bodies=tags.get_Tags(bodystring);
Packit df99a1
        if(! Bodies.isempty())
Packit df99a1
        {
Packit df99a1
          isdone=true;
Packit df99a1
          for(GPosition pos=Bodies;pos;++pos)
Packit df99a1
          {
Packit df99a1
            body.append(Bodies[pos]);
Packit df99a1
          }
Packit df99a1
        }
Packit df99a1
        GPList<lt_XMLTags> Head=tags.get_Tags(headstring);
Packit df99a1
        if(! Head.isempty())
Packit df99a1
        {
Packit df99a1
          isdone=true;
Packit df99a1
          GMap<GUTF8String, GP<lt_XMLTags> > includes;
Packit df99a1
          lt_XMLTags::get_Maps(includestring,namestring,Head,includes);
Packit df99a1
          for(GPosition pos=includes;pos;++pos)
Packit df99a1
          {
Packit df99a1
            const GUTF8String file=includes.key(pos);
Packit df99a1
            if(! map.contains(file))
Packit df99a1
            {
Packit df99a1
              GList<GURL> xpaths;
Packit df99a1
              xpaths.append(url.base());
Packit df99a1
              const GUTF8String err2(getbodies(xpaths,file,body,map));
Packit df99a1
              if(err2.length())
Packit df99a1
              {
Packit df99a1
                if(errors.length())
Packit df99a1
                {
Packit df99a1
                  errors+="\n"+err2;
Packit df99a1
                }else
Packit df99a1
                {
Packit df99a1
                  errors=err2;
Packit df99a1
                }
Packit df99a1
              }
Packit df99a1
            }
Packit df99a1
          }
Packit df99a1
        }
Packit df99a1
      }
Packit df99a1
    }
Packit df99a1
  }
Packit df99a1
  return errors;
Packit df99a1
}
Packit df99a1
Packit df99a1
static GUTF8String
Packit df99a1
parse(GMap<GUTF8String,GP<lt_XMLTags> > &retval)
Packit df99a1
{
Packit df99a1
  GUTF8String errors;
Packit df99a1
  GPList<lt_XMLTags> body;
Packit df99a1
  {
Packit df99a1
    GList<GURL> paths=DjVuMessage::GetProfilePaths();
Packit df99a1
    GMap<GUTF8String, void *> map;
Packit df99a1
    GUTF8String m(MessageFile);
Packit df99a1
    errors=getbodies(paths,m,body,map);
Packit df99a1
  }
Packit df99a1
  if(! body.isempty())
Packit df99a1
  {
Packit df99a1
    lt_XMLTags::get_Maps(messagestring,namestring,body,retval);
Packit df99a1
  }
Packit df99a1
  return errors;
Packit df99a1
}
Packit df99a1
Packit df99a1
Packit df99a1
const DjVuMessageLite &
Packit df99a1
DjVuMessage::create_full(void)
Packit df99a1
{
Packit df99a1
  GP<DjVuMessageLite> &static_message=getDjVuMessageLite();
Packit df99a1
  if(!static_message)
Packit df99a1
  {
Packit df99a1
    DjVuMessage *mesg=new DjVuMessage;
Packit df99a1
    static_message=mesg;
Packit df99a1
    mesg->init();
Packit df99a1
  }
Packit df99a1
  return DjVuMessageLite::create_lite();
Packit df99a1
}
Packit df99a1
Packit df99a1
void
Packit df99a1
DjVuMessage::set_programname(const GUTF8String &xprogramname)
Packit df99a1
{
Packit df99a1
  programname()=xprogramname;
Packit df99a1
  DjVuMessageLite::create=create_full; 
Packit df99a1
}
Packit df99a1
Packit df99a1
void
Packit df99a1
DjVuMessage::use_language(void)
Packit df99a1
{ 
Packit df99a1
  DjVuMessageLite::create=create_full; 
Packit df99a1
}
Packit df99a1
Packit df99a1
Packit df99a1
// Constructor
Packit df99a1
DjVuMessage::DjVuMessage( void ) {}
Packit df99a1
Packit df99a1
void
Packit df99a1
DjVuMessage::init(void)
Packit df99a1
{
Packit df99a1
  errors=parse(Map);
Packit df99a1
}
Packit df99a1
Packit df99a1
// Destructor
Packit df99a1
DjVuMessage::~DjVuMessage( )
Packit df99a1
{
Packit df99a1
}
Packit df99a1
Packit df99a1
Packit df99a1
//  A C function to perform a message lookup. Arguments are a buffer to receiv
Packit df99a1
//  translated message, a buffer size (bytes), and a message_list. The transla
Packit df99a1
//  result is returned in msg_buffer encoded in Native MBS encoding. In case
Packit df99a1
// of error, msg_b empty (i.e., msg_buffer[0] == '\0').
Packit df99a1
void
Packit df99a1
DjVuMessageLookUpNative( 
Packit df99a1
  char *msg_buffer, const unsigned int buffer_size, const char *message)
Packit df99a1
{
Packit df99a1
  const GNativeString converted(DjVuMessage::LookUpNative( message ));
Packit df99a1
  if( converted.length() >= buffer_size )
Packit df99a1
    msg_buffer[0] = '\0';
Packit df99a1
  else
Packit df99a1
    strcpy( msg_buffer, converted );
Packit df99a1
}
Packit df99a1
Packit df99a1
//  A C function to perform a message lookup. Arguments are a buffer to receiv
Packit df99a1
//  translated message, a buffer size (bytes), and a message_list. The transla
Packit df99a1
//  result is returned in msg_buffer encoded in UTF8 encoding. In case
Packit df99a1
// of error, msg_b empty (i.e., msg_buffer[0] == '\0').
Packit df99a1
void
Packit df99a1
DjVuMessageLookUpUTF8( 
Packit df99a1
  char *msg_buffer, const unsigned int buffer_size, const char *message)
Packit df99a1
{
Packit df99a1
  const GUTF8String converted(DjVuMessage::LookUpUTF8( message ));
Packit df99a1
  if( converted.length() >= buffer_size )
Packit df99a1
    msg_buffer[0] = '\0';
Packit df99a1
  else
Packit df99a1
    strcpy( msg_buffer, converted );
Packit df99a1
}
Packit df99a1
Packit df99a1
Packit df99a1
Packit df99a1
#ifdef HAVE_NAMESPACES
Packit df99a1
}
Packit df99a1
# ifndef NOT_USING_DJVU_NAMESPACE
Packit df99a1
using namespace DJVU;
Packit df99a1
# endif
Packit df99a1
#endif
Packit df99a1
Packit df99a1
void
Packit df99a1
DjVuFormatErrorUTF8( const char *fmt, ... )
Packit df99a1
{
Packit df99a1
  va_list args;
Packit df99a1
  va_start(args, fmt); 
Packit df99a1
  const GUTF8String message(fmt,args);
Packit df99a1
  DjVuWriteError( message );
Packit df99a1
}
Packit df99a1
Packit df99a1
void
Packit df99a1
DjVuFormatErrorNative( const char *fmt, ... )
Packit df99a1
{
Packit df99a1
  va_list args;
Packit df99a1
  va_start(args, fmt); 
Packit df99a1
  const GNativeString message(fmt,args);
Packit df99a1
  DjVuWriteError( message );
Packit df99a1
}
Packit df99a1
Packit df99a1
const char *
Packit df99a1
djvu_programname(const char *xprogramname)
Packit df99a1
{
Packit df99a1
  if(xprogramname)
Packit df99a1
    DjVuMessage::programname()=GNativeString(xprogramname);
Packit df99a1
  return DjVuMessage::programname();
Packit df99a1
}