aboutsummaryrefslogblamecommitdiffstats
path: root/embed/mozilla/mozilla-notifiers.cpp
blob: fbd8dd766c8c0fa70858ce61e21933e3bdc77111 (plain) (tree)
1
2
3
4
  


                                              












                                                                       
                                                                                 


       
 
                         
                   
 


                   
 
                       

                            
 
                     
                           
                              
                     











                                  
 
                                                     
                          
 

                           

                                                                    

                                  
 
              

                        

                                    
                           




                               
 

                                    

                                      
 
 
               
                                                     

                                                    
 
                            



                              


                                                  
                                                                                    

















                                                                       
                                                         













                                                             


               
                                          

                                         
 




                                                           




                                                                      


               
                                                   


                                                  
 
                            


                             


                                                               
 

                                                
 











                                                    
 



                                              

 
               
                                        


                                       
                            




                                 


                                                              

















                                                                                                         



                                                    
 

                            







                                                              
                                           
         
                              
         
 

                                            
 
                    

 
               
                                                 

                                                
 
                            



                          


                                                  
                                                                                    









                                                                       
 









                                                               
 
                                              


                         

 
               
                                          

                                         
 
                            
                         
                             
 


                                                              



                                                








                                            

                                              
 
                    

 
               
                                             

                                            
 




                                                            







                                                                        
                                                     

                                                    
 




                                                            







                                                                    
                                                            

                                                           
 




                                                            







                                                                     
                                                    

                                                   
 




                                                           







                                                               
                                                       

                                                      
 
                            

                        


                                                              


























                                                   


                                            


                                           




                                            

                                    

                                            

                                   









































                                              


                                           

  

                                   

                                      

                                            


                                                    
                    








                                                        
                                     
         
                                
                 



                                                                    




                                                                        
                                                                               
                         
                              
                 
                                    
                 



                                                                    






                                                                                
                              
                 
                                   
                 



                                                                     





















                                                                                        
                              
                 



                                                        

         
                                             


           

                               
                               
                          
 
                              
 
                                           
                                
 
                                                          
         
                                                              
                                       
         





                                                                

 




                                                  
 

                            


                             


                                                       

                                                
                                                 
 
                                      


                                                     
                                    

                                                             

                                                                             
                                                   





                                                                     
 
                         

         

                                                     
 
                                                

                       



                             
                               
 
                                                      


               
                                  
 

                 
                                   
 

                                                                 

                                     
                                                      
                                           

 




                                     


                                                                        




                                                            
 



                                                                                   
                                                                                       










                                                               
 





                                                                            

                                              







                                                                 

                                                                          







                                                            







                                                                                             


                                                                          
         
                                                   

                             
                                


                                                                      




                                                                      









                                          






























































                                                                              
                                                            
























                                                                              
                                         

                                       



                              
 


                                         
















































































                                                                       
 
                                                                  
                                                    
                                  
 

                                                           




























                                                                                           
 










                                                                                     
                                            
         
                                                          
                        
                                         


                                  
 
                                                          
                 




                                                                           

                                                                                 
                                                        

                                                                     

                 
                         

                                 







                                                                                   
                                                                                  
                                                                             
                                                                          













                                                                                   



                                                                               




                                                                             
 


                                 
                                                                   
                                                                            
                                                              


                 


                                                                             



                                                                         
 


                                 
                                                                       
                                                                              
                                                              


                 


                                                                                



                                                                         



                                 
                                                                       
                                                                            
                                                              

                 


         
           

                                      

                                                              











                                                                        


                                                

 
























































































                                                                                             








                                                     

                                                                           
                                                                               





                                                                




                                                                                            













                                                                       
                                    
 
                                                      

 

                                 
 




                                           


                                                        
 

                                                                      
 
/*
*  Copyright © 2000 Nate Case
*  Copyright © 2000-2004 Marco Pesenti Gritti
*  Copyright © 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*  $Id$
*/

#include <xpcom-config.h>
#include "config.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include <glib/gi18n.h>
#include <gdk/gdkx.h>
#include <gtk/gtksettings.h>

#include <nsCOMPtr.h>
#include <nsIPrefService.h>
#include <nsIServiceManager.h>
#include <nsMemory.h>
#include <nsServiceManagerUtils.h>

#include "eel-gconf-extensions.h"
#include "ephy-debug.h"
#include "ephy-embed-prefs.h"
#include "ephy-embed-shell.h"
#include "ephy-encodings.h"
#include "ephy-langs.h"
#include "ephy-node.h"
#include "ephy-prefs.h"

#include "mozilla-notifiers.h"

/* define to migrate epiphany 1.0 font preferences */
#define MIGRATE_PIXEL_SIZE

#define MAX_FONT_SIZE   128

/* Keeps the list of the notifiers we installed for mozilla prefs */
/* to be able to remove them when exiting */
static GList *notifiers = NULL;
static nsIPrefBranch *gPrefBranch;

typedef struct
{
    char *gconf_key;
    char *mozilla_pref;
    PrefValueTransformFunc func;
    gpointer user_data;
    guint cnxn_id;
} PrefData;

static void
free_pref_data (PrefData *data)
{
    g_free (data->gconf_key);
    g_free (data->mozilla_pref);

    g_slice_free (PrefData, data);
}

static gboolean
transform_accept_languages_list (GConfEntry *gcentry,
                 GValue *value,
                 gpointer user_data)
{
    GConfValue *gcvalue;
    GArray *array;
    GSList *languages, *l;
    char **langs;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_LIST ||
        gconf_value_get_list_type (gcvalue) != GCONF_VALUE_STRING) return FALSE;

    languages = gconf_value_get_list (gcvalue);

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

    for (l = languages; l != NULL; l = l->next)
    {
        const char *lang;

        lang = gconf_value_get_string ((GConfValue *) l->data);

        if (lang != NULL && strcmp (lang, "system") == 0)
        {
            ephy_langs_append_languages (array);
        }
        else if (lang != NULL && lang[0] != '\0')
        {
            char *str;
            str = g_ascii_strdown (lang, -1);
            g_array_append_val (array, str);
        }
    }

    ephy_langs_sanitise (array);

    langs = (char **) g_array_free (array, FALSE);

    g_value_init (value, G_TYPE_STRING);
    g_value_take_string (value, g_strjoinv (",", langs));

    g_strfreev (langs);

    return TRUE;
}

static gboolean
transform_cache_size (GConfEntry *gcentry,
              GValue *value,
              gpointer user_data)
{
    GConfValue *gcvalue;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_INT) return FALSE;

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, gconf_value_get_int (gcvalue) * 1024);

    return TRUE;
}

static gboolean
transform_cookies_accept_mode (GConfEntry *gcentry,
                   GValue *value,
                   gpointer user_data)

{
    GConfValue *gcvalue;
    const char *mode;
    int mozilla_mode = 0;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
         gcvalue->type != GCONF_VALUE_STRING) return FALSE;

    mode = gconf_value_get_string (gcvalue);
    if (mode == NULL) return FALSE;

    if (strcmp (mode, "anywhere") == 0)
    {
        mozilla_mode = 0;
    }
    else if (strcmp (mode, "current site") == 0)
    {
        mozilla_mode = 1;
    }
    else if (strcmp (mode, "nowhere") == 0)
    {
        mozilla_mode = 2;
    }

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, mozilla_mode);

    return TRUE;
}

static gboolean
transform_encoding (GConfEntry *gcentry,
            GValue *value,
            gpointer user_data)
{
    GConfValue *gcvalue;
    EphyEncodings *encodings;
    EphyNode *node;
    const char *code;
    gboolean is_autodetector;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_STRING) return FALSE;

    code = gconf_value_get_string (gcvalue);
    if (code == NULL) return FALSE;

    encodings = EPHY_ENCODINGS (ephy_embed_shell_get_encodings (embed_shell));
    node = ephy_encodings_get_node (encodings, code, FALSE);
    if (node == NULL) return FALSE;

    is_autodetector = ephy_node_get_property_boolean (node, EPHY_NODE_ENCODING_PROP_IS_AUTODETECTOR);
    if (is_autodetector != GPOINTER_TO_INT (user_data)) return FALSE;

    g_value_init (value, G_TYPE_STRING);
    g_value_set_string (value, code);

    return TRUE;
}

static gboolean
transform_image_animation_mode (GConfEntry *gcentry,
                    GValue *value,
                    gpointer user_data)

{
    GConfValue *gcvalue;
    const char *mode;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_STRING) return FALSE;

    mode = gconf_value_get_string (gcvalue);
    if (mode == NULL) return FALSE;

    if (strcmp (mode, "disabled") == 0)
    {
        mode = "none";
    }

    g_value_init (value, G_TYPE_STRING);
    g_value_set_string (value, mode);

    return TRUE;
}

static gboolean
transform_proxy_ignore_list (GConfEntry *gcentry,
                 GValue *value,
                 gpointer user_data)
{
    GConfValue *gcvalue;
    GArray *array;
    GSList *hosts, *l;
    char **strings;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_LIST ||
        gconf_value_get_list_type (gcvalue) != GCONF_VALUE_STRING) return FALSE;

    hosts = gconf_value_get_list (gcvalue);

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

    for (l = hosts; l != NULL; l = l->next)
    {
        const char *host;

        host = gconf_value_get_string ((GConfValue *) l->data);

        if (host != NULL && host[0] != '\0')
        {
            g_array_append_val (array, host);
        }
    }

    strings = (char **) g_array_free (array, FALSE);

    g_value_init (value, G_TYPE_STRING);
    g_value_take_string (value, g_strjoinv (",", strings));

    /* the strings themselves are const */
    g_free (strings);

    return TRUE;
}

static gboolean
transform_proxy_mode (GConfEntry *gcentry,
              GValue *value,
              gpointer user_data)
{
    GConfValue *gcvalue;
    const char *mode;
    int mozilla_mode = 0;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_STRING) return FALSE;

    mode = gconf_value_get_string (gcvalue);
    if (mode == NULL) return FALSE;

    if (strcmp (mode, "manual") == 0)
    {
        mozilla_mode = 1;
    }
    else if (strcmp (mode, "auto") == 0)
    {
        mozilla_mode = 2;
    }

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, mozilla_mode);

    return TRUE;
}

static gboolean
transform_use_own_fonts (GConfEntry *gcentry,
             GValue *value,
             gpointer user_data)
{
    GConfValue *gcvalue;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_BOOL) return FALSE;

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, gconf_value_get_bool (gcvalue) ? 0 : 1);

    return TRUE;
}

extern "C" gboolean
mozilla_notifier_transform_bool (GConfEntry *gcentry,
                 GValue *value,
                 gpointer user_data)
{
    GConfValue *gcvalue;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_BOOL) return FALSE;

    g_value_init (value, G_TYPE_BOOLEAN);
    g_value_set_boolean (value, gconf_value_get_bool (gcvalue));

    return TRUE;
}

extern "C" gboolean
mozilla_notifier_transform_bool_invert (GConfEntry *gcentry,
                    GValue *value,
                    gpointer user_data)
{
    GConfValue *gcvalue;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_BOOL) return FALSE;

    g_value_init (value, G_TYPE_BOOLEAN);
    g_value_set_boolean (value, !gconf_value_get_bool (gcvalue));

    return TRUE;
}

extern "C" gboolean
mozilla_notifier_transform_int (GConfEntry *gcentry,
                GValue *value,
                gpointer user_data)
{
    GConfValue *gcvalue;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_INT) return FALSE;

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, gconf_value_get_int (gcvalue));

    return TRUE;
}

extern "C" gboolean
mozilla_notifier_transform_string (GConfEntry *gcentry,
                   GValue *value,
                   gpointer user_data)
{
    GConfValue *gcvalue;
    const char *str;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_STRING) return FALSE;

    str = gconf_value_get_string (gcvalue);
    if (str == NULL) return FALSE;

    g_value_init (value, G_TYPE_STRING);
    g_value_set_string (value, str);

    return TRUE;
}

static const PrefData notifier_entries[] =
{
    { CONF_BROWSE_WITH_CARET,
      "accessibility.browsewithcaret",
      mozilla_notifier_transform_bool },
    { CONF_NETWORK_CACHE_SIZE,
      "browser.cache.disk.capacity",
      transform_cache_size },
    { CONF_RENDERING_USE_OWN_COLORS,
      "browser.display.use_document_colors",
      mozilla_notifier_transform_bool_invert },
    { CONF_RENDERING_USE_OWN_FONTS,
      "browser.display.use_document_fonts",
      transform_use_own_fonts },
    { CONF_SECURITY_ALLOW_POPUPS,
      "dom.disable_open_during_load",
      mozilla_notifier_transform_bool_invert },
    { CONF_DISPLAY_SMOOTHSCROLL,
      "general.smoothScroll",
      mozilla_notifier_transform_bool },
    { CONF_IMAGE_ANIMATION_MODE,
      "image.animation_mode",
      transform_image_animation_mode },
    { CONF_RENDERING_LANGUAGE,
      "intl.accept_languages",
      transform_accept_languages_list },
    { CONF_LANGUAGE_DEFAULT_ENCODING,
      "intl.charset.default",
      transform_encoding,
      GINT_TO_POINTER (FALSE) },
    { CONF_LANGUAGE_AUTODETECT_ENCODING,
      "intl.charset.detector",
      transform_encoding,
      GINT_TO_POINTER (TRUE) },
    { CONF_SECURITY_JAVA_ENABLED,
      "security.enable_java",
      mozilla_notifier_transform_bool },
    { CONF_SECURITY_JAVASCRIPT_ENABLED,
      "javascript.enabled",
      mozilla_notifier_transform_bool },
    { CONF_NETWORK_PROXY_AUTO_URL,
      "network.proxy.autoconfig_url",
      mozilla_notifier_transform_string },
    { CONF_NETWORK_HTTP_PROXY,
      "network.proxy.http",
      mozilla_notifier_transform_string },
    { CONF_NETWORK_HTTP_PROXY_PORT,
      "network.proxy.http_port",
      mozilla_notifier_transform_int },
    { CONF_NETWORK_FTP_PROXY,
      "network.proxy.ftp",
      mozilla_notifier_transform_string },
    { CONF_NETWORK_FTP_PROXY_PORT,
      "network.proxy.ftp_port",
      mozilla_notifier_transform_int },
    { CONF_NETWORK_SSL_PROXY,
      "network.proxy.ssl",
      mozilla_notifier_transform_string },
    { CONF_NETWORK_SSL_PROXY_PORT,
      "network.proxy.ssl_port",
      mozilla_notifier_transform_int },
    { CONF_NETWORK_SOCKS_PROXY,
      "network.proxy.socks",
      mozilla_notifier_transform_string },
    { CONF_NETWORK_SOCKS_PROXY_PORT,
      "network.proxy.socks_port",
      mozilla_notifier_transform_int },
    { CONF_NETWORK_PROXY_IGNORE_HOSTS,
      "network.proxy.no_proxies_on",
      transform_proxy_ignore_list },
    { CONF_NETWORK_PROXY_MODE,
      "network.proxy.type",
      transform_proxy_mode },
    { CONF_SECURITY_COOKIES_ACCEPT,
      "network.cookie.cookieBehavior",
      transform_cookies_accept_mode },
    { CONF_PRIVACY_REMEMBER_PASSWORDS,
      "signon.rememberSignons",
      mozilla_notifier_transform_bool }
};

gboolean
mozilla_pref_set (const char *pref,
          const GValue *value)
{
    NS_ENSURE_TRUE (gPrefBranch, FALSE);

    g_return_val_if_fail (pref != NULL, FALSE);
    g_return_val_if_fail (value != NULL, FALSE);

    nsresult rv;
        PRInt32 prefType;
        rv = gPrefBranch->GetPrefType (pref, &prefType);
        if (NS_FAILED (rv))
        {
                /* probably just an unset pref */
                prefType = nsIPrefBranch::PREF_INVALID;
                rv = NS_OK;
        }

    switch (G_VALUE_TYPE (value))
    {
        case G_TYPE_INT:
        {
                        if (prefType != nsIPrefBranch::PREF_INT &&
                            prefType != nsIPrefBranch::PREF_INVALID)
                          break;

            PRInt32 old_value = 0;
            PRInt32 new_value = g_value_get_int (value);
            rv = gPrefBranch->GetIntPref (pref, &old_value);
            if (NS_FAILED (rv) || old_value != new_value)
            {
                rv = gPrefBranch->SetIntPref (pref, new_value);
            }
            break;
        }
        case G_TYPE_BOOLEAN:
        {
                        if (prefType != nsIPrefBranch::PREF_BOOL &&
                            prefType != nsIPrefBranch::PREF_INVALID)
                          break;

            PRBool old_value = PR_FALSE;
            PRBool new_value = g_value_get_boolean (value);
            rv = gPrefBranch->GetBoolPref (pref, &old_value);
            if (NS_FAILED (rv) || old_value != new_value)
            {
                rv = gPrefBranch->SetBoolPref (pref, new_value);
            }
            break;
        }
        case G_TYPE_STRING:
        {
                        if (prefType != nsIPrefBranch::PREF_STRING &&
                            prefType != nsIPrefBranch::PREF_INVALID)
                          break;

            const char *new_value = g_value_get_string (value);
            if (new_value == NULL)
            {
                rv = gPrefBranch->ClearUserPref (pref);
            }
            else
            {
                char *old_value = nsnull;

                rv = gPrefBranch->GetCharPref (pref, &old_value);
                if (NS_FAILED (rv) ||
                    old_value == nsnull ||
                    strcmp (old_value, new_value) != 0)
                {
                    rv = gPrefBranch->SetCharPref (pref, new_value);
                }

                if (old_value)
                {
                    nsMemory::Free (old_value);
                }
            }
            break;
        }
        default:
            g_return_val_if_reached (FALSE);
            rv = NS_ERROR_FAILURE;
            break;
    }

    return NS_SUCCEEDED (rv) != PR_FALSE;
}

static void
notify_cb (GConfClient *client,
       guint cnxn_id,
       GConfEntry *gcentry,
       PrefData *data)
{
    GValue value = { 0, };

    g_return_if_fail (gcentry != NULL);
    g_assert (data != NULL);

    if (data->func (gcentry, &value, data->user_data))
    {
        mozilla_pref_set (data->mozilla_pref, &value);
        g_value_unset (&value);
    }
    else
    {
        /* Reset the pref */
        NS_ENSURE_TRUE (gPrefBranch, );
        gPrefBranch->ClearUserPref (data->mozilla_pref);
    }
}

extern "C" guint
mozilla_notifier_add (const char *gconf_key,
              const char *mozilla_pref,
              PrefValueTransformFunc func,
              gpointer user_data)
{
    GConfClient *client;
    PrefData *data;
    GError *error = NULL;
    guint cnxn_id;

    g_return_val_if_fail (gconf_key != NULL, 0);
    g_return_val_if_fail (mozilla_pref != NULL, 0);
    g_return_val_if_fail (func, 0);

    client = eel_gconf_client_get_global ();
    g_return_val_if_fail (client != NULL, 0);

    data = g_slice_new (PrefData);
    data->gconf_key = g_strdup (gconf_key);
    data->mozilla_pref = g_strdup (mozilla_pref);
    data->func = func;
    data->user_data = user_data;

    cnxn_id = gconf_client_notify_add (client, gconf_key,
                       (GConfClientNotifyFunc) notify_cb,
                       data, (GFreeFunc) free_pref_data,
                       &error);
    if (eel_gconf_handle_error (&error))
    {
        if (cnxn_id != EEL_GCONF_UNDEFINED_CONNECTION)
        {
            gconf_client_notify_remove (client, cnxn_id);
        }

        return 0;
    }

    data->cnxn_id = cnxn_id;
    notifiers = g_list_prepend (notifiers, data);

    gconf_client_notify (client, gconf_key);

    return cnxn_id;
}

static int
find_data (const PrefData *a,
       gconstpointer idptr)
{
    return a->cnxn_id != GPOINTER_TO_UINT (idptr);
}

extern "C" void
mozilla_notifier_remove (guint id)
{
    GList *l;

    g_return_if_fail (id != 0);

    l = g_list_find_custom (notifiers, GUINT_TO_POINTER (id),
                (GCompareFunc) find_data);
    g_return_if_fail (l != NULL);

    notifiers = g_list_delete_link (notifiers, l);
    eel_gconf_notification_remove (id);
}

#ifdef MIGRATE_PIXEL_SIZE

#define INT_ROUND(a) gint((a) + 0.5f)

/**
*  This function gets the dpi in the same way that mozilla gets the dpi,
*  this allows us to convert from pixels to points easily
*/
static gint
mozilla_get_dpi ()
{
    GtkSettings* settings = gtk_settings_get_default ();
    gint dpi = 0;

    /* Use the gdk-xft-dpi setting if it is set */
    if (g_object_class_find_property (G_OBJECT_GET_CLASS (G_OBJECT (settings)),
                      "gtk-xft-dpi"))
    {
        g_object_get (G_OBJECT (settings), "gtk-xft-dpi", &dpi, (char *) NULL);
        if (dpi) return INT_ROUND (dpi / PANGO_SCALE);
    }

    /* Fall back to what xft thinks it is */
    char *val = XGetDefault (GDK_DISPLAY (), "Xft", "dpi");
    if (val)
    {
        char *e;
        double d = strtod(val, &e);
        if (e != val) return INT_ROUND (d);
    }

    /* Fall back to calculating manually from the gdk screen settings */
    float screenWidthIn = float (gdk_screen_width_mm()) / 25.4f;
    return INT_ROUND (gdk_screen_width() / screenWidthIn);
}

static void
migrate_font_gconf_key (const char *pixel_key,
            const char *point_key)
{
    int size;

    size = eel_gconf_get_integer (pixel_key);

    if (size > 0)
    {
        /* Use doubles to get more accurate arithmetic */
        double dpi   = (double) mozilla_get_dpi ();
        double value = (double) eel_gconf_get_integer (pixel_key);
        gint point = INT_ROUND ((value * 72) / dpi);

        eel_gconf_set_integer (point_key, point);
    }
}

#endif

static gboolean
parse_pango_font (const char *font,
          char **name,
          int *size)
{
    PangoFontMask mask = (PangoFontMask) (PANGO_FONT_MASK_FAMILY | PANGO_FONT_MASK_SIZE);

    PangoFontDescription *desc = pango_font_description_from_string (font);
    if (desc == NULL) return FALSE;

    if ((pango_font_description_get_set_fields (desc) & mask) != mask)
    {
        pango_font_description_free (desc);
        return FALSE;
    }
                
    *size = PANGO_PIXELS (pango_font_description_get_size (desc));
    *name = g_strdup (pango_font_description_get_family (desc));

    pango_font_description_free (desc);

    LOG ("Parsed pango font description '%s' -> name:%s, size:%d",
         font, *name ? *name : "-", *size);

    return *name != NULL && *size > 0;
}

typedef struct
{
    guint cnxn_id;
    char **font_name_prefs;
    char **font_size_prefs;
    char *font_name;
    int font_size;
} DesktopFontData;

static gboolean
transform_font_name (GConfEntry *gcentry,
             GValue *value,
             gpointer user_data)
{
    DesktopFontData *data = (DesktopFontData *) user_data;
    GConfValue *gcvalue;
    const char *font_name;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gconf_entry_get_is_default (gcentry) ||
        gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_STRING)
    {
        font_name = data->font_name;
    }
    else
    {
        font_name = gconf_value_get_string (gcvalue);
        if ((font_name == NULL || font_name[0] == '\0'))
        {
            font_name = data->font_name;
        }
    }

    LOG ("%s value for key '%s'",
         gconf_entry_get_is_default (gcentry) ? "default" : "NON-default",
         gconf_entry_get_key (gcentry));

    if (font_name == NULL) return FALSE;

    LOG ("Inferred font name '%s' for key '%s'",
         font_name, gconf_entry_get_key (gcentry));

    g_value_init (value, G_TYPE_STRING);
    g_value_set_string (value, font_name);

    return TRUE;
}

static gboolean
transform_font_size (GConfEntry *gcentry,
             GValue *value,
             gpointer user_data)
{
    DesktopFontData *data = (DesktopFontData *) user_data;
    GConfValue *gcvalue;
    int size = 0;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gconf_entry_get_is_default (gcentry) ||
        gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_INT)
    {
        size = data->font_size;
    }
    else
    {
        size = gconf_value_get_int (gcvalue);
    }

    if (size <= 0 || size > MAX_FONT_SIZE) return FALSE;

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, size);

    return TRUE;
}

static void
notify_desktop_font_cb (GConfClient *client,
            guint cnxn_id,
            GConfEntry *gcentry,
            DesktopFontData *data)
{
    GConfValue *gcvalue;
    char *name = NULL;
    int size = 0, i;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue != NULL &&
        gcvalue->type == GCONF_VALUE_STRING &&
        parse_pango_font (gconf_value_get_string (gcvalue), &name, &size))
    {
        LOG ("Desktop font %s -> name=%s, size=%d", 
             gconf_entry_get_key (gcentry), name, size);

        g_free (data->font_name);
        data->font_name = name;
        data->font_size = size;
    }
    else
    {
        g_free (name);

        g_free (data->font_name);
        data->font_name = NULL;
        data->font_size = 0;
    }

    for (i = 0; data->font_name_prefs[i] != NULL; ++i)
    {
        gconf_client_notify (client, data->font_name_prefs[i]);
    }
    for (i = 0; data->font_size_prefs[i] != NULL; ++i)
    {
        gconf_client_notify (client, data->font_size_prefs[i]);
    }
}

typedef struct
{
    guint cnxn_id;
    char **prefs;
    int size;
} MinimumFontSizeData;

static gboolean
transform_minimum_font_size (GConfEntry *gcentry,
                 GValue *value,
                 gpointer user_data)
{
    MinimumFontSizeData *data = (MinimumFontSizeData *) user_data;
    GConfValue *gcvalue;
    int size = 0;

    gcvalue = gconf_entry_get_value (gcentry);
    if (gcvalue == NULL ||
        gcvalue->type != GCONF_VALUE_INT)
    {
        size = data->size;
    }
    else
    {
        size = MAX (gconf_value_get_int (gcvalue),
                data->size);
    }

    if (size <= 0) return FALSE;

    g_value_init (value, G_TYPE_INT);
    g_value_set_int (value, size);

    return TRUE;
}

static void
notify_minimum_size_cb (GConfClient *client,
            guint cnxn_id,
            GConfEntry *entry,
            MinimumFontSizeData *data)
{
    GConfValue *gcvalue;
    int i;

    data->size = 0;

    gcvalue = gconf_entry_get_value (entry);
    /* happens on initial notify if the key doesn't exist */
    if (gcvalue != NULL &&
        gcvalue->type == GCONF_VALUE_INT)
    {
        data->size = gconf_value_get_int (gcvalue);
        data->size = MAX (data->size, 0);
    }

    LOG ("Minimum font size now %d", data->size);

    for (i = 0; data->prefs[i] != NULL; ++i)
    {
        gconf_client_notify (client, data->prefs[i]);
    }
}

static DesktopFontData *desktop_font_data;
static MinimumFontSizeData *minimum_font_size_data;

static void
mozilla_font_notifiers_init (void)
{
    static const char *types [] = { "variable", "monospace" };
    const EphyFontsLanguageInfo *font_languages;
    guint n_font_languages, i;

    eel_gconf_monitor_add ("/desktop/gnome/interface");

    font_languages = ephy_font_languages ();
    n_font_languages = ephy_font_n_languages ();

    desktop_font_data = g_new0 (DesktopFontData, 2);
    desktop_font_data[0].font_name_prefs = g_new0 (char*, n_font_languages + 1);
    desktop_font_data[0].font_size_prefs = g_new0 (char*, n_font_languages + 1);
    desktop_font_data[1].font_name_prefs = g_new0 (char*, n_font_languages + 1);
    desktop_font_data[1].font_size_prefs = g_new0 (char*, n_font_languages + 1);

    desktop_font_data[0].cnxn_id =
        eel_gconf_notification_add (CONF_DESKTOP_FONT_VARIABLE,
                        (GConfClientNotifyFunc) notify_desktop_font_cb,
                        &desktop_font_data[0]);
    eel_gconf_notify (CONF_DESKTOP_FONT_VARIABLE);

    desktop_font_data[1].cnxn_id =
        eel_gconf_notification_add (CONF_DESKTOP_FONT_MONOSPACE,
                        (GConfClientNotifyFunc) notify_desktop_font_cb,
                        &desktop_font_data[1]);
    eel_gconf_notify (CONF_DESKTOP_FONT_MONOSPACE);

    minimum_font_size_data = g_new0 (MinimumFontSizeData, 1);
    minimum_font_size_data->prefs = g_new0 (char*, n_font_languages + 1);

    minimum_font_size_data->cnxn_id =
        eel_gconf_notification_add (CONF_RENDERING_FONT_MIN_SIZE,
                        (GConfClientNotifyFunc) notify_minimum_size_cb,
                        minimum_font_size_data);
    eel_gconf_notify (CONF_RENDERING_FONT_MIN_SIZE);

#ifdef MIGRATE_PIXEL_SIZE
    gboolean migrate_size;

    migrate_size = (eel_gconf_get_integer (CONF_SCHEMA_VERSION)
            < EPIPHANY_SCHEMA_VERSION);
    if (migrate_size)
    {
        eel_gconf_set_integer (CONF_SCHEMA_VERSION, EPIPHANY_SCHEMA_VERSION);
    }
#endif

    for (i=0; i < n_font_languages; i++)
    {
        const char *code = font_languages[i].code;
        guint k;
        char key[255], pref[255];
#ifdef MIGRATE_PIXEL_SIZE
        char old_key[255];
#endif

        for (k = 0; k < G_N_ELEMENTS (types); k++)
        {
            g_snprintf (key, sizeof (key), "%s_%s_%s",
                    CONF_RENDERING_FONT, types[k], code);
            g_snprintf (pref, sizeof (pref), "font.name.%s.%s",
                    types[k], code);

            desktop_font_data[k].font_name_prefs[i] = g_strdup (key);

            mozilla_notifier_add (key, pref,
                          transform_font_name,
                          &desktop_font_data[k]);
        }

#ifdef MIGRATE_PIXEL_SIZE
        if (migrate_size)
        {
            char *type;

            type = eel_gconf_get_string (CONF_RENDERING_FONT_TYPE_OLD);
            if (type && (strcmp (type, "serif") == 0 ||
                     strcmp (type, "sans-serif") == 0))
            {
                char *family;

                g_snprintf (old_key, sizeof (old_key), "%s_%s_%s",
                        CONF_RENDERING_FONT, type, code);
                g_snprintf (key, sizeof (key), "%s_%s_%s",
                        CONF_RENDERING_FONT, "variable", code);

                family = eel_gconf_get_string (old_key);
                if (family)
                {
                    eel_gconf_set_string (key, family);
                    g_free (family);
                }
            }

            g_free (type);
        }
#endif

        /* FIXME is it "minimum-size" or "min-size" !!? */
        g_snprintf (key, sizeof (key), "%s_%s",
                CONF_RENDERING_FONT_MIN_SIZE, code);
        g_snprintf (pref, sizeof (pref), "font.minimum-size.%s", code);

        minimum_font_size_data->prefs[i] = g_strdup (key);

        mozilla_notifier_add (key, pref, transform_minimum_font_size,
                      minimum_font_size_data);

#ifdef MIGRATE_PIXEL_SIZE
        if (migrate_size)
        {
            g_snprintf (old_key, sizeof (key), "%s_%s",
                    CONF_RENDERING_FONT_MIN_SIZE_OLD, code);
            migrate_font_gconf_key (old_key, key);
        }
#endif

        g_snprintf (key, sizeof (key), "%s_%s",
                CONF_RENDERING_FONT_FIXED_SIZE, code);
        g_snprintf (pref, sizeof (pref), "font.size.fixed.%s", code);

        desktop_font_data[1].font_size_prefs[i] = g_strdup (key);
        mozilla_notifier_add (key, pref, transform_font_size,
                      &desktop_font_data[1]);

#ifdef MIGRATE_PIXEL_SIZE
        if (migrate_size)
        {
            g_snprintf (old_key, sizeof (old_key), "%s_%s",
                    CONF_RENDERING_FONT_FIXED_SIZE_OLD, code);
            migrate_font_gconf_key (old_key, key);
        }
#endif

        g_snprintf (key, sizeof (key), "%s_%s",
                CONF_RENDERING_FONT_VAR_SIZE, code);
        g_snprintf (pref, sizeof (pref), "font.size.variable.%s", code);
    
        desktop_font_data[0].font_size_prefs[i] = g_strdup (key);
        mozilla_notifier_add (key, pref, transform_font_size,
                      &desktop_font_data[0]);

#ifdef MIGRATE_PIXEL_SIZE
        if (migrate_size)
        {
            g_snprintf (old_key, sizeof (old_key), "%s_%s",
                    CONF_RENDERING_FONT_VAR_SIZE_OLD, code);
            migrate_font_gconf_key (old_key, key);
        }
#endif
    }
}

static void
mozilla_font_notifiers_shutdown (void)
{
    eel_gconf_monitor_remove ("/desktop/gnome/interface");

    eel_gconf_notification_remove (desktop_font_data[0].cnxn_id);
    eel_gconf_notification_remove (desktop_font_data[1].cnxn_id);
    eel_gconf_notification_remove (minimum_font_size_data->cnxn_id);

    g_free (desktop_font_data[0].font_name);
    g_free (desktop_font_data[1].font_name);

    g_strfreev (desktop_font_data[0].font_name_prefs);
    g_strfreev (desktop_font_data[0].font_size_prefs);
    g_strfreev (desktop_font_data[1].font_name_prefs);
    g_strfreev (desktop_font_data[1].font_size_prefs);
    g_strfreev (minimum_font_size_data->prefs);

    g_free (desktop_font_data);
    g_free (minimum_font_size_data);    
}

#if 0
static void
notify_generic_cb (GConfClient *client,
           guint cnxn_id,
           GConfEntry *gcentry,
           gpointer data)
{
    GConfValue *gcvalue;
    const char *key, *pref;
    static const gsize len = strlen ("/apps/epiphany/web/gecko_prefs/");
    GValue value = { 0, };

    key = gconf_entry_get_key (gcentry);
    if (!key || strlen (key) <= len) return;

    pref = key + len;
    
    gcvalue = gconf_entry_get_value (gcentry);
    if (!gcvalue) return;

    switch (gcvalue->type)
    {
        case GCONF_VALUE_STRING:
            g_value_init (&value, G_TYPE_STRING);
            g_value_set_static_string (&value, gconf_value_get_string (gcvalue));
            break;
        case GCONF_VALUE_INT:
            g_value_init (&value, G_TYPE_INT);
            g_value_set_int (&value, gconf_value_get_int (gcvalue));
            break;
        case GCONF_VALUE_BOOL:
            g_value_init (&value, G_TYPE_BOOLEAN);
            g_value_set_boolean (&value, gconf_value_get_bool (gcvalue));
            break;
        /* case GCONF_VALUE_INVALID: */
        /* case GCONF_VALUE_FLOAT: */
        /* case GCONF_VALUE_SCHEMA: */
        /* case GCONF_VALUE_LIST: */
        /* case GCONF_VALUE_PAIR: */
        default:
            g_warning ("Unsupported value type for key '%s'\n", key);
            return;
    }

    g_assert (G_IS_VALUE (&value));
    mozilla_pref_set (pref, &value);
    g_value_unset (&value);
}

static void
mozilla_generic_notifier_init (void)
{
    GConfClient *client;
    GError *error = NULL;
    guint cnxn_id;
    PrefData *data;
    GSList *list, *l;

    client = eel_gconf_client_get_global ();

    cnxn_id = gconf_client_notify_add (client, "/apps/epiphany/web/gecko_prefs",
                       (GConfClientNotifyFunc) notify_generic_cb,
                       NULL, NULL, &error);
    if (eel_gconf_handle_error (&error))
    {
        if (cnxn_id != EEL_GCONF_UNDEFINED_CONNECTION)
        {
            gconf_client_notify_remove (client, cnxn_id);
        }

        return;
    }

    data = g_slice_new0 (PrefData);
    data->cnxn_id = cnxn_id;
    notifiers = g_list_prepend (notifiers, data);

    list = gconf_client_all_entries (client, "/apps/epiphany/web/gecko_prefs", NULL);
    for (l = list; l != NULL; l = l->next)
    {
        GConfEntry *gcentry = (GConfEntry *) l->data;

        notify_generic_cb (client, cnxn_id, gcentry, NULL);
        gconf_entry_unref (gcentry);
    }
    g_slist_free (list);
}
#endif

extern "C" gboolean
mozilla_notifiers_init (void)
{
    guint i;

    eel_gconf_monitor_add ("/apps/epiphany/web");
    eel_gconf_monitor_add ("/system/proxy");
    eel_gconf_monitor_add ("/system/http_proxy");

    /* the pref service conveniently implements the root pref branch */
    gPrefBranch = nsnull;
    nsresult rv = CallGetService (NS_PREFSERVICE_CONTRACTID, &gPrefBranch);
    if (NS_FAILED (rv) || !gPrefBranch)
    {
        g_warning ("Failed to get the pref service!\n");
        return FALSE;
    }

#if 0
    /* First init the generic notifier, so our regular prefs override prefs set there */
    mozilla_generic_notifier_init ();
#endif

    for (i = 0; i < G_N_ELEMENTS (notifier_entries); i++)
    {
        mozilla_notifier_add (notifier_entries[i].gconf_key,
                      notifier_entries[i].mozilla_pref,
                      notifier_entries[i].func,
                      notifier_entries[i].user_data);
    }

    mozilla_font_notifiers_init ();

    return TRUE;
}

static void
remove_notification (PrefData *data)
{
    eel_gconf_notification_remove (data->cnxn_id);
}

extern "C" void
mozilla_notifiers_shutdown (void)
{
    NS_IF_RELEASE (gPrefBranch);
    gPrefBranch = nsnull;

    mozilla_font_notifiers_shutdown ();

    eel_gconf_monitor_remove ("/apps/epiphany/web");
    eel_gconf_monitor_remove ("/system/proxy");
    eel_gconf_monitor_remove ("/system/http_proxy");

    g_list_foreach (notifiers, (GFunc) remove_notification, NULL);
    g_list_free (notifiers);
}