blob: 078368e049b37452c620ac5b5da8f0c9bfb2df13 [file] [log] [blame]
/*
* Copyright (C) 2007 Carlos Garcia Campos <carlosgc@gnome.org>
*
* 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 <gtk/gtk.h>
#include "fonts.h"
enum {
FONTS_NAME_COLUMN,
FONTS_DETAILS_COLUMN,
N_COLUMNS
};
typedef struct {
PopplerDocument *doc;
GtkWidget *treeview;
GtkWidget *progress;
guint idle_id;
} PgdFontsDemo;
static void
pgd_fonts_free (PgdFontsDemo *demo)
{
if (!demo)
return;
if (demo->idle_id > 0) {
g_source_remove (demo->idle_id);
demo->idle_id = 0;
}
if (demo->doc) {
g_object_unref (demo->doc);
demo->doc = NULL;
}
g_free (demo);
}
static void
pdg_fonts_cell_data_func (GtkTreeViewColumn *col,
GtkCellRenderer *renderer,
GtkTreeModel *model,
GtkTreeIter *iter,
gpointer user_data)
{
char *name;
char *details;
char *markup;
gtk_tree_model_get (model, iter,
FONTS_NAME_COLUMN, &name,
FONTS_DETAILS_COLUMN, &details,
-1);
if (details) {
markup = g_strdup_printf ("<b><big>%s</big></b>\n<small>%s</small>",
name, details);
} else {
markup = g_strdup_printf ("<b><big>%s</big></b>", name);
}
g_object_set (renderer, "markup", markup, NULL);
g_free (markup);
g_free (details);
g_free (name);
}
static const gchar *
font_type_to_string (PopplerFontType type)
{
switch (type) {
case POPPLER_FONT_TYPE_TYPE1:
return "Type 1";
case POPPLER_FONT_TYPE_TYPE1C:
return "Type 1C";
case POPPLER_FONT_TYPE_TYPE3:
return "Type 3";
case POPPLER_FONT_TYPE_TRUETYPE:
return "TrueType";
case POPPLER_FONT_TYPE_CID_TYPE0:
return "Type 1 (CID)";
case POPPLER_FONT_TYPE_CID_TYPE0C:
return "Type 1C (CID)";
case POPPLER_FONT_TYPE_CID_TYPE2:
return "TrueType (CID)";
default:
return "Unknown font type";
}
}
static void
pgd_fonts_update_progress (PgdFontsDemo *demo,
gint n_pages,
gint scanned)
{
gchar *str;
str = g_strdup_printf ("Scanning fonts (%d%%)",
MIN (scanned * 100 / n_pages, 100));
gtk_progress_bar_set_text (GTK_PROGRESS_BAR (demo->progress), str);
gtk_progress_bar_set_fraction (GTK_PROGRESS_BAR (demo->progress),
MIN ((gdouble)scanned / n_pages, 1.0));
g_free (str);
}
static gboolean
pgd_fonts_fill_model (PgdFontsDemo *demo)
{
GtkTreeModel *model;
PopplerFontInfo *font_info;
PopplerFontsIter *fonts_iter;
gint n_pages, scanned = 0;
n_pages = poppler_document_get_n_pages (demo->doc);
model = gtk_tree_view_get_model (GTK_TREE_VIEW (demo->treeview));
g_object_ref (model);
gtk_list_store_clear (GTK_LIST_STORE (model));
font_info = poppler_font_info_new (demo->doc);
while (poppler_font_info_scan (font_info, 20, &fonts_iter)) {
pgd_fonts_update_progress (demo, n_pages, scanned);
while (gtk_events_pending ())
gtk_main_iteration ();
scanned += 20;
if (!fonts_iter)
continue;
do {
GtkTreeIter iter;
const gchar *name;
const gchar *type;
const gchar *embedded;
const gchar *substitute;
const gchar *filename;
const gchar *encoding;
gchar *details;
name = poppler_fonts_iter_get_name (fonts_iter);
if (!name)
name = "No name";
encoding = poppler_fonts_iter_get_encoding (fonts_iter);
if (!encoding)
encoding = "None";
type = font_type_to_string (poppler_fonts_iter_get_font_type (fonts_iter));
if (poppler_fonts_iter_is_embedded (fonts_iter)) {
if (poppler_fonts_iter_is_subset (fonts_iter))
embedded = "Embedded subset";
else
embedded = "Embedded";
} else {
embedded = "Not embedded";
}
substitute = poppler_fonts_iter_get_substitute_name (fonts_iter);
filename = poppler_fonts_iter_get_file_name (fonts_iter);
if (substitute && filename)
details = g_markup_printf_escaped ("%s\nEncoding: %s\n%s, substituting with <b>%s</b>\n(%s)", type, encoding, embedded, substitute, filename);
else
details = g_markup_printf_escaped ("%s\nEncoding: %s\n%s", type, encoding, embedded);
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
gtk_list_store_set (GTK_LIST_STORE (model), &iter,
FONTS_NAME_COLUMN, name,
FONTS_DETAILS_COLUMN, details,
-1);
g_free (details);
} while (poppler_fonts_iter_next (fonts_iter));
poppler_fonts_iter_free (fonts_iter);
}
pgd_fonts_update_progress (demo, n_pages, scanned);
g_object_unref (font_info);
g_object_unref (model);
return FALSE;
}
static void
pgd_fonts_scan_button_clicked (GtkButton *button,
PgdFontsDemo *demo)
{
demo->idle_id = g_idle_add ((GSourceFunc)pgd_fonts_fill_model, demo);
}
GtkWidget *
pgd_fonts_create_widget (PopplerDocument *document)
{
PgdFontsDemo *demo;
GtkWidget *vbox;
GtkListStore *model;
GtkCellRenderer *renderer;
GtkTreeViewColumn *column;
GtkWidget *swindow;
GtkWidget *hbox, *button;
demo = g_new0 (PgdFontsDemo, 1);
demo->doc = g_object_ref (document);
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
demo->progress = gtk_progress_bar_new ();
gtk_progress_bar_set_ellipsize (GTK_PROGRESS_BAR (demo->progress),
PANGO_ELLIPSIZE_END);
gtk_box_pack_start (GTK_BOX (hbox), demo->progress, TRUE, TRUE, 0);
gtk_widget_show (demo->progress);
button = gtk_button_new_with_label ("Scan");
g_signal_connect (G_OBJECT (button), "clicked",
G_CALLBACK (pgd_fonts_scan_button_clicked),
(gpointer)demo);
gtk_box_pack_end (GTK_BOX (hbox), button, FALSE, FALSE, 0);
gtk_widget_show (button);
gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 6);
gtk_widget_show (hbox);
swindow = gtk_scrolled_window_new (NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (swindow),
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
model = gtk_list_store_new (N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING);
demo->treeview = gtk_tree_view_new_with_model (GTK_TREE_MODEL (model));
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (demo->treeview), FALSE);
gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (demo->treeview)),
GTK_SELECTION_NONE);
g_object_unref (model);
column = gtk_tree_view_column_new ();
gtk_tree_view_append_column (GTK_TREE_VIEW (demo->treeview), column);
renderer = gtk_cell_renderer_text_new ();
gtk_tree_view_column_pack_start (GTK_TREE_VIEW_COLUMN (column), renderer, FALSE);
gtk_tree_view_column_set_cell_data_func (column, renderer,
pdg_fonts_cell_data_func,
NULL, NULL);
gtk_container_add (GTK_CONTAINER (swindow), demo->treeview);
gtk_widget_show (demo->treeview);
gtk_box_pack_start (GTK_BOX (vbox), swindow, TRUE, TRUE, 0);
gtk_widget_show (swindow);
g_object_weak_ref (G_OBJECT (swindow),
(GWeakNotify)pgd_fonts_free,
(gpointer)demo);
return vbox;
}