aboutsummaryrefslogblamecommitdiffstats
path: root/lib/ephy-langs.c
blob: ff0ecb981b2af2ab25c1d050e4ce2fa01b87f7d7 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
  
                                             













                                                                              

        
   
 
                   
 
                       
 

                       
                       
 

                   
                             
 


                                                                                          
                                                      
 

                                                                                 
                                                                               

                                                                                 
                                                                                     

                                                                                 
                                                                                           

                                                                                 
                                                                                       

                                                                                 
                                                                                         

                                                                                 
                                                                               

                                                                                 
                                                                               

                                                                                 
                                                                               

                                                                                 
                                                                               

                                                                                 
                                                                                  

                                                                                 
                                                                                    

                                                                                 
                                                                               

                                                                                 
                                                                                  

                                                                                 
                                                                                  

                                                                                 
                                                                               




























                                                                                   

                                                                                 
                                                                                      

                                                                                 
                                                                                     
  
 

                             
 
                              

 

                            
 
                                             
 






































                                                                              
                                                  
         
                                                          












                                                                          
                                                   
         
                                                          



















                                                                          


                   







































                                                                              
 

















































































                                                                                               
           









                                               
                                          



























































                                                                                             




                                           






























                                                                    
/*
 *  Copyright (C) 2003, 2004 Christian Persch
 *
 *  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, 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 *
 *  $Id$
 */

#include "config.h"

#include "ephy-langs.h"

#include "ephy-debug.h"

#include <glib/gi18n.h>

#include <string.h>

#include <libxml/xmlreader.h>

/* If you add more language codes here, remember also to add them in the default font size
 * section in data/default-prefs-common.js.
 */
static const EphyFontsLanguageInfo font_languages [] =
{
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Arabic"),                "ar" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Baltic"),                "x-baltic" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Central European"),          "x-central-euro" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Cyrillic"),              "x-cyrillic" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Devanagari"),                "x-devanagari" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Greek"),                 "el" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Hebrew"),                "he" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Japanese"),              "ja" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Korean"),                "ko" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Simplified Chinese"),            "zh-CN" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Tamil"),                 "x-tamil" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Thai"),                  "th" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Traditional Chinese"),           "zh-TW" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Traditional Chinese (Hong Kong)"),   "zh-HK" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Turkish"),               "tr" },
#ifdef HAVE_GECKO_1_8
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Armenian"),              "x-armn" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Bengali"),               "x-beng" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Unified Canadian Syllabics"),        "x-cans" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Ethiopic"),              "x-ethi" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Georgian"),              "x-geor" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Gujarati"),              "x-gujr" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Gurmukhi"),              "x-guru" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Khmer"),                 "x-khmr" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Malayalam"),             "x-mlym" },
#endif /* HAVE_GECKO_1_8 */
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Western"),               "x-western" },
    /* Translators: The text before the "|" is context to help you decide on
     * the correct translation. You MUST OMIT it in the translated string. */
    { N_("select fonts for|Other Scripts"),             "x-unicode" }
};

const EphyFontsLanguageInfo *
ephy_font_languages (void)
{
    return font_languages;
}

guint            
ephy_font_n_languages (void)
{
    return G_N_ELEMENTS (font_languages);
}

/* sanitise the languages list according to the rules for HTTP accept-language
 * in RFC 2616, Sect. 14.4
 */
void
ephy_langs_sanitise (GArray *array)
{
    char *lang1, *lang2;
    int i, j;

    /* if we have 'xy-ab' in list but not 'xy', append 'xy' */
    for (i = 0; i < array->len; i++)
    {
        gboolean found = FALSE;
        char *dash, *prefix;

        lang1 = (char *) g_array_index (array,char *, i);

        dash = strchr (lang1, '-');
        if (dash == NULL) continue;

        for (j = i + 1; j < array->len; j++)
        {
            lang2 = (char *) g_array_index (array, char *, j);
            if (strchr (lang2, '-') == NULL &&
                g_str_has_prefix (lang1, lang2))
            {
                found = TRUE;
            }
        }

        if (found == FALSE)
        {
            prefix = g_strndup (lang1, dash - lang1);
            g_array_append_val (array, prefix);
        }
    }

    /* uniquify */
    for (i = 0; i < (int) array->len - 1; i++)
    {
        for (j = (int) array->len - 1; j > i; j--)
        {
            lang1 = (char *) g_array_index (array,char *, i);
            lang2 = (char *) g_array_index (array, char *, j);

            if (strcmp (lang1, lang2) == 0)
            {
                g_array_remove_index (array, j);
                g_free (lang2);
            }
        }
    }

    /* move 'xy' code behind all 'xy-ab' codes */
    for (i = (int) array->len - 2; i >= 0; i--)
    {
        for (j = (int) array->len - 1; j > i; j--)
        {
            lang1 = (char *) g_array_index (array, char *, i);
            lang2 = (char *) g_array_index (array, char *, j);

            if (strchr (lang1, '-') == NULL &&
                strchr (lang2, '-') != NULL &&
                g_str_has_prefix (lang2, lang1))
            {
                g_array_insert_val (array, j + 1, lang1);
                g_array_remove_index (array, i);
                break;
            }
        }
    }
}

void
ephy_langs_append_languages (GArray *array)
{
    const char * const * languages;
    char *lang;
    int i;

    languages = g_get_language_names ();
    g_return_if_fail (languages != NULL);

    /* FIXME: maybe just use the first, instead of all of them? */
    for (i = 0; languages[i] != NULL; i++)
    {

        if (strstr (languages[i], ".") == 0 &&
            strstr (languages[i], "@") == 0 &&
            strcmp (languages[i], "C") != 0)
        {
            /* change to lowercase and '_' to '-' */
            lang = g_strdelimit (g_ascii_strdown
                        (languages[i], -1), "_", '-');

            g_array_append_val (array, lang);
        }
    }

    /* Fallback: add "en" if list is empty */
    if (array->len == 0)
    {
        lang = g_strdup ("en");
        g_array_append_val (array, lang);
    }
}

char **
ephy_langs_get_languages (void)
{
    GArray *array;

    array = g_array_new (TRUE, FALSE, sizeof (char *));

    ephy_langs_append_languages (array);

    ephy_langs_sanitise (array);

    return (char **) g_array_free (array, FALSE);
}

#define ISOCODESLOCALEDIR   ISO_CODES_PREFIX "/share/locale"

static void
ephy_langs_bind_iso_domains (void)
{
#ifdef ENABLE_NLS
    static gboolean bound = FALSE;

    if (bound == FALSE)
    {
            bindtextdomain (ISO_639_DOMAIN, ISOCODESLOCALEDIR);
            bind_textdomain_codeset (ISO_639_DOMAIN, "UTF-8");

            bindtextdomain(ISO_3166_DOMAIN, ISOCODESLOCALEDIR);
            bind_textdomain_codeset (ISO_3166_DOMAIN, "UTF-8");

        bound = TRUE;
    }
#endif
}

static void
read_iso_639_entry (xmlTextReaderPtr reader,
            GHashTable *table)
{
    xmlChar *code, *name;

    code = xmlTextReaderGetAttribute (reader, (const xmlChar *) "iso_639_1_code");
    name = xmlTextReaderGetAttribute (reader, (const xmlChar *) "name");

    /* Get iso-639-2 code */
    if (code == NULL || code[0] == '\0')
    {
        xmlFree (code);
        /* FIXME: use the 2T or 2B code? */
        code = xmlTextReaderGetAttribute (reader, (const xmlChar *) "iso_639_2T_code");
    }

    if (code != NULL && code[0] != '\0' && name != NULL && name[0] != '\0')
    {
        g_hash_table_insert (table, code, name);
    }
    else
    {
        xmlFree (code);
        xmlFree (name);
    }
}

static void
read_iso_3166_entry (xmlTextReaderPtr reader,
             GHashTable *table)
{
    xmlChar *code, *name;

    code = xmlTextReaderGetAttribute (reader, (const xmlChar *) "alpha_2_code");
    name = xmlTextReaderGetAttribute (reader, (const xmlChar *) "name");

    if (code != NULL && code[0] != '\0' && name != NULL && name[0] != '\0')
    {
        char *lcode;

        lcode = g_ascii_strdown ((char *) code, -1);
        xmlFree (code);

        g_hash_table_insert (table, lcode, name);
    }
    else
    {
        xmlFree (code);
        xmlFree (name);
    }

}

typedef enum
{
    STATE_START,
    STATE_STOP,
    STATE_ENTRIES,
} ParserState;

static void
load_iso_entries (int iso,
          GFunc read_entry_func,
          gpointer user_data)
{
    xmlTextReaderPtr reader;
    ParserState state = STATE_START;
    xmlChar iso_entries[32], iso_entry[32];
    char *filename;
    int ret = -1;

    LOG ("Loading ISO-%d codes", iso);

    START_PROFILER ("Loading ISO codes")

    filename = g_strdup_printf (ISO_CODES_PREFIX "/share/xml/iso-codes/iso_%d.xml", iso);
    reader = xmlNewTextReaderFilename (filename);
    if (reader == NULL) goto out;

    xmlStrPrintf (iso_entries, sizeof (iso_entries), "iso_%d_entries", iso);
    xmlStrPrintf (iso_entry, sizeof (iso_entry), "iso_%d_entry", iso);

    ret = xmlTextReaderRead (reader);

    while (ret == 1)
    {
        const xmlChar *tag;
        xmlReaderTypes type;

        tag = xmlTextReaderConstName (reader);
        type = xmlTextReaderNodeType (reader);

        if (state == STATE_ENTRIES &&
            type == XML_READER_TYPE_ELEMENT &&
            xmlStrEqual (tag, iso_entry))
        {
            read_entry_func (reader, user_data);
        }
        else if (state == STATE_START &&
             type == XML_READER_TYPE_ELEMENT &&
             xmlStrEqual (tag, iso_entries))
        {
            state = STATE_ENTRIES;
        }
        else if (state == STATE_ENTRIES &&
             type == XML_READER_TYPE_END_ELEMENT &&
             xmlStrEqual (tag, iso_entries))
        {
            state = STATE_STOP;
        }
        else if (type == XML_READER_TYPE_SIGNIFICANT_WHITESPACE ||
             type == XML_READER_TYPE_WHITESPACE ||
             type == XML_READER_TYPE_TEXT ||
             type == XML_READER_TYPE_COMMENT)
        {
            /* eat it */
        }
        else
        {
            /* ignore it */
        }

        ret = xmlTextReaderRead (reader);
    }

    xmlFreeTextReader (reader);

out:
    if (ret < 0 || state != STATE_STOP)
    {
        g_warning ("Failed to load ISO-%d codes from %s!\n",
               iso, filename);
    }

    g_free (filename);

    STOP_PROFILER ("Loading ISO codes")
}

GHashTable *
ephy_langs_iso_639_table (void)
{
    GHashTable *table;

    ephy_langs_bind_iso_domains ();
    table = g_hash_table_new_full (g_str_hash, g_str_equal,
                       (GDestroyNotify) xmlFree,
                       (GDestroyNotify) xmlFree);

    load_iso_entries (639, (GFunc) read_iso_639_entry, table);

    return table;
}

GHashTable *
ephy_langs_iso_3166_table (void)
{
    GHashTable *table;

    ephy_langs_bind_iso_domains ();
    table = g_hash_table_new_full (g_str_hash, g_str_equal,
                       (GDestroyNotify) g_free,
                       (GDestroyNotify) xmlFree);
    
    load_iso_entries (3166, (GFunc) read_iso_3166_entry, table);

    return table;
}