aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--configure.ac2
-rw-r--r--lib/widgets/Makefile.am2
-rw-r--r--lib/widgets/ephy-certificate-dialog.c330
-rw-r--r--lib/widgets/ephy-certificate-dialog.h65
-rw-r--r--po/POTFILES.in1
-rw-r--r--src/ephy-window.c27
6 files changed, 427 insertions, 0 deletions
diff --git a/configure.ac b/configure.ac
index 7fd5d4e63..7ffc0071a 100644
--- a/configure.ac
+++ b/configure.ac
@@ -89,6 +89,7 @@ LIBSOUP_GNOME_REQUIRED=2.39.6
GNOME_KEYRING_REQUIRED=2.26.0
GSETTINGS_DESKTOP_SCHEMAS_REQUIRED=0.0.1
LIBNOTIFY_REQUIRED=0.5.1
+GCR_REQUIRED=3.5.5
AC_ARG_WITH(webkit2,
[AC_HELP_STRING([--with-webkit2], [build with WebKit2 [default=no]])],
@@ -130,6 +131,7 @@ PKG_CHECK_MODULES([DEPENDENCIES], [
gsettings-desktop-schemas >= $GSETTINGS_DESKTOP_SCHEMAS_REQUIRED
libnotify >= $LIBNOTIFY_REQUIRED
sqlite3
+ gcr-3 >= $GCR_REQUIRED
])
# ******************
diff --git a/lib/widgets/Makefile.am b/lib/widgets/Makefile.am
index 9f751e0d8..8a7e54b81 100644
--- a/lib/widgets/Makefile.am
+++ b/lib/widgets/Makefile.am
@@ -1,6 +1,8 @@
noinst_LTLIBRARIES = libephywidgets.la
libephywidgets_la_SOURCES = \
+ ephy-certificate-dialog.c \
+ ephy-certificate-dialog.h \
ephy-download-widget.c \
ephy-download-widget.h \
ephy-history-view.c \
diff --git a/lib/widgets/ephy-certificate-dialog.c b/lib/widgets/ephy-certificate-dialog.c
new file mode 100644
index 000000000..a1632cdef
--- /dev/null
+++ b/lib/widgets/ephy-certificate-dialog.c
@@ -0,0 +1,330 @@
+/*
+ * Copyright © 2012 Igalia S.L.
+ *
+ * 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.
+ *
+ */
+
+#include "config.h"
+#include "ephy-certificate-dialog.h"
+
+#define GCR_API_SUBJECT_TO_CHANGE
+#include <gcr/gcr.h>
+#include <glib/gi18n.h>
+#include <libsoup/soup.h>
+
+/**
+ * SECTION:ephy-certificate-dialog
+ * @short_description: A dialog to show SSL certificate information
+ *
+ * #EphyCertificateDialog shows information about SSL certificates.
+ */
+
+enum
+{
+ PROP_0,
+ PROP_ADDRESS,
+ PROP_CERTIFICATE,
+ PROP_TLS_ERRORS
+};
+
+struct _EphyCertificateDialogPrivate
+{
+ GtkWidget *icon;
+ GtkWidget *title;
+ GtkWidget *text;
+};
+
+G_DEFINE_TYPE (EphyCertificateDialog, ephy_certificate_dialog, GTK_TYPE_DIALOG)
+
+static void
+ephy_certificate_dialog_set_address (EphyCertificateDialog *dialog,
+ const char *address)
+{
+ SoupURI *uri;
+
+ uri = soup_uri_new (address);
+ gtk_window_set_title (GTK_WINDOW (dialog), uri->host);
+ soup_uri_free (uri);
+}
+
+static void
+ephy_certificate_dialog_set_certificate (EphyCertificateDialog *dialog,
+ GTlsCertificate *certificate)
+{
+ GcrCertificate *simple_certificate;
+ GByteArray *certificate_data;
+ GtkWidget *certificate_widget;
+ GtkWidget *content_area;
+
+ g_object_get (certificate, "certificate", &certificate_data, NULL);
+ simple_certificate = gcr_simple_certificate_new ((const guchar *)certificate_data->data,
+ certificate_data->len);
+ g_byte_array_unref (certificate_data);
+
+ certificate_widget = GTK_WIDGET (gcr_certificate_widget_new (simple_certificate));
+ g_object_unref (simple_certificate);
+
+ content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
+ gtk_box_pack_start (GTK_BOX (content_area), certificate_widget, TRUE, TRUE, 0);
+ gtk_widget_show (certificate_widget);
+}
+
+static char *
+get_error_messages_from_tls_errors (GTlsCertificateFlags tls_errors)
+{
+ GPtrArray *errors = g_ptr_array_new ();
+ char *retval;
+
+ if (tls_errors & G_TLS_CERTIFICATE_BAD_IDENTITY)
+ g_ptr_array_add (errors, _("The certificate does not match the expected identity"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_EXPIRED)
+ g_ptr_array_add (errors, _("The certificate has expired"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_UNKNOWN_CA)
+ g_ptr_array_add (errors, _("The signing certificate authority is not known"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_GENERIC_ERROR)
+ g_ptr_array_add (errors, _("The certificate contains errors"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_REVOKED)
+ g_ptr_array_add (errors, _("The certificate has been revoked"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_INSECURE)
+ g_ptr_array_add (errors, _("The certificate is signed using a weak signature algorithm"));
+
+ if (tls_errors & G_TLS_CERTIFICATE_NOT_ACTIVATED)
+ g_ptr_array_add (errors, _("The certificate activation time is still in the future"));
+
+ if (errors->len == 1)
+ retval = g_strdup (g_ptr_array_index (errors, 0));
+ else {
+ GString *message = g_string_new (NULL);
+ guint i;
+
+ for (i = 0; i < errors->len; i++) {
+ g_string_append_printf (message, "• %s",
+ (char *)g_ptr_array_index (errors, i));
+ if (i < errors->len - 1)
+ g_string_append_c (message, '\n');
+ }
+
+ retval = g_string_free (message, FALSE);
+ }
+
+ g_ptr_array_free (errors, TRUE);
+
+ return retval;
+}
+
+static void
+ephy_certificate_dialog_set_tls_errors (EphyCertificateDialog *dialog,
+ GTlsCertificateFlags tls_errors)
+{
+ EphyCertificateDialogPrivate *priv = dialog->priv;
+ GIcon *icon;
+ char *markup;
+
+ icon = tls_errors == 0 ?
+ g_themed_icon_new_with_default_fallbacks ("channel-secure-symbolic") :
+ g_themed_icon_new_with_default_fallbacks ("channel-insecure-symbolic");
+ gtk_image_set_from_gicon (GTK_IMAGE (priv->icon), icon, GTK_ICON_SIZE_DIALOG);
+ g_object_unref (icon);
+
+ markup = g_strdup_printf ("<span weight=\"bold\" size=\"large\">%s</span>",
+ tls_errors == 0 ?
+ _("The identity of this website has been verified") :
+ _("The identity of this website has not been verified"));
+ gtk_label_set_markup (GTK_LABEL (priv->title), markup);
+ g_free (markup);
+
+ if (tls_errors) {
+ char *text = get_error_messages_from_tls_errors (tls_errors);
+
+ gtk_label_set_text (GTK_LABEL (priv->text), text);
+ g_free (text);
+
+ gtk_widget_show (priv->text);
+ }
+}
+
+static void
+ephy_certificate_dialog_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ EphyCertificateDialog *dialog = EPHY_CERTIFICATE_DIALOG (object);
+
+ switch (prop_id) {
+ case PROP_ADDRESS:
+ ephy_certificate_dialog_set_address (dialog, g_value_get_string (value));
+ break;
+ case PROP_CERTIFICATE:
+ ephy_certificate_dialog_set_certificate (dialog, g_value_get_object (value));
+ break;
+ case PROP_TLS_ERRORS:
+ ephy_certificate_dialog_set_tls_errors (dialog, g_value_get_flags (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ephy_certificate_dialog_class_init (EphyCertificateDialogClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->set_property = ephy_certificate_dialog_set_property;
+
+ /**
+ * EphyCertificateDialog:address:
+ *
+ * The address of the website.
+ */
+ g_object_class_install_property (object_class,
+ PROP_ADDRESS,
+ g_param_spec_string ("address",
+ "Address",
+ "The address of the website",
+ NULL,
+ G_PARAM_WRITABLE |
+ G_PARAM_CONSTRUCT_ONLY |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_NICK |
+ G_PARAM_STATIC_BLURB));
+
+ /**
+ * EphyCertificateDialog:certificate:
+ *
+ * The certificate of the website.
+ */
+ g_object_class_install_property (object_class,
+ PROP_CERTIFICATE,
+ g_param_spec_object ("certificate",
+ "Certificate",
+ "The certificate of the website",
+ G_TYPE_TLS_CERTIFICATE,
+ G_PARAM_WRITABLE |
+ G_PARAM_CONSTRUCT_ONLY |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_NICK |
+ G_PARAM_STATIC_BLURB));
+
+ /**
+ * EphyCertificateDialog:tls-errors:
+ *
+ * The verification errors on the TLS certificate.
+ */
+ g_object_class_install_property (object_class,
+ PROP_TLS_ERRORS,
+ g_param_spec_flags ("tls-errors",
+ "TLS Errors",
+ "The verification errors on the TLS certificate",
+ G_TYPE_TLS_CERTIFICATE_FLAGS,
+ 0,
+ G_PARAM_WRITABLE |
+ G_PARAM_CONSTRUCT_ONLY |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_NICK |
+ G_PARAM_STATIC_BLURB));
+
+ g_type_class_add_private (object_class, sizeof (EphyCertificateDialogPrivate));
+}
+
+static void
+ephy_certificate_dialog_init (EphyCertificateDialog *dialog)
+{
+ GtkWidget *vbox, *hbox;
+ GtkWidget *content_area, *action_area;
+ EphyCertificateDialogPrivate *priv;
+
+ dialog->priv = G_TYPE_INSTANCE_GET_PRIVATE (dialog,
+ EPHY_TYPE_CERTIFICATE_DIALOG,
+ EphyCertificateDialogPrivate);
+ priv = dialog->priv;
+
+ gtk_container_set_border_width (GTK_CONTAINER (dialog), 5);
+ gtk_window_set_skip_taskbar_hint (GTK_WINDOW (dialog), TRUE);
+
+ hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12);
+ gtk_container_set_border_width (GTK_CONTAINER (hbox), 5);
+
+ priv->icon = gtk_image_new ();
+ gtk_widget_set_halign (priv->icon, GTK_ALIGN_CENTER);
+ gtk_widget_set_valign (priv->icon, GTK_ALIGN_START);
+ gtk_box_pack_start (GTK_BOX (hbox), priv->icon, FALSE, FALSE, 0);
+ gtk_widget_show (priv->icon);
+
+ vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
+
+ priv->title = gtk_label_new (NULL);
+ gtk_label_set_use_markup (GTK_LABEL (priv->title), TRUE);
+ gtk_label_set_line_wrap (GTK_LABEL (priv->title), TRUE);
+ gtk_label_set_selectable (GTK_LABEL (priv->title), TRUE);
+ gtk_widget_set_halign (priv->title, GTK_ALIGN_START);
+ gtk_widget_set_valign (priv->title, GTK_ALIGN_START);
+ gtk_misc_set_alignment (GTK_MISC (priv->title), 0.0, 0.0);
+ gtk_box_pack_start (GTK_BOX (vbox), priv->title, FALSE, FALSE, 0);
+ gtk_widget_show (priv->title);
+
+ priv->text = gtk_label_new (NULL);
+ gtk_label_set_line_wrap (GTK_LABEL (priv->text), TRUE);
+ gtk_label_set_selectable (GTK_LABEL (priv->text), TRUE);
+ gtk_widget_set_halign (priv->text, GTK_ALIGN_START);
+ gtk_widget_set_valign (priv->text, GTK_ALIGN_START);
+ gtk_misc_set_alignment (GTK_MISC (priv->text), 0.0, 0.0);
+ gtk_box_pack_start (GTK_BOX (vbox), priv->text, TRUE, TRUE, 0);
+
+ gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0);
+ gtk_widget_show (vbox);
+
+ content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
+ gtk_box_set_spacing (GTK_BOX (content_area), 14);
+ gtk_box_pack_start (GTK_BOX (content_area), hbox, FALSE, FALSE, 0);
+ gtk_widget_show (hbox);
+
+ action_area = gtk_dialog_get_action_area (GTK_DIALOG (dialog));
+ gtk_container_set_border_width (GTK_CONTAINER (action_area), 5);
+ gtk_box_set_spacing (GTK_BOX (action_area), 6);
+}
+
+GtkWidget *
+ephy_certificate_dialog_new (GtkWindow *parent,
+ const char *address,
+ GTlsCertificate *certificate,
+ GTlsCertificateFlags tls_errors)
+{
+ GtkWidget *dialog;
+
+ g_return_val_if_fail (address != NULL, NULL);
+ g_return_val_if_fail (G_IS_TLS_CERTIFICATE (certificate), NULL);
+
+ dialog = GTK_WIDGET (g_object_new (EPHY_TYPE_CERTIFICATE_DIALOG,
+ "address", address,
+ "certificate", certificate,
+ "tls-errors", tls_errors,
+ NULL));
+ gtk_dialog_add_buttons (GTK_DIALOG (dialog),
+ GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
+ NULL);
+ if (parent)
+ gtk_window_set_transient_for (GTK_WINDOW (dialog), parent);
+
+ return dialog;
+}
+
diff --git a/lib/widgets/ephy-certificate-dialog.h b/lib/widgets/ephy-certificate-dialog.h
new file mode 100644
index 000000000..2f2a612b6
--- /dev/null
+++ b/lib/widgets/ephy-certificate-dialog.h
@@ -0,0 +1,65 @@
+/*
+ * Copyright © 2012 Igalia S.L.
+ *
+ * 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.
+ *
+ */
+
+#if !defined (__EPHY_EPIPHANY_H_INSIDE__) && !defined (EPIPHANY_COMPILATION)
+#error "Only <epiphany/epiphany.h> can be included directly."
+#endif
+
+#ifndef EPHY_CERTIFICATE_DIALOG_H
+#define EPHY_CERTIFICATE_DIALOG_H
+
+#include <gio/gio.h>
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define EPHY_TYPE_CERTIFICATE_DIALOG (ephy_certificate_dialog_get_type())
+#define EPHY_CERTIFICATE_DIALOG(object) (G_TYPE_CHECK_INSTANCE_CAST((object), EPHY_TYPE_CERTIFICATE_DIALOG, EphyCertificateDialog))
+#define EPHY_IS_CERTIFICATE_DIALOG(object) (G_TYPE_CHECK_INSTANCE_TYPE((object), EPHY_TYPE_CERTIFICATE_DIALOG))
+#define EPHY_CERTIFICATE_DIALOG_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass), EPHY_TYPE_CERTIFICATE_DIALOG, EphyCertificateDialogClass))
+#define EPHY_IS_CERTIFICATE_DIALOG_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), EPHY_TYPE_CERTIFICATE_DIALOG))
+#define EPHY_CERTIFICATE_DIALOG_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj), EPHY_TYPE_CERTIFICATE_DIALOG, EphyCertificateDialogClass))
+
+typedef struct _EphyCertificateDialog EphyCertificateDialog;
+typedef struct _EphyCertificateDialogClass EphyCertificateDialogClass;
+typedef struct _EphyCertificateDialogPrivate EphyCertificateDialogPrivate;
+
+struct _EphyCertificateDialog
+{
+ GtkDialog parent_object;
+
+ /*< private >*/
+ EphyCertificateDialogPrivate *priv;
+};
+
+struct _EphyCertificateDialogClass
+{
+ GtkDialogClass parent_class;
+};
+
+GType ephy_certificate_dialog_get_type (void);
+
+GtkWidget *ephy_certificate_dialog_new (GtkWindow *parent,
+ const char *address,
+ GTlsCertificate *certificate,
+ GTlsCertificateFlags tls_errors);
+
+G_END_DECLS
+
+#endif
diff --git a/po/POTFILES.in b/po/POTFILES.in
index 108396bb2..e7211b88b 100644
--- a/po/POTFILES.in
+++ b/po/POTFILES.in
@@ -25,6 +25,7 @@ lib/ephy-string.c
lib/ephy-time-helpers.c
lib/ephy-zoom.h
lib/history/ephy-history-service-hosts-table.c
+lib/widgets/ephy-certificate-dialog.c
lib/widgets/ephy-download-widget.c
lib/widgets/ephy-hosts-store.c
lib/widgets/ephy-hosts-view.c
diff --git a/src/ephy-window.c b/src/ephy-window.c
index bf91d30b6..c9113c2eb 100644
--- a/src/ephy-window.c
+++ b/src/ephy-window.c
@@ -25,6 +25,7 @@
#include "ephy-action-helper.h"
#include "ephy-bookmarks-ui.h"
+#include "ephy-certificate-dialog.h"
#include "ephy-combined-stop-reload-action.h"
#include "ephy-debug.h"
#include "ephy-download-widget.h"
@@ -3429,6 +3430,30 @@ zoom_to_level_cb (GtkAction *action,
ephy_window_set_zoom (window, zoom);
}
+static void
+lock_clicked_cb (EphyLocationController *controller,
+ EphyWindow *window)
+{
+ EphyWindowPrivate *priv = window->priv;
+ EphyWebView *view;
+ GTlsCertificate *certificate;
+ GTlsCertificateFlags tls_errors;
+ GtkWidget *certificate_dialog;
+
+ view = ephy_embed_get_web_view (priv->active_embed);
+ ephy_web_view_get_security_level (view, NULL, &certificate, &tls_errors);
+
+ certificate_dialog = ephy_certificate_dialog_new (GTK_WINDOW (window),
+ ephy_location_controller_get_address (controller),
+ certificate,
+ tls_errors);
+ gtk_window_set_destroy_with_parent (GTK_WINDOW (certificate_dialog), TRUE);
+ g_signal_connect (certificate_dialog, "response",
+ G_CALLBACK (gtk_widget_destroy),
+ NULL);
+ gtk_widget_show (certificate_dialog);
+}
+
static GtkWidget *
setup_toolbar (EphyWindow *window)
{
@@ -3559,6 +3584,8 @@ ephy_window_constructor (GType type,
G_CALLBACK (sync_user_input_cb), window);
g_signal_connect_swapped (priv->location_controller, "open-link",
G_CALLBACK (ephy_link_open), window);
+ g_signal_connect (priv->location_controller, "lock-clicked",
+ G_CALLBACK (lock_clicked_cb), window);
g_signal_connect_swapped (priv->notebook, "open-link",
G_CALLBACK (ephy_link_open), window);