/* Pango/Font Explorer * * 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. * * If the selected font supports OpenType font variations, then the * axes are also offered for customization. */ #include #include #include #include #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)) typedef struct { unsigned int tag; const char *name; GtkWidget *icon; GtkWidget *dflt; GtkWidget *feat; } FeatureItem; typedef struct { unsigned int start; unsigned int end; PangoFontDescription *desc; char *features; PangoLanguage *language; } Range; typedef struct { guint32 tag; GtkAdjustment *adjustment; double default_value; guint tick_cb; guint64 start_time; gboolean increasing; GtkWidget *button; } Axis; typedef struct { GtkWidget *the_label; GtkWidget *settings; GtkWidget *description; GtkWidget *font; GtkWidget *script_lang; GtkWidget *feature_list; GtkWidget *variations_grid; GtkWidget *instance_combo; GtkWidget *stack; GtkWidget *entry; GtkWidget *plain_toggle; GtkWidget *waterfall_toggle; GtkWidget *edit_toggle; GtkAdjustment *size_adjustment; GtkAdjustment *letterspacing_adjustment; GtkAdjustment *line_height_adjustment; GtkWidget *foreground; GtkWidget *background; GtkWidget *size_scale; GtkWidget *size_entry; GtkWidget *letterspacing_entry; GtkWidget *line_height_entry; GList *feature_items; GList *ranges; GHashTable *instances; GHashTable *axes; char *text; GtkWidget *swin; GtkCssProvider *provider; int sample; } FontFeaturesDemo; static void demo_free (gpointer data) { FontFeaturesDemo *demo = data; g_list_free_full (demo->feature_items, g_free); g_list_free_full (demo->ranges, g_free); g_clear_pointer (&demo->instances, g_hash_table_unref); g_clear_pointer (&demo->axes, g_hash_table_unref); g_clear_pointer (&demo->text, g_free); g_free (demo); } static FontFeaturesDemo *demo; static void update_display (void); static void font_features_toggle_plain (void) { if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (demo->plain_toggle)) || gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (demo->waterfall_toggle))) { gtk_stack_set_visible_child_name (GTK_STACK (demo->stack), "label"); update_display (); } } static void font_features_notify_waterfall (void) { gboolean can_change_size; can_change_size = !gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (demo->waterfall_toggle)); gtk_widget_set_sensitive (demo->size_scale, can_change_size); gtk_widget_set_sensitive (demo->size_entry, can_change_size); } typedef struct { GtkAdjustment *adjustment; GtkEntry *entry; } BasicData; static gboolean update_in_idle (gpointer data) { BasicData *bd = data; char *str; str = g_strdup_printf ("%g", gtk_adjustment_get_value (bd->adjustment)); gtk_editable_set_text (GTK_EDITABLE (bd->entry), str); g_free (str); update_display (); g_free (bd); return G_SOURCE_REMOVE; } static void basic_value_changed (GtkAdjustment *adjustment, gpointer data) { BasicData *bd; bd = g_new (BasicData, 1); bd->adjustment = adjustment; bd->entry = GTK_ENTRY (data); g_idle_add (update_in_idle, bd); } static void basic_entry_activated (GtkEntry *entry, gpointer data) { GtkAdjustment *adjustment = data; double value; char *err = NULL; value = g_strtod (gtk_editable_get_text (GTK_EDITABLE (entry)), &err); if (err != NULL) gtk_adjustment_set_value (adjustment, value); } static void color_set_cb (void) { update_display (); } static void swap_colors (void) { GdkRGBA fg; GdkRGBA bg; gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (demo->foreground), &fg); gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (demo->background), &bg); gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (demo->foreground), &bg); gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (demo->background), &fg); } static void font_features_reset_basic (void) { gtk_adjustment_set_value (demo->size_adjustment, 20); gtk_adjustment_set_value (demo->letterspacing_adjustment, 0); gtk_adjustment_set_value (demo->line_height_adjustment, 1); gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (demo->foreground), &(GdkRGBA){0.,0.,0.,1.}); gtk_color_chooser_set_rgba (GTK_COLOR_CHOOSER (demo->background), &(GdkRGBA){1.,1.,1.,1.}); } static void update_basic (void) { PangoFontDescription *desc; desc = gtk_font_chooser_get_font_desc (GTK_FONT_CHOOSER (demo->font)); gtk_adjustment_set_value (demo->size_adjustment, pango_font_description_get_size (desc) / (double) PANGO_SCALE); } 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 = demo->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; demo->ranges = g_list_insert_sorted (demo->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 char * get_feature_display_name (unsigned int tag) { int i; static char buf[5] = { 0, }; if (tag == MAKE_TAG ('x', 'x', 'x', 'x')) return g_strdup (_("Default")); hb_tag_to_string (tag, buf); if (g_str_has_prefix (buf, "ss") && g_ascii_isdigit (buf[2]) && g_ascii_isdigit (buf[3])) { int num = (buf[2] - '0') * 10 + (buf[3] - '0'); return g_strdup_printf (g_dpgettext2 (NULL, "OpenType layout", "Stylistic Set %d"), num); } else if (g_str_has_prefix (buf, "cv") && g_ascii_isdigit (buf[2]) && g_ascii_isdigit (buf[3])) { int num = (buf[2] - '0') * 10 + (buf[3] - '0'); return g_strdup_printf (g_dpgettext2 (NULL, "OpenType layout", "Character Variant %d"), num); } for (i = 0; i < G_N_ELEMENTS (open_type_layout_features); i++) { if (tag == open_type_layout_features[i].tag) return g_strdup (g_dpgettext2 (NULL, "OpenType layout", open_type_layout_features[i].name)); } g_warning ("unknown OpenType layout feature tag: %s", buf); return g_strdup (buf); } 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_pressed (GtkGestureClick *gesture, int n_press, double x, double y, GtkWidget *feat) { const guint button = gtk_gesture_single_get_current_button (GTK_GESTURE_SINGLE (gesture)); if (button == GDK_BUTTON_PRIMARY) { g_signal_handlers_block_by_func (feat, feat_pressed, NULL); if (gtk_check_button_get_inconsistent (GTK_CHECK_BUTTON (feat))) { set_inconsistent (GTK_CHECK_BUTTON (feat), FALSE); gtk_check_button_set_active (GTK_CHECK_BUTTON (feat), TRUE); } g_signal_handlers_unblock_by_func (feat, feat_pressed, NULL); } else if (button == GDK_BUTTON_SECONDARY) { gboolean inconsistent = gtk_check_button_get_inconsistent (GTK_CHECK_BUTTON (feat)); set_inconsistent (GTK_CHECK_BUTTON (feat), !inconsistent); } } static void feat_toggled_cb (GtkCheckButton *check_button, gpointer data) { set_inconsistent (check_button, FALSE); } static void add_check_group (GtkWidget *box, const char *title, const char **tags) { GtkWidget *label; GtkWidget *group; 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); gtk_widget_add_css_class (label, "heading"); gtk_box_append (GTK_BOX (group), label); for (i = 0; tags[i]; i++) { unsigned int tag; GtkWidget *feat; FeatureItem *item; GtkGesture *gesture; char *name; tag = hb_tag_from_string (tags[i], -1); name = get_feature_display_name (tag); feat = gtk_check_button_new_with_label (name); g_free (name); 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, "toggled", G_CALLBACK (feat_toggled_cb), NULL); gesture = gtk_gesture_click_new (); gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (gesture), GDK_BUTTON_SECONDARY); g_signal_connect (gesture, "pressed", G_CALLBACK (feat_pressed), feat); gtk_widget_add_controller (feat, GTK_EVENT_CONTROLLER (gesture)); gtk_box_append (GTK_BOX (group), feat); item = g_new (FeatureItem, 1); item->name = tags[i]; item->tag = tag; item->icon = NULL; item->dflt = NULL; item->feat = feat; demo->feature_items = g_list_prepend (demo->feature_items, item); } gtk_box_append (GTK_BOX (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; 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); gtk_widget_add_css_class (label, "heading"); gtk_box_append (GTK_BOX (group), label); for (i = 0; tags[i]; i++) { unsigned int tag; GtkWidget *feat; FeatureItem *item; char *name; tag = hb_tag_from_string (tags[i], -1); name = get_feature_display_name (tag); feat = gtk_check_button_new_with_label (name ? name : _("Default")); g_free (name); if (group_button == NULL) group_button = feat; else gtk_check_button_set_group (GTK_CHECK_BUTTON (feat), GTK_CHECK_BUTTON (group_button)); g_signal_connect (feat, "notify::active", G_CALLBACK (update_display), NULL); g_object_set_data (G_OBJECT (feat), "default", group_button); gtk_box_append (GTK_BOX (group), feat); item = g_new (FeatureItem, 1); item->name = tags[i]; item->tag = tag; item->icon = NULL; item->dflt = NULL; item->feat = feat; demo->feature_items = g_list_prepend (demo->feature_items, item); } gtk_box_append (GTK_BOX (box), group); } static void update_display (void) { GString *s; char *text; gboolean has_feature; GtkTreeIter iter; GtkTreeModel *model; PangoFontDescription *desc; GList *l; PangoAttrList *attrs; PangoAttribute *attr; int ins, bound; guint start, end; PangoLanguage *lang; char *font_desc; char *features; double value; int text_len; gboolean do_waterfall; GString *waterfall; { GtkTextBuffer *buffer; GtkTextIter start_iter, end_iter; buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (demo->entry)); gtk_text_buffer_get_bounds (buffer, &start_iter, &end_iter); text = gtk_text_buffer_get_text (buffer, &start_iter, &end_iter, FALSE); text_len = strlen (text); } do_waterfall = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (demo->waterfall_toggle)); gtk_label_set_wrap (GTK_LABEL (demo->the_label), !do_waterfall); if (do_waterfall) { start = PANGO_ATTR_INDEX_FROM_TEXT_BEGINNING; end = PANGO_ATTR_INDEX_TO_TEXT_END; } else if (gtk_label_get_selection_bounds (GTK_LABEL (demo->the_label), &ins, &bound)) { 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 (demo->font)); value = gtk_adjustment_get_value (demo->size_adjustment); pango_font_description_set_size (desc, value * PANGO_SCALE); 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); s = g_string_new (""); has_feature = FALSE; for (l = demo->feature_items; l; l = l->next) { FeatureItem *item = l->data; if (!gtk_widget_is_sensitive (item->feat)) continue; if (GTK_IS_CHECK_BUTTON (item->feat)) { if (g_object_get_data (G_OBJECT (item->feat), "default")) { if (gtk_check_button_get_active (GTK_CHECK_BUTTON (item->feat)) && strcmp (item->name, "xxxx") != 0) { if (has_feature) g_string_append (s, ", "); g_string_append (s, item->name); g_string_append (s, " 1"); has_feature = TRUE; } } else { if (gtk_check_button_get_inconsistent (GTK_CHECK_BUTTON (item->feat))) continue; if (has_feature) g_string_append (s, ", "); g_string_append (s, item->name); if (gtk_check_button_get_active (GTK_CHECK_BUTTON (item->feat))) g_string_append (s, " 1"); else g_string_append (s, " 0"); has_feature = TRUE; } } } features = g_string_free (s, FALSE); if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (demo->script_lang), &iter)) { hb_tag_t lang_tag; model = gtk_combo_box_get_model (GTK_COMBO_BOX (demo->script_lang)); gtk_tree_model_get (model, &iter, 3, &lang_tag, -1); lang = pango_language_from_string (hb_language_to_string (hb_ot_tag_to_language (lang_tag))); } else lang = NULL; attrs = pango_attr_list_new (); if (gtk_adjustment_get_value (demo->letterspacing_adjustment) != 0.) { attr = pango_attr_letter_spacing_new (gtk_adjustment_get_value (demo->letterspacing_adjustment)); attr->start_index = start; attr->end_index = end; pango_attr_list_insert (attrs, attr); } if (gtk_adjustment_get_value (demo->line_height_adjustment) != 1.) { attr = pango_attr_line_height_new (gtk_adjustment_get_value (demo->line_height_adjustment)); attr->start_index = start; attr->end_index = end; pango_attr_list_insert (attrs, attr); } { GdkRGBA rgba; char *fg, *bg, *css; gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (demo->foreground), &rgba); attr = pango_attr_foreground_new (65535 * rgba.red, 65535 * rgba.green, 65535 * rgba.blue); attr->start_index = start; attr->end_index = end; pango_attr_list_insert (attrs, attr); attr = pango_attr_foreground_alpha_new (65535 * rgba.alpha); attr->start_index = start; attr->end_index = end; pango_attr_list_insert (attrs, attr); fg = gdk_rgba_to_string (&rgba); gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (demo->background), &rgba); bg = gdk_rgba_to_string (&rgba); css = g_strdup_printf (".font_features_background { caret-color: %s; background-color: %s; }", fg, bg); gtk_css_provider_load_from_data (demo->provider, css, strlen (css)); g_free (css); g_free (fg); g_free (bg); } if (do_waterfall) { attr = pango_attr_font_desc_new (desc); pango_attr_list_insert (attrs, attr); attr = pango_attr_font_features_new (features); pango_attr_list_insert (attrs, attr); attr = pango_attr_language_new (lang); pango_attr_list_insert (attrs, attr); } else { ensure_range (start, end, desc, features, lang); for (l = demo->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); 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 (demo->description), font_desc); gtk_label_set_text (GTK_LABEL (demo->settings), features); if (do_waterfall) { waterfall = g_string_new (""); int sizes[] = { 7, 8, 9, 10, 12, 14, 16, 20, 24, 30, 40, 50, 60, 70, 90 }; start = 0; for (int i = 0; i < G_N_ELEMENTS (sizes); i++) { g_string_append (waterfall, text); g_string_append_c (waterfall, '\n'); attr = pango_attr_size_new (sizes[i] * PANGO_SCALE); attr->start_index = start; attr->end_index = start + text_len; pango_attr_list_insert (attrs, attr); start += text_len + 1; } gtk_label_set_text (GTK_LABEL (demo->the_label), waterfall->str); g_string_free (waterfall, TRUE); } else gtk_label_set_text (GTK_LABEL (demo->the_label), text); gtk_label_set_attributes (GTK_LABEL (demo->the_label), attrs); g_free (font_desc); pango_font_description_free (desc); g_free (features); pango_attr_list_unref (attrs); g_free (text); } static PangoFont * get_pango_font (void) { PangoFontDescription *desc; PangoContext *context; desc = gtk_font_chooser_get_font_desc (GTK_FONT_CHOOSER (demo->font)); context = gtk_widget_get_pango_context (demo->font); return pango_context_load_font (context, desc); } 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; } 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; } static void update_script_combo (void) { GtkListStore *store; hb_font_t *hb_font; int i, j, k; PangoFont *pango_font; GHashTable *tags; GHashTableIter iter; TagPair *pair; char *lang; hb_tag_t active; GtkTreeIter active_iter; gboolean have_active = FALSE; lang = gtk_font_chooser_get_language (GTK_FONT_CHOOSER (demo->font)); G_GNUC_BEGIN_IGNORE_DEPRECATIONS active = hb_ot_tag_from_language (hb_language_from_string (lang, -1)); G_GNUC_END_IGNORE_DEPRECATIONS g_free (lang); store = gtk_list_store_new (4, G_TYPE_STRING, G_TYPE_UINT, G_TYPE_UINT, G_TYPE_UINT); pango_font = get_pango_font (); hb_font = pango_font_get_hb_font (pango_font); tags = g_hash_table_new_full (tag_pair_hash, tag_pair_equal, g_free, NULL); pair = g_new (TagPair, 1); pair->script_tag = 0; pair->lang_tag = 0; g_hash_table_add (tags, pair); pair = g_new (TagPair, 1); pair->script_tag = HB_OT_TAG_DEFAULT_SCRIPT; pair->lang_tag = HB_OT_TAG_DEFAULT_LANGUAGE; g_hash_table_add (tags, pair); 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]; GtkTreeIter tree_iter; if (pair->lang_tag == 0 && pair->script_tag == 0) langname = NC_("Language", "None"); else if (pair->lang_tag == HB_OT_TAG_DEFAULT_LANGUAGE) langname = NC_("Language", "Default"); else { langname = get_language_name_for_tag (pair->lang_tag); if (!langname) { hb_tag_to_string (pair->lang_tag, langbuf); langbuf[4] = 0; langname = langbuf; } } gtk_list_store_insert_with_values (store, &tree_iter, -1, 0, langname, 1, pair->script_index, 2, pair->lang_index, 3, pair->lang_tag, -1); if (pair->lang_tag == active) { have_active = TRUE; active_iter = tree_iter; } } g_hash_table_destroy (tags); 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); gtk_combo_box_set_model (GTK_COMBO_BOX (demo->script_lang), GTK_TREE_MODEL (store)); if (have_active) gtk_combo_box_set_active_iter (GTK_COMBO_BOX (demo->script_lang), &active_iter); else gtk_combo_box_set_active_iter (GTK_COMBO_BOX (demo->script_lang), 0); } static char * get_name (hb_face_t *hbface, hb_ot_name_id_t id) { unsigned int len; char *text; if (id == HB_OT_NAME_ID_INVALID) return NULL; len = hb_ot_name_get_utf8 (hbface, id, HB_LANGUAGE_INVALID, NULL, NULL); len++; text = g_new (char, len); hb_ot_name_get_utf8 (hbface, id, HB_LANGUAGE_INVALID, &len, text); return text; } static void update_features (void) { int i, j; GtkTreeModel *model; GtkTreeIter iter; guint script_index, lang_index; hb_tag_t lang_tag; PangoFont *pango_font; hb_font_t *hb_font; GList *l; /* set feature presence checks from the font features */ if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (demo->script_lang), &iter)) return; model = gtk_combo_box_get_model (GTK_COMBO_BOX (demo->script_lang)); gtk_tree_model_get (model, &iter, 1, &script_index, 2, &lang_index, 3, &lang_tag, -1); if (lang_tag == 0) /* None is selected */ { for (l = demo->feature_items; l; l = l->next) { FeatureItem *item = l->data; gtk_widget_show (item->feat); gtk_widget_show (gtk_widget_get_parent (item->feat)); if (strcmp (item->name, "xxxx") == 0) gtk_check_button_set_active (GTK_CHECK_BUTTON (item->feat), TRUE); } return; } for (l = demo->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_check_button_set_active (GTK_CHECK_BUTTON (item->feat), TRUE); } pango_font = get_pango_font (); hb_font = pango_font_get_hb_font (pango_font); if (hb_font) { hb_tag_t tables[2] = { HB_OT_TAG_GSUB, HB_OT_TAG_GPOS }; hb_face_t *hb_face; char *feat; 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++) { char buf[5]; hb_tag_to_string (features[j], buf); buf[4] = 0; #if 0 g_print ("%s present in %s\n", buf, i == 0 ? "GSUB" : "GPOS"); #endif if (g_str_has_prefix (buf, "ss") || g_str_has_prefix (buf, "cv")) { unsigned int feature_index; hb_ot_name_id_t label_id, tooltip_id, sample_id, first_param_id; unsigned int num_params; hb_ot_layout_language_find_feature (hb_face, tables[i], script_index, lang_index, features[j], &feature_index); if (hb_ot_layout_feature_get_name_ids (hb_face, tables[i], feature_index, &label_id, &tooltip_id, &sample_id, &num_params, &first_param_id)) { char *label = get_name (hb_face, label_id); if (label) { for (l = demo->feature_items; l; l = l->next) { FeatureItem *item = l->data; if (item->tag == features[j]) { gtk_check_button_set_label (GTK_CHECK_BUTTON (item->feat), label); break; } } } g_free (label); } } for (l = demo->feature_items; l; l = l->next) { 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_CHECK_BUTTON (item->feat)) { GtkWidget *def = GTK_WIDGET (g_object_get_data (G_OBJECT (item->feat), "default")); if (def) { gtk_widget_show (def); gtk_widget_show (gtk_widget_get_parent (def)); gtk_check_button_set_active (GTK_CHECK_BUTTON (def), TRUE); } else set_inconsistent (GTK_CHECK_BUTTON (item->feat), TRUE); } } } } } feat = gtk_font_chooser_get_font_features (GTK_FONT_CHOOSER (demo->font)); if (feat) { for (l = demo->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_CHECK_BUTTON (item->feat) && g_object_get_data (G_OBJECT (item->feat), "default")) { gtk_check_button_set_active (GTK_CHECK_BUTTON (item->feat), p[6] == '1'); } else if (GTK_IS_CHECK_BUTTON (item->feat)) { set_inconsistent (GTK_CHECK_BUTTON (item->feat), FALSE); gtk_check_button_set_active (GTK_CHECK_BUTTON (item->feat), p[6] == '1'); } } } g_free (feat); } } g_object_unref (pango_font); } #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)); gtk_editable_set_text (GTK_EDITABLE (entry), str); g_free (str); update_display (); } static void entry_activated (GtkEntry *entry, GtkAdjustment *adjustment) { double value; char *err = NULL; value = g_strtod (gtk_editable_get_text (GTK_EDITABLE (entry)), &err); if (err != NULL) gtk_adjustment_set_value (adjustment, value); } static void unset_instance (GtkAdjustment *adjustment); static void start_or_stop_axis_animation (GtkButton *button, gpointer data); static void font_features_reset_variations (void) { GHashTableIter iter; Axis *axis; g_hash_table_iter_init (&iter, demo->axes); while (g_hash_table_iter_next (&iter, (gpointer *)NULL, (gpointer *)&axis)) { if (axis->tick_cb) start_or_stop_axis_animation (GTK_BUTTON (axis->button), axis); gtk_adjustment_set_value (axis->adjustment, axis->default_value); } } static void add_font_variations (GString *s) { GHashTableIter iter; Axis *axis; char buf[G_ASCII_DTOSTR_BUF_SIZE]; const char *sep = ""; g_hash_table_iter_init (&iter, demo->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 double ease_out_cubic (double t) { double p = t - 1; return p * p * p + 1; } static gboolean animate_axis (GtkWidget *widget, GdkFrameClock *frame_clock, gpointer data) { Axis *axis = data; guint64 now; double upper, lower, value; now = g_get_monotonic_time (); if (now >= axis->start_time + G_TIME_SPAN_SECOND) { axis->start_time += G_TIME_SPAN_SECOND; axis->increasing = !axis->increasing; } value = (now - axis->start_time) / (double) G_TIME_SPAN_SECOND; value = ease_out_cubic (value); lower = gtk_adjustment_get_lower (axis->adjustment); upper = gtk_adjustment_get_upper (axis->adjustment); if (axis->increasing) gtk_adjustment_set_value (axis->adjustment, lower + (upper - lower) * value); else gtk_adjustment_set_value (axis->adjustment, upper - (upper - lower) * value); return G_SOURCE_CONTINUE; } static void start_or_stop_axis_animation (GtkButton *button, gpointer data) { Axis *axis = data; if (axis->tick_cb) { gtk_widget_remove_tick_callback (GTK_WIDGET (button), axis->tick_cb); axis->tick_cb = 0; gtk_button_set_icon_name (button, "media-playback-start"); } else { double value, upper, lower; gtk_button_set_icon_name (button, "media-playback-stop"); axis->tick_cb = gtk_widget_add_tick_callback (GTK_WIDGET (button), animate_axis, axis, NULL); value = gtk_adjustment_get_value (axis->adjustment); lower = gtk_adjustment_get_lower (axis->adjustment); upper = gtk_adjustment_get_upper (axis->adjustment); value = value / (upper - lower); axis->start_time = g_get_monotonic_time () - value * G_TIME_SPAN_SECOND; axis->increasing = TRUE; } } static void add_axis (hb_face_t *hb_face, hb_ot_var_axis_info_t *ax, float value, int i) { GtkWidget *axis_label; GtkWidget *axis_entry; GtkWidget *axis_scale; GtkAdjustment *adjustment; Axis *axis; char name[20]; unsigned int name_len = 20; hb_ot_name_get_utf8 (hb_face, ax->name_id, HB_LANGUAGE_INVALID, &name_len, name); axis_label = gtk_label_new (name); gtk_widget_set_halign (axis_label, GTK_ALIGN_START); gtk_widget_set_valign (axis_label, GTK_ALIGN_BASELINE); gtk_grid_attach (GTK_GRID (demo->variations_grid), axis_label, 0, i, 1, 1); adjustment = gtk_adjustment_new (value, ax->min_value, ax->max_value, 1.0, 10.0, 0.0); axis_scale = gtk_scale_new (GTK_ORIENTATION_HORIZONTAL, adjustment); gtk_scale_add_mark (GTK_SCALE (axis_scale), ax->default_value, GTK_POS_TOP, NULL); 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_grid_attach (GTK_GRID (demo->variations_grid), axis_scale, 1, i, 1, 1); axis_entry = gtk_entry_new (); gtk_widget_set_valign (axis_entry, GTK_ALIGN_BASELINE); gtk_editable_set_width_chars (GTK_EDITABLE (axis_entry), 4); gtk_editable_set_max_width_chars (GTK_EDITABLE (axis_entry), 4); gtk_widget_set_hexpand (axis_entry, FALSE); gtk_grid_attach (GTK_GRID (demo->variations_grid), axis_entry, 2, i, 1, 1); axis = g_new0 (Axis, 1); axis->tag = ax->tag; axis->adjustment = adjustment; axis->default_value = ax->default_value; g_hash_table_add (demo->axes, axis); axis->button = gtk_button_new_from_icon_name ("media-playback-start"); gtk_widget_add_css_class (GTK_WIDGET (axis->button), "circular"); gtk_widget_set_valign (GTK_WIDGET (axis->button), GTK_ALIGN_CENTER); g_signal_connect (axis->button, "clicked", G_CALLBACK (start_or_stop_axis_animation), axis); gtk_grid_attach (GTK_GRID (demo->variations_grid), axis->button, 3, i, 1, 1); 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; unsigned int index; } 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 void add_instance (hb_face_t *face, unsigned int index, GtkWidget *combo, int pos) { Instance *instance; hb_ot_name_id_t name_id; char name[20]; unsigned int name_len = 20; instance = g_new0 (Instance, 1); 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); instance->name = g_strdup (name); instance->index = index; g_hash_table_add (demo->instances, instance); gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo), instance->name); } static void unset_instance (GtkAdjustment *adjustment) { if (demo->instance_combo) gtk_combo_box_set_active (GTK_COMBO_BOX (demo->instance_combo), 0); } static void instance_changed (GtkComboBox *combo) { char *text; Instance *instance; Instance ikey; int i; 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; 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 (demo->instances, &ikey); if (!instance) { g_print ("did not find instance %s\n", text); goto out; } 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++) { Axis *axis; Axis akey; double value; value = coords[ai[i].axis_index]; akey.tag = ai[i].tag; axis = g_hash_table_lookup (demo->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); g_clear_object (&pango_font); g_free (ai); g_free (coords); } static gboolean matches_instance (hb_face_t *hb_face, unsigned int index, unsigned int n_axes, float *coords) { float *instance_coords; unsigned int coords_length; int i; instance_coords = g_new (float, n_axes); coords_length = n_axes; 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]) 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 (demo->axes, &key); key.tag = MAKE_TAG('w','d','t','h'); width_axis = g_hash_table_lookup (demo->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 (demo->variations_grid), plane, 0, i, 3, 1); } } /* 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); } static void update_font_variations (void) { GtkWidget *child; 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; while ((child = gtk_widget_get_first_child (demo->variations_grid))) gtk_grid_remove (GTK_GRID (demo->variations_grid), child); demo->instance_combo = NULL; g_hash_table_remove_all (demo->axes); g_hash_table_remove_all (demo->instances); 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); if (n_axes == 0) goto done; ai = g_new0 (hb_ot_var_axis_info_t, n_axes); design_coords = g_new (float, n_axes); 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) { 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 (demo->variations_grid), label, 0, -1, 1, 1); combo = gtk_combo_box_text_new (); gtk_widget_set_halign (combo, GTK_ALIGN_START); gtk_widget_set_valign (combo, GTK_ALIGN_BASELINE); gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo), ""); 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++) { if (matches_instance (hb_face, i, n_axes, design_coords)) { gtk_combo_box_set_active (GTK_COMBO_BOX (combo), i + 1); break; } } gtk_grid_attach (GTK_GRID (demo->variations_grid), combo, 1, -1, 3, 1); g_signal_connect (combo, "changed", G_CALLBACK (instance_changed), NULL); demo->instance_combo = combo; } for (i = 0; i < n_axes; i++) add_axis (hb_face, &ai[i], design_coords[i], i); add_font_plane (n_axes); done: g_clear_object (&pango_font); g_free (ai); g_free (design_coords); } static void font_features_font_changed (void) { update_basic (); update_script_combo (); update_features (); update_font_variations (); } static void font_features_script_changed (void) { update_features (); update_display (); } static void font_features_reset_features (void) { GList *l; gtk_label_select_region (GTK_LABEL (demo->the_label), 0, 0); g_list_free_full (demo->ranges, free_range); demo->ranges = NULL; for (l = demo->feature_items; l; l = l->next) { FeatureItem *item = l->data; if (GTK_IS_CHECK_BUTTON (item->feat)) { if (strcmp (item->name, "xxxx") == 0) gtk_check_button_set_active (GTK_CHECK_BUTTON (item->feat), TRUE); else { gtk_check_button_set_active (GTK_CHECK_BUTTON (item->feat), FALSE); set_inconsistent (GTK_CHECK_BUTTON (item->feat), TRUE); } } } } static void font_features_toggle_edit (void) { if (strcmp (gtk_stack_get_visible_child_name (GTK_STACK (demo->stack)), "entry") != 0) { GtkTextBuffer *buffer; GtkTextIter start, end; buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (demo->entry)); gtk_text_buffer_get_bounds (buffer, &start, &end); g_free (demo->text); demo->text = gtk_text_buffer_get_text (buffer, &start, &end, FALSE); gtk_stack_set_visible_child_name (GTK_STACK (demo->stack), "entry"); gtk_widget_grab_focus (demo->entry); gtk_adjustment_set_value (gtk_scrolled_window_get_vadjustment (GTK_SCROLLED_WINDOW (demo->swin)), 0); } else { g_clear_pointer (&demo->text, g_free); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (demo->plain_toggle), TRUE); update_display (); } } static void font_features_stop_edit (void) { g_signal_emit_by_name (demo->edit_toggle, "clicked"); } static gboolean entry_key_press (GtkEventController *controller, guint keyval, guint keycode, GdkModifierType modifiers, GtkTextView *entry) { if (keyval == GDK_KEY_Escape) { gtk_text_buffer_set_text (gtk_text_view_get_buffer (entry), demo->text, -1); return GDK_EVENT_STOP; } return GDK_EVENT_PROPAGATE; } static const char *paragraphs[] = { "Grumpy wizards make toxic brew for the evil Queen and Jack. A quick movement of the enemy will jeopardize six gunboats. The job of waxing linoleum frequently peeves chintzy kids. My girl wove six dozen plaid jackets before she quit. Twelve ziggurats quickly jumped a finch box.", "Разъяренный чтец эгоистично бьёт пятью жердями шустрого фехтовальщика. Наш банк вчера же выплатил Ф.Я. Эйхгольду комиссию за ценные вещи. Эх, чужак, общий съём цен шляп (юфть) – вдрызг! В чащах юга жил бы цитрус? Да, но фальшивый экземпляр!", "Τάχιστη αλώπηξ βαφής ψημένη γη, δρασκελίζει υπέρ νωθρού κυνός", }; static const char *alphabets[] = { "abcdefghijklmnopqrstuvwxzy", "ABCDEFGHIJKLMNOPQRSTUVWXYZ", "0123456789", "!@#$%^&*/?;", }; static void set_text_alphabet (void) { demo->sample++; gtk_text_buffer_set_text (gtk_text_view_get_buffer (GTK_TEXT_VIEW (demo->entry)), alphabets[demo->sample % G_N_ELEMENTS (alphabets)], -1); update_display (); } static void set_text_paragraph (void) { demo->sample++; gtk_text_buffer_set_text (gtk_text_view_get_buffer (GTK_TEXT_VIEW (demo->entry)), paragraphs[demo->sample % G_N_ELEMENTS (paragraphs)], -1); update_display (); } GtkWidget * do_font_features (GtkWidget *do_widget) { static GtkWidget *window = NULL; if (!window) { GtkBuilder *builder; GtkBuilderScope *scope; GtkEventController *controller; builder = gtk_builder_new (); scope = gtk_builder_cscope_new (); gtk_builder_cscope_add_callback (scope, basic_value_changed); gtk_builder_cscope_add_callback (scope, basic_entry_activated); gtk_builder_cscope_add_callback (scope, color_set_cb); gtk_builder_cscope_add_callback (scope, swap_colors); gtk_builder_cscope_add_callback (scope, font_features_reset_basic); gtk_builder_cscope_add_callback (scope, font_features_reset_features); gtk_builder_cscope_add_callback (scope, font_features_reset_variations); gtk_builder_cscope_add_callback (scope, font_features_toggle_plain); gtk_builder_cscope_add_callback (scope, font_features_toggle_edit); gtk_builder_cscope_add_callback (scope, font_features_stop_edit); gtk_builder_cscope_add_callback (scope, font_features_font_changed); gtk_builder_cscope_add_callback (scope, font_features_script_changed); gtk_builder_cscope_add_callback (scope, font_features_notify_waterfall); gtk_builder_cscope_add_callback (scope, set_text_alphabet); gtk_builder_cscope_add_callback (scope, set_text_paragraph); gtk_builder_set_scope (builder, scope); demo = g_new0 (FontFeaturesDemo, 1); gtk_builder_add_from_resource (builder, "/font_features/font_features.ui", NULL); window = GTK_WIDGET (gtk_builder_get_object (builder, "window")); g_object_set_data_full (G_OBJECT (window), "demo", demo, demo_free); demo->the_label = GTK_WIDGET (gtk_builder_get_object (builder, "label")); demo->settings = GTK_WIDGET (gtk_builder_get_object (builder, "settings")); demo->description = GTK_WIDGET (gtk_builder_get_object (builder, "description")); demo->font = GTK_WIDGET (gtk_builder_get_object (builder, "font")); demo->script_lang = GTK_WIDGET (gtk_builder_get_object (builder, "script_lang")); demo->feature_list = GTK_WIDGET (gtk_builder_get_object (builder, "feature_list")); demo->stack = GTK_WIDGET (gtk_builder_get_object (builder, "stack")); demo->entry = GTK_WIDGET (gtk_builder_get_object (builder, "entry")); demo->plain_toggle = GTK_WIDGET (gtk_builder_get_object (builder, "plain_toggle")); demo->waterfall_toggle = GTK_WIDGET (gtk_builder_get_object (builder, "waterfall_toggle")); demo->edit_toggle = GTK_WIDGET (gtk_builder_get_object (builder, "edit_toggle")); demo->size_scale = GTK_WIDGET (gtk_builder_get_object (builder, "size_scale")); demo->size_entry = GTK_WIDGET (gtk_builder_get_object (builder, "size_entry")); demo->size_adjustment = GTK_ADJUSTMENT (gtk_builder_get_object (builder, "size_adjustment")); demo->letterspacing_entry = GTK_WIDGET (gtk_builder_get_object (builder, "letterspacing_entry")); demo->letterspacing_adjustment = GTK_ADJUSTMENT (gtk_builder_get_object (builder, "letterspacing_adjustment")); demo->line_height_entry = GTK_WIDGET (gtk_builder_get_object (builder, "line_height_entry")); demo->line_height_adjustment = GTK_ADJUSTMENT (gtk_builder_get_object (builder, "line_height_adjustment")); demo->foreground = GTK_WIDGET (gtk_builder_get_object (builder, "foreground")); demo->background = GTK_WIDGET (gtk_builder_get_object (builder, "background")); demo->swin = GTK_WIDGET (gtk_builder_get_object (builder, "swin")); demo->provider = gtk_css_provider_new (); gtk_style_context_add_provider (gtk_widget_get_style_context (demo->swin), GTK_STYLE_PROVIDER (demo->provider), 800); basic_value_changed (demo->size_adjustment, demo->size_entry); basic_value_changed (demo->letterspacing_adjustment, demo->letterspacing_entry); basic_value_changed (demo->line_height_adjustment, demo->line_height_entry); controller = gtk_event_controller_key_new (); g_signal_connect (controller, "key-pressed", G_CALLBACK (entry_key_press), demo->entry); gtk_widget_add_controller (demo->entry, controller); add_check_group (demo->feature_list, _("Kerning"), (const char *[]){ "kern", NULL }); add_check_group (demo->feature_list, _("Ligatures"), (const char *[]){ "liga", "dlig", "hlig", "clig", "rlig", NULL }); add_check_group (demo->feature_list, _("Letter Case"), (const char *[]){ "smcp", "c2sc", "pcap", "c2pc", "unic", "cpsp", "case",NULL }); add_radio_group (demo->feature_list, _("Number Case"), (const char *[]){ "xxxx", "lnum", "onum", NULL }); add_radio_group (demo->feature_list, _("Number Spacing"), (const char *[]){ "xxxx", "pnum", "tnum", NULL }); add_radio_group (demo->feature_list, _("Fractions"), (const char *[]){ "xxxx", "frac", "afrc", NULL }); add_check_group (demo->feature_list, _("Numeric Extras"), (const char *[]){ "zero", "nalt", "sinf", NULL }); add_check_group (demo->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 (demo->feature_list, _("Positional Alternatives"), (const char *[]){ "init", "medi", "med2", "fina", "fin2", "fin3", "isol", NULL }); add_check_group (demo->feature_list, _("Width Variants"), (const char *[]){ "fwid", "hwid", "halt", "pwid", "palt", "twid", "qwid", NULL }); add_check_group (demo->feature_list, _("Alternative Stylistic Sets"), (const char *[]){ "ss01", "ss02", "ss03", "ss04", "ss05", "ss06", "ss07", "ss08", "ss09", "ss10", "ss11", "ss12", "ss13", "ss14", "ss15", "ss16", "ss17", "ss18", "ss19", "ss20", NULL }); add_check_group (demo->feature_list, _("Character Variants"), (const char *[]){ "cv01", "cv02", "cv03", "cv04", "cv05", "cv06", "cv07", "cv08", "cv09", "cv10", "cv11", "cv12", "cv13", "cv14", "cv15", "cv16", "cv17", "cv18", "cv19", "cv20", NULL }); add_check_group (demo->feature_list, _("Mathematical"), (const char *[]){ "dtls", "flac", "mgrk", "ssty", NULL }); add_check_group (demo->feature_list, _("Optical Bounds"), (const char *[]){ "opbd", "lfbd", "rtbd", NULL }); demo->feature_items = g_list_reverse (demo->feature_items); demo->variations_grid = GTK_WIDGET (gtk_builder_get_object (builder, "variations_grid")); if (demo->instances == NULL) demo->instances = g_hash_table_new_full (instance_hash, instance_equal, NULL, free_instance); else g_hash_table_remove_all (demo->instances); if (demo->axes == NULL) demo->axes = g_hash_table_new_full (axes_hash, axes_equal, NULL, g_free); else g_hash_table_remove_all (demo->axes); font_features_font_changed (); g_object_add_weak_pointer (G_OBJECT (window), (gpointer *)&window); g_object_unref (builder); update_display (); } if (!gtk_widget_get_visible (window)) gtk_window_present (GTK_WINDOW (window)); else gtk_window_destroy (GTK_WINDOW (window)); return window; }