2018-01-03 19:23:33 +00:00
|
|
|
/* Pango/Font Explorer
|
2015-06-19 13:28:09 +00:00
|
|
|
*
|
2016-01-25 05:40:49 +00:00
|
|
|
* This example demonstrates support for OpenType font features with
|
|
|
|
* Pango attributes. The attributes can be used manually or via Pango
|
|
|
|
* markup.
|
|
|
|
*
|
|
|
|
* It can also be used to explore available features in OpenType fonts
|
|
|
|
* and their effect.
|
2018-01-03 19:23:33 +00:00
|
|
|
*
|
|
|
|
* If the selected font supports OpenType font variations, then the
|
|
|
|
* axes are also offered for customization.
|
2015-06-19 13:28:09 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gtk/gtk.h>
|
2016-01-25 05:40:49 +00:00
|
|
|
#include <pango/pangofc-font.h>
|
|
|
|
#include <hb.h>
|
|
|
|
#include <hb-ot.h>
|
2018-01-03 19:23:33 +00:00
|
|
|
#include <glib/gi18n.h>
|
|
|
|
|
|
|
|
#include "open-type-layout.h"
|
|
|
|
#include "fontplane.h"
|
|
|
|
#include "script-names.h"
|
|
|
|
#include "language-names.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define MAKE_TAG(a,b,c,d) (unsigned int)(((a) << 24) | ((b) << 16) | ((c) << 8) | (d))
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2020-03-07 14:01:21 +00:00
|
|
|
static GtkWidget *the_label;
|
2015-06-19 13:28:09 +00:00
|
|
|
static GtkWidget *settings;
|
2018-01-03 19:23:33 +00:00
|
|
|
static GtkWidget *description;
|
2015-06-19 13:28:09 +00:00
|
|
|
static GtkWidget *font;
|
2016-01-25 05:40:49 +00:00
|
|
|
static GtkWidget *script_lang;
|
2015-06-19 13:28:09 +00:00
|
|
|
static GtkWidget *resetbutton;
|
|
|
|
static GtkWidget *stack;
|
2020-03-07 14:01:21 +00:00
|
|
|
static GtkWidget *the_entry;
|
2018-01-03 19:23:33 +00:00
|
|
|
static GtkWidget *variations_heading;
|
|
|
|
static GtkWidget *variations_grid;
|
|
|
|
static GtkWidget *instance_combo;
|
|
|
|
static GtkWidget *edit_toggle;
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
typedef struct {
|
|
|
|
unsigned int tag;
|
|
|
|
const char *name;
|
|
|
|
GtkWidget *icon;
|
|
|
|
GtkWidget *dflt;
|
|
|
|
GtkWidget *feat;
|
|
|
|
} FeatureItem;
|
2016-01-25 05:40:49 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
static GList *feature_items;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
unsigned int start;
|
|
|
|
unsigned int end;
|
|
|
|
PangoFontDescription *desc;
|
|
|
|
char *features;
|
|
|
|
PangoLanguage *language;
|
|
|
|
} Range;
|
|
|
|
|
|
|
|
static GList *ranges;
|
|
|
|
|
|
|
|
static void add_font_variations (GString *s);
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_range (gpointer data)
|
|
|
|
{
|
|
|
|
Range *range = data;
|
|
|
|
|
|
|
|
if (range->desc)
|
|
|
|
pango_font_description_free (range->desc);
|
|
|
|
g_free (range->features);
|
|
|
|
g_free (range);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
compare_range (gconstpointer a, gconstpointer b)
|
|
|
|
{
|
|
|
|
const Range *ra = a;
|
|
|
|
const Range *rb = b;
|
|
|
|
|
|
|
|
if (ra->start < rb->start)
|
|
|
|
return -1;
|
|
|
|
else if (ra->start > rb->start)
|
|
|
|
return 1;
|
|
|
|
else if (ra->end < rb->end)
|
|
|
|
return 1;
|
|
|
|
else if (ra->end > rb->end)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ensure_range (unsigned int start,
|
|
|
|
unsigned int end,
|
|
|
|
PangoFontDescription *desc,
|
|
|
|
const char *features,
|
|
|
|
PangoLanguage *language)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
Range *range;
|
|
|
|
|
|
|
|
for (l = ranges; l; l = l->next)
|
|
|
|
{
|
|
|
|
Range *r = l->data;
|
|
|
|
|
|
|
|
if (r->start == start && r->end == end)
|
|
|
|
{
|
|
|
|
range = r;
|
|
|
|
goto set;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
range = g_new0 (Range, 1);
|
|
|
|
range->start = start;
|
|
|
|
range->end = end;
|
|
|
|
|
|
|
|
ranges = g_list_insert_sorted (ranges, range, compare_range);
|
|
|
|
|
|
|
|
set:
|
|
|
|
if (range->desc)
|
|
|
|
pango_font_description_free (range->desc);
|
|
|
|
if (desc)
|
|
|
|
range->desc = pango_font_description_copy (desc);
|
|
|
|
g_free (range->features);
|
|
|
|
range->features = g_strdup (features);
|
|
|
|
range->language = language;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
get_feature_display_name (unsigned int tag)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (open_type_layout_features); i++)
|
|
|
|
{
|
|
|
|
if (tag == open_type_layout_features[i].tag)
|
|
|
|
return g_dpgettext2 (NULL, "OpenType layout", open_type_layout_features[i].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void update_display (void);
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_inconsistent (GtkCheckButton *button,
|
|
|
|
gboolean inconsistent)
|
|
|
|
{
|
|
|
|
gtk_check_button_set_inconsistent (GTK_CHECK_BUTTON (button), inconsistent);
|
|
|
|
gtk_widget_set_opacity (gtk_widget_get_first_child (GTK_WIDGET (button)), inconsistent ? 0.0 : 1.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
feat_clicked (GtkWidget *feat,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
g_signal_handlers_block_by_func (feat, feat_clicked, NULL);
|
|
|
|
|
|
|
|
if (gtk_check_button_get_inconsistent (GTK_CHECK_BUTTON (feat)))
|
|
|
|
{
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (feat), FALSE);
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (feat), TRUE);
|
|
|
|
}
|
|
|
|
else if (!gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (feat)))
|
|
|
|
{
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (feat), FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (feat), TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_signal_handlers_unblock_by_func (feat, feat_clicked, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_check_group (GtkWidget *box,
|
|
|
|
const char *title,
|
|
|
|
const char **tags)
|
|
|
|
{
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *group;
|
|
|
|
PangoAttrList *attrs;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_widget_set_halign (group, GTK_ALIGN_START);
|
|
|
|
|
|
|
|
label = gtk_label_new (title);
|
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_widget_set_halign (label, GTK_ALIGN_START);
|
|
|
|
g_object_set (label, "margin-top", 10, "margin-bottom", 10, NULL);
|
|
|
|
attrs = pango_attr_list_new ();
|
|
|
|
pango_attr_list_insert (attrs, pango_attr_weight_new (PANGO_WEIGHT_BOLD));
|
|
|
|
gtk_label_set_attributes (GTK_LABEL (label), attrs);
|
|
|
|
pango_attr_list_unref (attrs);
|
|
|
|
gtk_container_add (GTK_CONTAINER (group), label);
|
|
|
|
|
|
|
|
for (i = 0; tags[i]; i++)
|
|
|
|
{
|
|
|
|
unsigned int tag;
|
|
|
|
GtkWidget *feat;
|
|
|
|
FeatureItem *item;
|
|
|
|
|
|
|
|
tag = hb_tag_from_string (tags[i], -1);
|
|
|
|
|
|
|
|
feat = gtk_check_button_new_with_label (get_feature_display_name (tag));
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (feat), TRUE);
|
|
|
|
|
|
|
|
g_signal_connect (feat, "notify::active", G_CALLBACK (update_display), NULL);
|
|
|
|
g_signal_connect (feat, "notify::inconsistent", G_CALLBACK (update_display), NULL);
|
|
|
|
g_signal_connect (feat, "clicked", G_CALLBACK (feat_clicked), NULL);
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (group), feat);
|
|
|
|
|
|
|
|
item = g_new (FeatureItem, 1);
|
|
|
|
item->name = tags[i];
|
|
|
|
item->tag = tag;
|
|
|
|
item->icon = NULL;
|
|
|
|
item->dflt = NULL;
|
|
|
|
item->feat = feat;
|
|
|
|
|
|
|
|
feature_items = g_list_prepend (feature_items, item);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (box), group);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_radio_group (GtkWidget *box,
|
|
|
|
const char *title,
|
|
|
|
const char **tags)
|
|
|
|
{
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *group;
|
|
|
|
int i;
|
|
|
|
GtkWidget *group_button = NULL;
|
|
|
|
PangoAttrList *attrs;
|
|
|
|
|
|
|
|
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_widget_set_halign (group, GTK_ALIGN_START);
|
|
|
|
|
|
|
|
label = gtk_label_new (title);
|
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_widget_set_halign (label, GTK_ALIGN_START);
|
|
|
|
g_object_set (label, "margin-top", 10, "margin-bottom", 10, NULL);
|
|
|
|
attrs = pango_attr_list_new ();
|
|
|
|
pango_attr_list_insert (attrs, pango_attr_weight_new (PANGO_WEIGHT_BOLD));
|
|
|
|
gtk_label_set_attributes (GTK_LABEL (label), attrs);
|
|
|
|
pango_attr_list_unref (attrs);
|
|
|
|
gtk_container_add (GTK_CONTAINER (group), label);
|
|
|
|
|
|
|
|
for (i = 0; tags[i]; i++)
|
|
|
|
{
|
|
|
|
unsigned int tag;
|
|
|
|
GtkWidget *feat;
|
|
|
|
FeatureItem *item;
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
tag = hb_tag_from_string (tags[i], -1);
|
|
|
|
name = get_feature_display_name (tag);
|
|
|
|
|
|
|
|
feat = gtk_radio_button_new_with_label_from_widget (GTK_RADIO_BUTTON (group_button),
|
|
|
|
name ? name : _("Default"));
|
|
|
|
if (group_button == NULL)
|
|
|
|
group_button = feat;
|
|
|
|
|
|
|
|
g_signal_connect (feat, "notify::active", G_CALLBACK (update_display), NULL);
|
|
|
|
g_object_set_data (G_OBJECT (feat), "default", group_button);
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (group), feat);
|
|
|
|
|
|
|
|
item = g_new (FeatureItem, 1);
|
|
|
|
item->name = tags[i];
|
|
|
|
item->tag = tag;
|
|
|
|
item->icon = NULL;
|
|
|
|
item->dflt = NULL;
|
|
|
|
item->feat = feat;
|
|
|
|
|
|
|
|
feature_items = g_list_prepend (feature_items, item);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (box), group);
|
|
|
|
}
|
2015-06-19 13:28:09 +00:00
|
|
|
|
|
|
|
static void
|
2016-01-25 05:40:49 +00:00
|
|
|
update_display (void)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
|
|
|
GString *s;
|
|
|
|
const char *text;
|
|
|
|
gboolean has_feature;
|
2016-01-25 05:40:49 +00:00
|
|
|
GtkTreeIter iter;
|
2018-01-03 19:23:33 +00:00
|
|
|
GtkTreeModel *model;
|
|
|
|
PangoFontDescription *desc;
|
|
|
|
GList *l;
|
|
|
|
PangoAttrList *attrs;
|
|
|
|
PangoAttribute *attr;
|
|
|
|
gint ins, bound;
|
|
|
|
guint start, end;
|
|
|
|
PangoLanguage *lang;
|
|
|
|
char *font_desc;
|
|
|
|
char *features;
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2020-03-07 14:01:21 +00:00
|
|
|
text = gtk_editable_get_text (GTK_EDITABLE (the_entry));
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2020-03-07 14:01:21 +00:00
|
|
|
if (gtk_label_get_selection_bounds (GTK_LABEL (the_label), &ins, &bound))
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
|
|
|
start = g_utf8_offset_to_pointer (text, ins) - text;
|
|
|
|
end = g_utf8_offset_to_pointer (text, bound) - text;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
start = PANGO_ATTR_INDEX_FROM_TEXT_BEGINNING;
|
|
|
|
end = PANGO_ATTR_INDEX_TO_TEXT_END;
|
|
|
|
}
|
|
|
|
|
|
|
|
desc = gtk_font_chooser_get_font_desc (GTK_FONT_CHOOSER (font));
|
|
|
|
|
|
|
|
s = g_string_new ("");
|
|
|
|
add_font_variations (s);
|
|
|
|
if (s->len > 0)
|
|
|
|
{
|
|
|
|
pango_font_description_set_variations (desc, s->str);
|
|
|
|
g_string_free (s, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
font_desc = pango_font_description_to_string (desc);
|
2015-06-19 13:28:09 +00:00
|
|
|
|
|
|
|
s = g_string_new ("");
|
|
|
|
|
|
|
|
has_feature = FALSE;
|
2018-01-03 19:23:33 +00:00
|
|
|
for (l = feature_items; l; l = l->next)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
FeatureItem *item = l->data;
|
|
|
|
|
|
|
|
if (!gtk_widget_is_sensitive (item->feat))
|
2015-06-19 13:28:09 +00:00
|
|
|
continue;
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
if (GTK_IS_RADIO_BUTTON (item->feat))
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (item->feat)) &&
|
|
|
|
strcmp (item->name, "xxxx") != 0)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
|
|
|
if (has_feature)
|
|
|
|
g_string_append (s, ", ");
|
2018-01-03 19:23:33 +00:00
|
|
|
g_string_append (s, item->name);
|
2015-06-19 13:28:09 +00:00
|
|
|
g_string_append (s, " 1");
|
|
|
|
has_feature = TRUE;
|
|
|
|
}
|
|
|
|
}
|
2018-01-03 19:23:33 +00:00
|
|
|
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
if (gtk_check_button_get_inconsistent (GTK_CHECK_BUTTON (item->feat)))
|
|
|
|
continue;
|
|
|
|
|
2015-06-19 13:28:09 +00:00
|
|
|
if (has_feature)
|
|
|
|
g_string_append (s, ", ");
|
2018-01-03 19:23:33 +00:00
|
|
|
g_string_append (s, item->name);
|
|
|
|
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (item->feat)))
|
2015-06-19 13:28:09 +00:00
|
|
|
g_string_append (s, " 1");
|
|
|
|
else
|
|
|
|
g_string_append (s, " 0");
|
|
|
|
has_feature = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
features = g_string_free (s, FALSE);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (script_lang), &iter))
|
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
hb_tag_t lang_tag;
|
|
|
|
|
2016-01-25 05:40:49 +00:00
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (script_lang));
|
|
|
|
gtk_tree_model_get (model, &iter,
|
|
|
|
3, &lang_tag,
|
|
|
|
-1);
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
lang = pango_language_from_string (hb_language_to_string (hb_ot_tag_to_language (lang_tag)));
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
lang = NULL;
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
ensure_range (start, end, desc, features, lang);
|
|
|
|
|
|
|
|
attrs = pango_attr_list_new ();
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
for (l = ranges; l; l = l->next)
|
|
|
|
{
|
|
|
|
Range *range = l->data;
|
|
|
|
|
|
|
|
attr = pango_attr_font_desc_new (range->desc);
|
|
|
|
attr->start_index = range->start;
|
|
|
|
attr->end_index = range->end;
|
|
|
|
pango_attr_list_insert (attrs, attr);
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
attr = pango_attr_font_features_new (range->features);
|
|
|
|
attr->start_index = range->start;
|
|
|
|
attr->end_index = range->end;
|
|
|
|
pango_attr_list_insert (attrs, attr);
|
|
|
|
|
|
|
|
if (range->language)
|
|
|
|
{
|
|
|
|
attr = pango_attr_language_new (range->language);
|
|
|
|
attr->start_index = range->start;
|
|
|
|
attr->end_index = range->end;
|
|
|
|
pango_attr_list_insert (attrs, attr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_label_set_text (GTK_LABEL (description), font_desc);
|
|
|
|
gtk_label_set_text (GTK_LABEL (settings), features);
|
2020-03-07 14:01:21 +00:00
|
|
|
gtk_label_set_text (GTK_LABEL (the_label), text);
|
|
|
|
gtk_label_set_attributes (GTK_LABEL (the_label), attrs);
|
2015-06-19 13:28:09 +00:00
|
|
|
|
|
|
|
g_free (font_desc);
|
2018-01-03 19:23:33 +00:00
|
|
|
pango_font_description_free (desc);
|
|
|
|
g_free (features);
|
|
|
|
pango_attr_list_unref (attrs);
|
2015-06-19 13:28:09 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 05:40:49 +00:00
|
|
|
static PangoFont *
|
|
|
|
get_pango_font (void)
|
|
|
|
{
|
|
|
|
PangoFontDescription *desc;
|
|
|
|
PangoContext *context;
|
|
|
|
|
|
|
|
desc = gtk_font_chooser_get_font_desc (GTK_FONT_CHOOSER (font));
|
|
|
|
context = gtk_widget_get_pango_context (font);
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
return pango_context_load_font (context, desc);
|
|
|
|
}
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
hb_tag_t script_tag;
|
|
|
|
hb_tag_t lang_tag;
|
|
|
|
unsigned int script_index;
|
|
|
|
unsigned int lang_index;
|
|
|
|
} TagPair;
|
|
|
|
|
|
|
|
static guint
|
|
|
|
tag_pair_hash (gconstpointer data)
|
|
|
|
{
|
|
|
|
const TagPair *pair = data;
|
|
|
|
|
|
|
|
return pair->script_tag + pair->lang_tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
tag_pair_equal (gconstpointer a, gconstpointer b)
|
|
|
|
{
|
|
|
|
const TagPair *pair_a = a;
|
|
|
|
const TagPair *pair_b = b;
|
|
|
|
|
|
|
|
return pair_a->script_tag == pair_b->script_tag && pair_a->lang_tag == pair_b->lang_tag;
|
|
|
|
}
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
static int
|
|
|
|
script_sort_func (GtkTreeModel *model,
|
|
|
|
GtkTreeIter *a,
|
|
|
|
GtkTreeIter *b,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
char *sa, *sb;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, a, 0, &sa, -1);
|
|
|
|
gtk_tree_model_get (model, b, 0, &sb, -1);
|
|
|
|
|
|
|
|
ret = strcmp (sa, sb);
|
|
|
|
|
|
|
|
g_free (sa);
|
|
|
|
g_free (sb);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-01-25 05:40:49 +00:00
|
|
|
static void
|
|
|
|
update_script_combo (void)
|
|
|
|
{
|
|
|
|
GtkListStore *store;
|
|
|
|
hb_font_t *hb_font;
|
2018-01-03 19:23:33 +00:00
|
|
|
gint i, j, k;
|
2016-01-25 05:40:49 +00:00
|
|
|
PangoFont *pango_font;
|
|
|
|
GHashTable *tags;
|
|
|
|
GHashTableIter iter;
|
|
|
|
TagPair *pair;
|
2018-01-03 19:23:33 +00:00
|
|
|
char *lang;
|
|
|
|
hb_tag_t active;
|
|
|
|
GtkTreeIter active_iter;
|
|
|
|
gboolean have_active = FALSE;
|
|
|
|
|
|
|
|
lang = gtk_font_chooser_get_language (GTK_FONT_CHOOSER (font));
|
2019-04-12 17:42:38 +00:00
|
|
|
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
2018-01-03 19:23:33 +00:00
|
|
|
active = hb_ot_tag_from_language (hb_language_from_string (lang, -1));
|
2019-04-12 17:42:38 +00:00
|
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
g_free (lang);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
store = gtk_list_store_new (4, G_TYPE_STRING, G_TYPE_UINT, G_TYPE_UINT, G_TYPE_UINT);
|
|
|
|
|
|
|
|
pango_font = get_pango_font ();
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_font = pango_font_get_hb_font (pango_font);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
tags = g_hash_table_new_full (tag_pair_hash, tag_pair_equal, g_free, NULL);
|
|
|
|
|
|
|
|
pair = g_new (TagPair, 1);
|
|
|
|
pair->script_tag = HB_OT_TAG_DEFAULT_SCRIPT;
|
|
|
|
pair->lang_tag = HB_OT_TAG_DEFAULT_LANGUAGE;
|
2018-01-03 19:23:33 +00:00
|
|
|
g_hash_table_add (tags, pair);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
if (hb_font)
|
|
|
|
{
|
|
|
|
hb_tag_t tables[2] = { HB_OT_TAG_GSUB, HB_OT_TAG_GPOS };
|
|
|
|
hb_face_t *hb_face;
|
|
|
|
|
|
|
|
hb_face = hb_font_get_face (hb_font);
|
|
|
|
|
|
|
|
for (i= 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
hb_tag_t scripts[80];
|
|
|
|
unsigned int script_count = G_N_ELEMENTS (scripts);
|
|
|
|
|
|
|
|
hb_ot_layout_table_get_script_tags (hb_face, tables[i], 0, &script_count, scripts);
|
|
|
|
for (j = 0; j < script_count; j++)
|
|
|
|
{
|
|
|
|
hb_tag_t languages[80];
|
|
|
|
unsigned int language_count = G_N_ELEMENTS (languages);
|
|
|
|
|
|
|
|
hb_ot_layout_script_get_language_tags (hb_face, tables[i], j, 0, &language_count, languages);
|
|
|
|
for (k = 0; k < language_count; k++)
|
|
|
|
{
|
|
|
|
pair = g_new (TagPair, 1);
|
|
|
|
pair->script_tag = scripts[j];
|
|
|
|
pair->lang_tag = languages[k];
|
|
|
|
pair->script_index = j;
|
|
|
|
pair->lang_index = k;
|
|
|
|
g_hash_table_add (tags, pair);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (pango_font);
|
|
|
|
|
|
|
|
g_hash_table_iter_init (&iter, tags);
|
|
|
|
while (g_hash_table_iter_next (&iter, (gpointer *)&pair, NULL))
|
|
|
|
{
|
|
|
|
const char *langname;
|
|
|
|
char langbuf[5];
|
2020-03-07 14:01:21 +00:00
|
|
|
GtkTreeIter tree_iter;
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
if (pair->lang_tag == HB_OT_TAG_DEFAULT_LANGUAGE)
|
2018-01-03 19:23:33 +00:00
|
|
|
langname = NC_("Language", "Default");
|
2016-01-25 05:40:49 +00:00
|
|
|
else
|
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
langname = get_language_name_for_tag (pair->lang_tag);
|
|
|
|
if (!langname)
|
2016-01-25 05:40:49 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
hb_tag_to_string (pair->lang_tag, langbuf);
|
|
|
|
langbuf[4] = 0;
|
|
|
|
langname = langbuf;
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-07 14:01:21 +00:00
|
|
|
gtk_list_store_insert_with_values (store, &tree_iter, -1,
|
2018-01-03 19:23:33 +00:00
|
|
|
0, langname,
|
2016-01-25 05:40:49 +00:00
|
|
|
1, pair->script_index,
|
|
|
|
2, pair->lang_index,
|
|
|
|
3, pair->lang_tag,
|
|
|
|
-1);
|
2018-01-03 19:23:33 +00:00
|
|
|
if (pair->lang_tag == active)
|
|
|
|
{
|
|
|
|
have_active = TRUE;
|
2020-03-07 14:01:21 +00:00
|
|
|
active_iter = tree_iter;
|
2018-01-03 19:23:33 +00:00
|
|
|
}
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_hash_table_destroy (tags);
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (store),
|
|
|
|
script_sort_func, NULL, NULL);
|
|
|
|
gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
|
|
|
|
GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
|
|
|
|
GTK_SORT_ASCENDING);
|
2016-01-25 05:40:49 +00:00
|
|
|
gtk_combo_box_set_model (GTK_COMBO_BOX (script_lang), GTK_TREE_MODEL (store));
|
2018-01-03 19:23:33 +00:00
|
|
|
if (have_active)
|
|
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (script_lang), &active_iter);
|
|
|
|
else
|
|
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (script_lang), 0);
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
update_features (void)
|
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
gint i, j;
|
2016-01-25 05:40:49 +00:00
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
guint script_index, lang_index;
|
|
|
|
PangoFont *pango_font;
|
|
|
|
hb_font_t *hb_font;
|
2018-01-03 19:23:33 +00:00
|
|
|
GList *l;
|
2016-01-25 05:40:49 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
for (l = feature_items; l; l = l->next)
|
|
|
|
{
|
|
|
|
FeatureItem *item = l->data;
|
|
|
|
gtk_widget_hide (item->feat);
|
|
|
|
gtk_widget_hide (gtk_widget_get_parent (item->feat));
|
|
|
|
if (strcmp (item->name, "xxxx") == 0)
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (item->feat), TRUE);
|
|
|
|
}
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
/* set feature presence checks from the font features */
|
|
|
|
|
|
|
|
if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (script_lang), &iter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (script_lang));
|
|
|
|
gtk_tree_model_get (model, &iter,
|
|
|
|
1, &script_index,
|
|
|
|
2, &lang_index,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
pango_font = get_pango_font ();
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_font = pango_font_get_hb_font (pango_font);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
if (hb_font)
|
|
|
|
{
|
|
|
|
hb_tag_t tables[2] = { HB_OT_TAG_GSUB, HB_OT_TAG_GPOS };
|
|
|
|
hb_face_t *hb_face;
|
2018-01-03 19:23:33 +00:00
|
|
|
char *feat;
|
2016-01-25 05:40:49 +00:00
|
|
|
|
|
|
|
hb_face = hb_font_get_face (hb_font);
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
hb_tag_t features[80];
|
|
|
|
unsigned int count = G_N_ELEMENTS(features);
|
|
|
|
|
|
|
|
hb_ot_layout_language_get_feature_tags (hb_face,
|
|
|
|
tables[i],
|
|
|
|
script_index,
|
|
|
|
lang_index,
|
|
|
|
0,
|
|
|
|
&count,
|
|
|
|
features);
|
|
|
|
|
|
|
|
for (j = 0; j < count; j++)
|
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
#if 0
|
|
|
|
char buf[5];
|
|
|
|
hb_tag_to_string (features[j], buf);
|
|
|
|
buf[4] = 0;
|
|
|
|
g_print ("%s present in %s\n", buf, i == 0 ? "GSUB" : "GPOS");
|
|
|
|
#endif
|
|
|
|
for (l = feature_items; l; l = l->next)
|
2016-01-25 05:40:49 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
FeatureItem *item = l->data;
|
|
|
|
|
|
|
|
if (item->tag == features[j])
|
|
|
|
{
|
|
|
|
gtk_widget_show (item->feat);
|
|
|
|
gtk_widget_show (gtk_widget_get_parent (item->feat));
|
|
|
|
if (GTK_IS_RADIO_BUTTON (item->feat))
|
|
|
|
{
|
|
|
|
GtkWidget *def = GTK_WIDGET (g_object_get_data (G_OBJECT (item->feat), "default"));
|
|
|
|
gtk_widget_show (def);
|
|
|
|
}
|
|
|
|
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
|
|
|
{
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (item->feat), TRUE);
|
|
|
|
}
|
|
|
|
}
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
feat = gtk_font_chooser_get_font_features (GTK_FONT_CHOOSER (font));
|
|
|
|
if (feat)
|
|
|
|
{
|
|
|
|
for (l = feature_items; l; l = l->next)
|
|
|
|
{
|
|
|
|
FeatureItem *item = l->data;
|
|
|
|
char buf[5];
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
hb_tag_to_string (item->tag, buf);
|
|
|
|
buf[4] = 0;
|
|
|
|
|
|
|
|
p = strstr (feat, buf);
|
|
|
|
if (p)
|
|
|
|
{
|
|
|
|
if (GTK_IS_RADIO_BUTTON (item->feat))
|
|
|
|
{
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (item->feat), p[6] == '1');
|
|
|
|
}
|
|
|
|
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
|
|
|
{
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (item->feat), FALSE);
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (item->feat), p[6] == '1');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (feat);
|
|
|
|
}
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (pango_font);
|
|
|
|
}
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
#define FixedToFloat(f) (((float)(f))/65536.0)
|
|
|
|
|
|
|
|
static void
|
|
|
|
adjustment_changed (GtkAdjustment *adjustment,
|
|
|
|
GtkEntry *entry)
|
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
|
|
|
|
str = g_strdup_printf ("%g", gtk_adjustment_get_value (adjustment));
|
2019-02-28 19:31:36 +00:00
|
|
|
gtk_editable_set_text (GTK_EDITABLE (entry), str);
|
2018-01-03 19:23:33 +00:00
|
|
|
g_free (str);
|
|
|
|
|
|
|
|
update_display ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
entry_activated (GtkEntry *entry,
|
|
|
|
GtkAdjustment *adjustment)
|
|
|
|
{
|
|
|
|
gdouble value;
|
|
|
|
gchar *err = NULL;
|
|
|
|
|
2019-02-28 19:31:36 +00:00
|
|
|
value = g_strtod (gtk_editable_get_text (GTK_EDITABLE (entry)), &err);
|
2018-01-03 19:23:33 +00:00
|
|
|
if (err != NULL)
|
|
|
|
gtk_adjustment_set_value (adjustment, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unset_instance (GtkAdjustment *adjustment);
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
guint32 tag;
|
|
|
|
GtkAdjustment *adjustment;
|
|
|
|
} Axis;
|
|
|
|
|
|
|
|
static GHashTable *axes;
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_font_variations (GString *s)
|
|
|
|
{
|
|
|
|
GHashTableIter iter;
|
|
|
|
Axis *axis;
|
|
|
|
char buf[G_ASCII_DTOSTR_BUF_SIZE];
|
2020-03-07 14:01:21 +00:00
|
|
|
const char *sep = "";
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
g_hash_table_iter_init (&iter, axes);
|
|
|
|
while (g_hash_table_iter_next (&iter, (gpointer *)NULL, (gpointer *)&axis))
|
|
|
|
{
|
|
|
|
char tag[5];
|
|
|
|
double value;
|
|
|
|
|
|
|
|
hb_tag_to_string (axis->tag, tag);
|
|
|
|
tag[4] = '\0';
|
|
|
|
value = gtk_adjustment_get_value (axis->adjustment);
|
|
|
|
|
|
|
|
g_string_append_printf (s, "%s%s=%s", sep, tag, g_ascii_dtostr (buf, sizeof (buf), value));
|
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static guint
|
|
|
|
axes_hash (gconstpointer v)
|
|
|
|
{
|
|
|
|
const Axis *p = v;
|
|
|
|
|
|
|
|
return p->tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
axes_equal (gconstpointer v1, gconstpointer v2)
|
|
|
|
{
|
|
|
|
const Axis *p1 = v1;
|
|
|
|
const Axis *p2 = v2;
|
|
|
|
|
|
|
|
return p1->tag == p2->tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2019-07-30 20:32:34 +00:00
|
|
|
add_axis (hb_face_t *hb_face,
|
|
|
|
hb_ot_var_axis_info_t *ax,
|
|
|
|
float value,
|
|
|
|
int i)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
|
|
|
GtkWidget *axis_label;
|
|
|
|
GtkWidget *axis_entry;
|
|
|
|
GtkWidget *axis_scale;
|
|
|
|
GtkAdjustment *adjustment;
|
|
|
|
Axis *axis;
|
2019-07-30 20:32:34 +00:00
|
|
|
char name[20];
|
|
|
|
unsigned int name_len = 20;
|
|
|
|
|
|
|
|
hb_ot_name_get_utf8 (hb_face, ax->name_id, HB_LANGUAGE_INVALID, &name_len, name);
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
axis_label = gtk_label_new (name);
|
2018-01-03 19:23:33 +00:00
|
|
|
gtk_widget_set_halign (axis_label, GTK_ALIGN_START);
|
|
|
|
gtk_widget_set_valign (axis_label, GTK_ALIGN_BASELINE);
|
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), axis_label, 0, i, 1, 1);
|
2019-07-30 20:32:34 +00:00
|
|
|
adjustment = gtk_adjustment_new (value, ax->min_value, ax->max_value,
|
2018-01-03 19:23:33 +00:00
|
|
|
1.0, 10.0, 0.0);
|
|
|
|
axis_scale = gtk_scale_new (GTK_ORIENTATION_HORIZONTAL, adjustment);
|
2019-07-30 20:32:34 +00:00
|
|
|
gtk_scale_add_mark (GTK_SCALE (axis_scale), ax->default_value, GTK_POS_TOP, NULL);
|
2018-01-03 19:23:33 +00:00
|
|
|
gtk_widget_set_valign (axis_scale, GTK_ALIGN_BASELINE);
|
|
|
|
gtk_widget_set_hexpand (axis_scale, TRUE);
|
|
|
|
gtk_widget_set_size_request (axis_scale, 100, -1);
|
|
|
|
gtk_scale_set_draw_value (GTK_SCALE (axis_scale), FALSE);
|
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), axis_scale, 1, i, 1, 1);
|
|
|
|
axis_entry = gtk_entry_new ();
|
|
|
|
gtk_widget_set_valign (axis_entry, GTK_ALIGN_BASELINE);
|
2019-02-28 19:31:36 +00:00
|
|
|
gtk_editable_set_width_chars (GTK_EDITABLE (axis_entry), 4);
|
2018-01-03 19:23:33 +00:00
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), axis_entry, 2, i, 1, 1);
|
|
|
|
|
|
|
|
axis = g_new (Axis, 1);
|
|
|
|
axis->tag = ax->tag;
|
|
|
|
axis->adjustment = adjustment;
|
|
|
|
g_hash_table_add (axes, axis);
|
|
|
|
|
|
|
|
adjustment_changed (adjustment, GTK_ENTRY (axis_entry));
|
|
|
|
|
|
|
|
g_signal_connect (adjustment, "value-changed", G_CALLBACK (adjustment_changed), axis_entry);
|
|
|
|
g_signal_connect (adjustment, "value-changed", G_CALLBACK (unset_instance), NULL);
|
|
|
|
g_signal_connect (axis_entry, "activate", G_CALLBACK (entry_activated), adjustment);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
char *name;
|
2019-07-30 20:32:34 +00:00
|
|
|
unsigned int index;
|
2018-01-03 19:23:33 +00:00
|
|
|
} Instance;
|
|
|
|
|
|
|
|
static guint
|
|
|
|
instance_hash (gconstpointer v)
|
|
|
|
{
|
|
|
|
const Instance *p = v;
|
|
|
|
|
|
|
|
return g_str_hash (p->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
instance_equal (gconstpointer v1, gconstpointer v2)
|
|
|
|
{
|
|
|
|
const Instance *p1 = v1;
|
|
|
|
const Instance *p2 = v2;
|
|
|
|
|
|
|
|
return g_str_equal (p1->name, p2->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_instance (gpointer data)
|
|
|
|
{
|
|
|
|
Instance *instance = data;
|
|
|
|
|
|
|
|
g_free (instance->name);
|
|
|
|
g_free (instance);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GHashTable *instances;
|
|
|
|
|
|
|
|
static void
|
2019-07-30 20:32:34 +00:00
|
|
|
add_instance (hb_face_t *face,
|
|
|
|
unsigned int index,
|
|
|
|
GtkWidget *combo,
|
|
|
|
int pos)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
|
|
|
Instance *instance;
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_ot_name_id_t name_id;
|
|
|
|
char name[20];
|
|
|
|
unsigned int name_len = 20;
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
instance = g_new0 (Instance, 1);
|
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
name_id = hb_ot_var_named_instance_get_subfamily_name_id (face, index);
|
|
|
|
hb_ot_name_get_utf8 (face, name_id, HB_LANGUAGE_INVALID, &name_len, name);
|
|
|
|
|
2019-11-25 21:11:28 +00:00
|
|
|
instance->name = g_strdup (name);
|
2019-07-30 20:32:34 +00:00
|
|
|
instance->index = index;
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
g_hash_table_add (instances, instance);
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo), instance->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
unset_instance (GtkAdjustment *adjustment)
|
|
|
|
{
|
|
|
|
if (instance_combo)
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (instance_combo), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
instance_changed (GtkComboBox *combo)
|
|
|
|
{
|
|
|
|
char *text;
|
|
|
|
Instance *instance;
|
|
|
|
Instance ikey;
|
|
|
|
int i;
|
2019-07-30 20:32:34 +00:00
|
|
|
unsigned int coords_length;
|
|
|
|
float *coords = NULL;
|
|
|
|
hb_ot_var_axis_info_t *ai = NULL;
|
|
|
|
unsigned int n_axes;
|
|
|
|
PangoFont *pango_font = NULL;
|
|
|
|
hb_font_t *hb_font;
|
|
|
|
hb_face_t *hb_face;
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
text = gtk_combo_box_text_get_active_text (GTK_COMBO_BOX_TEXT (combo));
|
|
|
|
if (text[0] == '\0')
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ikey.name = text;
|
|
|
|
instance = g_hash_table_lookup (instances, &ikey);
|
|
|
|
if (!instance)
|
|
|
|
{
|
|
|
|
g_print ("did not find instance %s\n", text);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
pango_font = get_pango_font ();
|
|
|
|
hb_font = pango_font_get_hb_font (pango_font);
|
|
|
|
hb_face = hb_font_get_face (hb_font);
|
|
|
|
|
|
|
|
n_axes = hb_ot_var_get_axis_infos (hb_face, 0, NULL, NULL);
|
|
|
|
ai = g_new (hb_ot_var_axis_info_t, n_axes);
|
|
|
|
hb_ot_var_get_axis_infos (hb_face, 0, &n_axes, ai);
|
|
|
|
|
|
|
|
coords = g_new (float, n_axes);
|
|
|
|
hb_ot_var_named_instance_get_design_coords (hb_face,
|
|
|
|
instance->index,
|
|
|
|
&coords_length,
|
|
|
|
coords);
|
|
|
|
|
|
|
|
for (i = 0; i < n_axes; i++)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
|
|
|
Axis *axis;
|
|
|
|
Axis akey;
|
|
|
|
gdouble value;
|
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
value = coords[ai[i].axis_index];
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
akey.tag = ai[i].tag;
|
2018-01-03 19:23:33 +00:00
|
|
|
axis = g_hash_table_lookup (axes, &akey);
|
|
|
|
if (axis)
|
|
|
|
{
|
|
|
|
g_signal_handlers_block_by_func (axis->adjustment, unset_instance, NULL);
|
|
|
|
gtk_adjustment_set_value (axis->adjustment, value);
|
|
|
|
g_signal_handlers_unblock_by_func (axis->adjustment, unset_instance, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
g_free (text);
|
2019-07-30 20:32:34 +00:00
|
|
|
g_clear_object (&pango_font);
|
|
|
|
g_free (ai);
|
|
|
|
g_free (coords);
|
2018-01-03 19:23:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2019-07-30 20:32:34 +00:00
|
|
|
matches_instance (hb_face_t *hb_face,
|
|
|
|
unsigned int index,
|
|
|
|
unsigned int n_axes,
|
|
|
|
float *coords)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
2019-07-30 20:32:34 +00:00
|
|
|
float *instance_coords;
|
|
|
|
unsigned int coords_length;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
instance_coords = g_new (float, n_axes);
|
|
|
|
coords_length = n_axes;
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_ot_var_named_instance_get_design_coords (hb_face,
|
|
|
|
index,
|
|
|
|
&coords_length,
|
|
|
|
instance_coords);
|
|
|
|
|
|
|
|
for (i = 0; i < n_axes; i++)
|
|
|
|
if (instance_coords[i] != coords[i])
|
2018-01-03 19:23:33 +00:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_font_plane (int i)
|
|
|
|
{
|
|
|
|
GtkWidget *plane;
|
|
|
|
Axis *weight_axis;
|
|
|
|
Axis *width_axis;
|
|
|
|
|
|
|
|
Axis key;
|
|
|
|
|
|
|
|
key.tag = MAKE_TAG('w','g','h','t');
|
|
|
|
weight_axis = g_hash_table_lookup (axes, &key);
|
|
|
|
key.tag = MAKE_TAG('w','d','t','h');
|
|
|
|
width_axis = g_hash_table_lookup (axes, &key);
|
|
|
|
|
|
|
|
if (weight_axis && width_axis)
|
|
|
|
{
|
|
|
|
plane = gtk_font_plane_new (weight_axis->adjustment,
|
|
|
|
width_axis->adjustment);
|
|
|
|
|
|
|
|
gtk_widget_set_size_request (plane, 300, 300);
|
|
|
|
gtk_widget_set_halign (plane, GTK_ALIGN_CENTER);
|
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), plane, 0, i, 3, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
/* FIXME: This doesn't work if the font has an avar table */
|
|
|
|
static float
|
|
|
|
denorm_coord (hb_ot_var_axis_info_t *axis, int coord)
|
|
|
|
{
|
|
|
|
float r = coord / 16384.0;
|
|
|
|
|
|
|
|
if (coord < 0)
|
|
|
|
return axis->default_value + r * (axis->default_value - axis->min_value);
|
|
|
|
else
|
|
|
|
return axis->default_value + r * (axis->max_value - axis->default_value);
|
|
|
|
}
|
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
static void
|
|
|
|
update_font_variations (void)
|
|
|
|
{
|
|
|
|
GtkWidget *child, *next;
|
2019-07-30 20:32:34 +00:00
|
|
|
PangoFont *pango_font = NULL;
|
|
|
|
hb_font_t *hb_font;
|
|
|
|
hb_face_t *hb_face;
|
|
|
|
unsigned int n_axes;
|
|
|
|
hb_ot_var_axis_info_t *ai = NULL;
|
|
|
|
float *design_coords = NULL;
|
|
|
|
const int *coords;
|
|
|
|
unsigned int length;
|
|
|
|
int i;
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
child = gtk_widget_get_first_child (variations_grid);
|
|
|
|
while (child != NULL)
|
|
|
|
{
|
|
|
|
next = gtk_widget_get_next_sibling (child);
|
|
|
|
gtk_widget_destroy (child);
|
|
|
|
child = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
instance_combo = NULL;
|
|
|
|
|
|
|
|
g_hash_table_remove_all (axes);
|
|
|
|
g_hash_table_remove_all (instances);
|
|
|
|
|
|
|
|
pango_font = get_pango_font ();
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_font = pango_font_get_hb_font (pango_font);
|
|
|
|
hb_face = hb_font_get_face (hb_font);
|
|
|
|
|
|
|
|
n_axes = hb_ot_var_get_axis_infos (hb_face, 0, NULL, NULL);
|
|
|
|
if (n_axes == 0)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
ai = g_new0 (hb_ot_var_axis_info_t, n_axes);
|
|
|
|
design_coords = g_new (float, n_axes);
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
hb_ot_var_get_axis_infos (hb_face, 0, &n_axes, ai);
|
|
|
|
coords = hb_font_get_var_coords_normalized (hb_font, &length);
|
|
|
|
for (i = 0; i < length; i++)
|
|
|
|
design_coords[i] = denorm_coord (&ai[i], coords[i]);
|
|
|
|
|
|
|
|
if (hb_ot_var_get_named_instance_count (hb_face) > 0)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
2019-07-30 20:32:34 +00:00
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *combo;
|
|
|
|
|
|
|
|
label = gtk_label_new ("Instance");
|
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0);
|
|
|
|
gtk_widget_set_halign (label, GTK_ALIGN_START);
|
|
|
|
gtk_widget_set_valign (label, GTK_ALIGN_BASELINE);
|
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), label, 0, -1, 2, 1);
|
|
|
|
|
|
|
|
combo = gtk_combo_box_text_new ();
|
|
|
|
gtk_widget_set_valign (combo, GTK_ALIGN_BASELINE);
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo), "");
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
for (i = 0; i < hb_ot_var_get_named_instance_count (hb_face); i++)
|
|
|
|
add_instance (hb_face, i, combo, i);
|
|
|
|
|
|
|
|
for (i = 0; i < hb_ot_var_get_named_instance_count (hb_face); i++)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
2019-07-30 20:32:34 +00:00
|
|
|
if (matches_instance (hb_face, i, n_axes, design_coords))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo), i + 1);
|
|
|
|
break;
|
|
|
|
}
|
2018-01-03 19:23:33 +00:00
|
|
|
}
|
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
gtk_grid_attach (GTK_GRID (variations_grid), combo, 1, -1, 2, 1);
|
|
|
|
g_signal_connect (combo, "changed", G_CALLBACK (instance_changed), NULL);
|
|
|
|
instance_combo = combo;
|
|
|
|
}
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
for (i = 0; i < n_axes; i++)
|
|
|
|
add_axis (hb_face, &ai[i], design_coords[i], i);
|
2018-01-03 19:23:33 +00:00
|
|
|
|
2019-07-30 20:32:34 +00:00
|
|
|
add_font_plane (n_axes);
|
|
|
|
|
|
|
|
done:
|
|
|
|
g_clear_object (&pango_font);
|
|
|
|
g_free (ai);
|
|
|
|
g_free (design_coords);
|
2018-01-03 19:23:33 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
void
|
|
|
|
font_features_font_changed (void)
|
2016-01-25 05:40:49 +00:00
|
|
|
{
|
|
|
|
update_script_combo ();
|
2018-01-03 19:23:33 +00:00
|
|
|
update_features ();
|
|
|
|
update_font_variations ();
|
2016-01-25 05:40:49 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
void
|
|
|
|
font_features_script_changed (void)
|
2016-01-25 05:40:49 +00:00
|
|
|
{
|
|
|
|
update_features ();
|
|
|
|
update_display ();
|
|
|
|
}
|
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
void
|
|
|
|
font_features_reset_features (void)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
GList *l;
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2020-03-07 14:01:21 +00:00
|
|
|
gtk_label_select_region (GTK_LABEL (the_label), 0, 0);
|
2018-01-03 19:23:33 +00:00
|
|
|
|
|
|
|
g_list_free_full (ranges, free_range);
|
|
|
|
ranges = NULL;
|
|
|
|
|
|
|
|
for (l = feature_items; l; l = l->next)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
FeatureItem *item = l->data;
|
|
|
|
|
|
|
|
if (GTK_IS_RADIO_BUTTON (item->feat))
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2018-01-03 19:23:33 +00:00
|
|
|
if (strcmp (item->name, "xxxx") == 0)
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (item->feat), TRUE);
|
|
|
|
}
|
|
|
|
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
|
|
|
{
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (item->feat), FALSE);
|
|
|
|
set_inconsistent (GTK_CHECK_BUTTON (item->feat), TRUE);
|
2015-06-19 13:28:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *text;
|
|
|
|
|
|
|
|
static void
|
|
|
|
switch_to_entry (void)
|
|
|
|
{
|
2020-03-07 14:01:21 +00:00
|
|
|
text = g_strdup (gtk_editable_get_text (GTK_EDITABLE (the_entry)));
|
2015-06-19 13:28:09 +00:00
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (stack), "entry");
|
2020-03-07 14:01:21 +00:00
|
|
|
gtk_widget_grab_focus (the_entry);
|
2015-06-19 13:28:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
switch_to_label (void)
|
|
|
|
{
|
|
|
|
g_free (text);
|
|
|
|
text = NULL;
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (stack), "label");
|
2016-01-25 05:40:49 +00:00
|
|
|
update_display ();
|
2015-06-19 13:28:09 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
void
|
|
|
|
font_features_toggle_edit (void)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
|
|
|
if (strcmp (gtk_stack_get_visible_child_name (GTK_STACK (stack)), "label") == 0)
|
|
|
|
switch_to_entry ();
|
|
|
|
else
|
|
|
|
switch_to_label ();
|
|
|
|
}
|
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
void
|
|
|
|
font_features_stop_edit (void)
|
2018-01-03 19:23:33 +00:00
|
|
|
{
|
2019-05-18 18:22:12 +00:00
|
|
|
g_signal_emit_by_name (edit_toggle, "clicked");
|
2018-01-03 19:23:33 +00:00
|
|
|
}
|
|
|
|
|
2015-06-19 13:28:09 +00:00
|
|
|
static gboolean
|
2018-01-31 11:42:43 +00:00
|
|
|
entry_key_press (GtkEventController *controller,
|
|
|
|
guint keyval,
|
|
|
|
guint keycode,
|
|
|
|
GdkModifierType modifiers,
|
|
|
|
GtkEntry *entry)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2017-08-28 22:47:03 +00:00
|
|
|
if (keyval == GDK_KEY_Escape)
|
2015-06-19 13:28:09 +00:00
|
|
|
{
|
2019-02-28 19:31:36 +00:00
|
|
|
gtk_editable_set_text (GTK_EDITABLE (entry), text);
|
2019-11-29 07:48:48 +00:00
|
|
|
font_features_stop_edit ();
|
2015-06-19 13:28:09 +00:00
|
|
|
return GDK_EVENT_STOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return GDK_EVENT_PROPAGATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
do_font_features (GtkWidget *do_widget)
|
|
|
|
{
|
|
|
|
static GtkWidget *window = NULL;
|
|
|
|
|
|
|
|
if (!window)
|
|
|
|
{
|
|
|
|
GtkBuilder *builder;
|
2018-01-03 19:23:33 +00:00
|
|
|
GtkWidget *feature_list;
|
2018-01-31 11:42:43 +00:00
|
|
|
GtkEventController *controller;
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
builder = gtk_builder_new_from_resource ("/font_features/font-features.ui");
|
2015-06-19 13:28:09 +00:00
|
|
|
|
|
|
|
window = GTK_WIDGET (gtk_builder_get_object (builder, "window"));
|
2018-01-03 19:23:33 +00:00
|
|
|
feature_list = GTK_WIDGET (gtk_builder_get_object (builder, "feature_list"));
|
2020-03-07 14:01:21 +00:00
|
|
|
the_label = GTK_WIDGET (gtk_builder_get_object (builder, "label"));
|
2015-06-19 13:28:09 +00:00
|
|
|
settings = GTK_WIDGET (gtk_builder_get_object (builder, "settings"));
|
2018-01-03 19:23:33 +00:00
|
|
|
description = GTK_WIDGET (gtk_builder_get_object (builder, "description"));
|
2015-06-19 13:28:09 +00:00
|
|
|
resetbutton = GTK_WIDGET (gtk_builder_get_object (builder, "reset"));
|
|
|
|
font = GTK_WIDGET (gtk_builder_get_object (builder, "font"));
|
2016-01-25 05:40:49 +00:00
|
|
|
script_lang = GTK_WIDGET (gtk_builder_get_object (builder, "script_lang"));
|
2015-06-19 13:28:09 +00:00
|
|
|
stack = GTK_WIDGET (gtk_builder_get_object (builder, "stack"));
|
2020-03-07 14:01:21 +00:00
|
|
|
the_entry = GTK_WIDGET (gtk_builder_get_object (builder, "entry"));
|
2018-01-03 19:23:33 +00:00
|
|
|
edit_toggle = GTK_WIDGET (gtk_builder_get_object (builder, "edit_toggle"));
|
|
|
|
|
2018-04-20 17:58:06 +00:00
|
|
|
controller = gtk_event_controller_key_new ();
|
2020-03-07 14:01:21 +00:00
|
|
|
g_object_set_data_full (G_OBJECT (the_entry), "controller", g_object_ref (controller), g_object_unref);
|
|
|
|
g_signal_connect (controller, "key-pressed", G_CALLBACK (entry_key_press), the_entry);
|
|
|
|
gtk_widget_add_controller (the_entry, controller);
|
2018-01-31 11:42:43 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
add_check_group (feature_list, _("Kerning"), (const char *[]){ "kern", NULL });
|
|
|
|
add_check_group (feature_list, _("Ligatures"), (const char *[]){ "liga",
|
|
|
|
"dlig",
|
|
|
|
"hlig",
|
|
|
|
"clig",
|
|
|
|
"rlig", NULL });
|
|
|
|
add_check_group (feature_list, _("Letter Case"), (const char *[]){ "smcp",
|
|
|
|
"c2sc",
|
|
|
|
"pcap",
|
|
|
|
"c2pc",
|
|
|
|
"unic",
|
|
|
|
"cpsp",
|
|
|
|
"case",NULL });
|
|
|
|
add_radio_group (feature_list, _("Number Case"), (const char *[]){ "xxxx",
|
|
|
|
"lnum",
|
|
|
|
"onum", NULL });
|
|
|
|
add_radio_group (feature_list, _("Number Spacing"), (const char *[]){ "xxxx",
|
|
|
|
"pnum",
|
|
|
|
"tnum", NULL });
|
|
|
|
add_radio_group (feature_list, _("Fractions"), (const char *[]){ "xxxx",
|
|
|
|
"frac",
|
|
|
|
"afrc", NULL });
|
|
|
|
add_check_group (feature_list, _("Numeric Extras"), (const char *[]){ "zero",
|
|
|
|
"nalt",
|
|
|
|
"sinf", NULL });
|
|
|
|
add_check_group (feature_list, _("Character Alternatives"), (const char *[]){ "swsh",
|
|
|
|
"cswh",
|
|
|
|
"locl",
|
|
|
|
"calt",
|
|
|
|
"falt",
|
|
|
|
"hist",
|
|
|
|
"salt",
|
|
|
|
"jalt",
|
|
|
|
"titl",
|
|
|
|
"rand",
|
|
|
|
"subs",
|
|
|
|
"sups",
|
|
|
|
"ordn",
|
|
|
|
"ltra",
|
|
|
|
"ltrm",
|
|
|
|
"rtla",
|
|
|
|
"rtlm",
|
|
|
|
"rclt", NULL });
|
|
|
|
add_check_group (feature_list, _("Positional Alternatives"), (const char *[]){ "init",
|
|
|
|
"medi",
|
|
|
|
"med2",
|
|
|
|
"fina",
|
|
|
|
"fin2",
|
|
|
|
"fin3",
|
|
|
|
"isol", NULL });
|
|
|
|
add_check_group (feature_list, _("Width Variants"), (const char *[]){ "fwid",
|
|
|
|
"hwid",
|
|
|
|
"halt",
|
|
|
|
"pwid",
|
|
|
|
"palt",
|
|
|
|
"twid",
|
|
|
|
"qwid", NULL });
|
|
|
|
add_check_group (feature_list, _("Alternative Stylistic Sets"), (const char *[]){ "ss00",
|
|
|
|
"ss01",
|
|
|
|
"ss02",
|
|
|
|
"ss03",
|
|
|
|
"ss04",
|
|
|
|
"ss05",
|
|
|
|
"ss06",
|
|
|
|
"ss07",
|
|
|
|
"ss08",
|
|
|
|
"ss09",
|
|
|
|
"ss10",
|
|
|
|
"ss11",
|
|
|
|
"ss12",
|
|
|
|
"ss13",
|
|
|
|
"ss14",
|
|
|
|
"ss15",
|
|
|
|
"ss16",
|
|
|
|
"ss17",
|
|
|
|
"ss18",
|
|
|
|
"ss19",
|
|
|
|
"ss20", NULL });
|
|
|
|
add_check_group (feature_list, _("Mathematical"), (const char *[]){ "dtls",
|
|
|
|
"flac",
|
|
|
|
"mgrk",
|
|
|
|
"ssty", NULL });
|
|
|
|
add_check_group (feature_list, _("Optical Bounds"), (const char *[]){ "opbd",
|
|
|
|
"lfbd",
|
|
|
|
"rtbd", NULL });
|
|
|
|
feature_items = g_list_reverse (feature_items);
|
|
|
|
|
|
|
|
variations_heading = GTK_WIDGET (gtk_builder_get_object (builder, "variations_heading"));
|
|
|
|
variations_grid = GTK_WIDGET (gtk_builder_get_object (builder, "variations_grid"));
|
|
|
|
if (instances == NULL)
|
|
|
|
instances = g_hash_table_new_full (instance_hash, instance_equal, NULL, free_instance);
|
|
|
|
else
|
|
|
|
g_hash_table_remove_all (instances);
|
2015-06-19 13:28:09 +00:00
|
|
|
|
2018-01-03 19:23:33 +00:00
|
|
|
if (axes == NULL)
|
|
|
|
axes = g_hash_table_new_full (axes_hash, axes_equal, NULL, g_free);
|
|
|
|
else
|
|
|
|
g_hash_table_remove_all (axes);
|
2016-01-25 05:40:49 +00:00
|
|
|
|
2019-11-29 07:48:48 +00:00
|
|
|
font_features_font_changed ();
|
2015-06-20 01:59:15 +00:00
|
|
|
|
|
|
|
g_signal_connect (window, "destroy",
|
|
|
|
G_CALLBACK (gtk_widget_destroyed), &window);
|
2015-06-28 13:23:28 +00:00
|
|
|
|
|
|
|
g_object_unref (builder);
|
2019-12-24 03:35:44 +00:00
|
|
|
|
|
|
|
update_display ();
|
2015-06-19 13:28:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!gtk_widget_get_visible (window))
|
2015-06-28 13:23:28 +00:00
|
|
|
gtk_window_present (GTK_WINDOW (window));
|
2015-06-19 13:28:09 +00:00
|
|
|
else
|
2015-06-28 13:23:28 +00:00
|
|
|
gtk_widget_destroy (window);
|
2015-06-19 13:28:09 +00:00
|
|
|
|
|
|
|
return window;
|
|
|
|
}
|