mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-12 21:40:19 +00:00
font chooser: Add examples for font features
For some font features, we can figure out affected glyphs, and show before/after. For some others, we hardcode typical sequences. Still to do: figure out how to find ligatures and show them.
This commit is contained in:
parent
63a7d99d25
commit
d0e46d257c
@ -1836,7 +1836,9 @@ update_language_combo (GtkFontChooserWidget *fontchooser,
|
|||||||
typedef struct {
|
typedef struct {
|
||||||
hb_tag_t tag;
|
hb_tag_t tag;
|
||||||
const char *name;
|
const char *name;
|
||||||
|
GtkWidget *top;
|
||||||
GtkWidget *feat;
|
GtkWidget *feat;
|
||||||
|
GtkWidget *example;
|
||||||
} FeatureItem;
|
} FeatureItem;
|
||||||
|
|
||||||
static const char *
|
static const char *
|
||||||
@ -1889,6 +1891,180 @@ feat_pressed (GtkGesture *gesture,
|
|||||||
set_inconsistent (GTK_CHECK_BUTTON (feat), !inconsistent);
|
set_inconsistent (GTK_CHECK_BUTTON (feat), !inconsistent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
find_affected_text (hb_tag_t feature_tag,
|
||||||
|
hb_face_t *hb_face,
|
||||||
|
hb_tag_t script_tag,
|
||||||
|
hb_tag_t lang_tag,
|
||||||
|
int max_chars)
|
||||||
|
{
|
||||||
|
unsigned int script_index = 0;
|
||||||
|
unsigned int lang_index = 0;
|
||||||
|
unsigned int feature_index = 0;
|
||||||
|
GString *chars;
|
||||||
|
|
||||||
|
chars = g_string_new ("");
|
||||||
|
|
||||||
|
hb_ot_layout_table_find_script (hb_face, HB_OT_TAG_GSUB, script_tag, &script_index);
|
||||||
|
hb_ot_layout_script_find_language (hb_face, HB_OT_TAG_GSUB, script_index, lang_tag, &lang_index);
|
||||||
|
if (hb_ot_layout_language_find_feature (hb_face, HB_OT_TAG_GSUB, script_index, lang_index, feature_tag, &feature_index))
|
||||||
|
{
|
||||||
|
unsigned int lookup_indexes[32];
|
||||||
|
unsigned int lookup_count = 32;
|
||||||
|
int count;
|
||||||
|
int n_chars = 0;
|
||||||
|
|
||||||
|
count = hb_ot_layout_feature_get_lookups (hb_face,
|
||||||
|
HB_OT_TAG_GSUB,
|
||||||
|
feature_index,
|
||||||
|
0,
|
||||||
|
&lookup_count,
|
||||||
|
lookup_indexes);
|
||||||
|
if (count > 0)
|
||||||
|
{
|
||||||
|
hb_set_t* glyphs_before = NULL;
|
||||||
|
hb_set_t* glyphs_input = NULL;
|
||||||
|
hb_set_t* glyphs_after = NULL;
|
||||||
|
hb_set_t* glyphs_output = NULL;
|
||||||
|
hb_font_t *hb_font = NULL;
|
||||||
|
hb_codepoint_t gid;
|
||||||
|
|
||||||
|
glyphs_input = hb_set_create ();
|
||||||
|
|
||||||
|
// XXX For now, just look at first index
|
||||||
|
hb_ot_layout_lookup_collect_glyphs (hb_face,
|
||||||
|
HB_OT_TAG_GSUB,
|
||||||
|
lookup_indexes[0],
|
||||||
|
glyphs_before,
|
||||||
|
glyphs_input,
|
||||||
|
glyphs_after,
|
||||||
|
glyphs_output);
|
||||||
|
|
||||||
|
hb_font = hb_font_create (hb_face);
|
||||||
|
hb_ft_font_set_funcs (hb_font);
|
||||||
|
|
||||||
|
gid = -1;
|
||||||
|
while (hb_set_next (glyphs_input, &gid)) {
|
||||||
|
hb_codepoint_t ch;
|
||||||
|
if (n_chars == max_chars)
|
||||||
|
{
|
||||||
|
g_string_append (chars, "…");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
for (ch = 0; ch < 0xffff; ch++) {
|
||||||
|
hb_codepoint_t glyph = 0;
|
||||||
|
hb_font_get_nominal_glyph (hb_font, ch, &glyph);
|
||||||
|
if (glyph == gid) {
|
||||||
|
g_string_append_unichar (chars, (gunichar)ch);
|
||||||
|
n_chars++;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hb_set_destroy (glyphs_input);
|
||||||
|
hb_font_destroy (hb_font);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return g_string_free (chars, FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
update_feature_example (FeatureItem *item,
|
||||||
|
hb_face_t *hb_face,
|
||||||
|
hb_tag_t script_tag,
|
||||||
|
hb_tag_t lang_tag,
|
||||||
|
PangoFontDescription *font_desc)
|
||||||
|
{
|
||||||
|
const char *letter_case[] = { "smcp", "c2sc", "pcap", "c2pc", "unic", "cpsp", "case", NULL };
|
||||||
|
const char *number_case[] = { "xxxx", "lnum", "onum", NULL };
|
||||||
|
const char *number_spacing[] = { "xxxx", "pnum", "tnum", NULL };
|
||||||
|
const char *number_formatting[] = { "zero", "nalt", NULL };
|
||||||
|
const char *char_variants[] = {
|
||||||
|
"swsh", "cswh", "calt", "falt", "hist", "salt", "jalt", "titl", "rand",
|
||||||
|
"ss01", "ss02", "ss03", "ss04", "ss05", "ss06", "ss07", "ss08", "ss09", "ss10",
|
||||||
|
"ss11", "ss12", "ss13", "ss14", "ss15", "ss16", "ss17", "ss18", "ss19", "ss20",
|
||||||
|
NULL };
|
||||||
|
|
||||||
|
if (g_strv_contains (number_case, item->name) ||
|
||||||
|
g_strv_contains (number_spacing, item->name))
|
||||||
|
{
|
||||||
|
PangoAttrList *attrs;
|
||||||
|
PangoAttribute *attr;
|
||||||
|
PangoFontDescription *desc;
|
||||||
|
char *str;
|
||||||
|
|
||||||
|
attrs = pango_attr_list_new ();
|
||||||
|
|
||||||
|
desc = pango_font_description_copy (font_desc);
|
||||||
|
pango_font_description_unset_fields (desc, PANGO_FONT_MASK_SIZE);
|
||||||
|
pango_attr_list_insert (attrs, pango_attr_font_desc_new (desc));
|
||||||
|
pango_font_description_free (desc);
|
||||||
|
str = g_strconcat (item->name, " 1", NULL);
|
||||||
|
attr = pango_attr_font_features_new (str);
|
||||||
|
pango_attr_list_insert (attrs, attr);
|
||||||
|
|
||||||
|
gtk_label_set_text (GTK_LABEL (item->example), "0123456789");
|
||||||
|
gtk_label_set_attributes (GTK_LABEL (item->example), attrs);
|
||||||
|
|
||||||
|
pango_attr_list_unref (attrs);
|
||||||
|
}
|
||||||
|
else if (g_strv_contains (letter_case, item->name) ||
|
||||||
|
g_strv_contains (number_formatting, item->name) ||
|
||||||
|
g_strv_contains (char_variants, item->name))
|
||||||
|
{
|
||||||
|
char *input = NULL;
|
||||||
|
char *text;
|
||||||
|
|
||||||
|
if (strcmp (item->name, "case") == 0)
|
||||||
|
input = g_strdup ("A-B[Cq]");
|
||||||
|
else if (g_strv_contains (letter_case, item->name))
|
||||||
|
input = g_strdup ("AaBbCc…");
|
||||||
|
else if (strcmp (item->name, "zero") == 0)
|
||||||
|
input = g_strdup ("0");
|
||||||
|
else if (strcmp (item->name, "nalt") == 0)
|
||||||
|
input = find_affected_text (item->tag, hb_face, script_tag, lang_tag, 3);
|
||||||
|
else
|
||||||
|
input = find_affected_text (item->tag, hb_face, script_tag, lang_tag, 10);
|
||||||
|
|
||||||
|
if (input[0] != '\0')
|
||||||
|
{
|
||||||
|
PangoAttrList *attrs;
|
||||||
|
PangoAttribute *attr;
|
||||||
|
PangoFontDescription *desc;
|
||||||
|
char *str;
|
||||||
|
|
||||||
|
text = g_strconcat (input, " ⟶ ", input, NULL);
|
||||||
|
|
||||||
|
attrs = pango_attr_list_new ();
|
||||||
|
|
||||||
|
desc = pango_font_description_copy (font_desc);
|
||||||
|
pango_font_description_unset_fields (desc, PANGO_FONT_MASK_SIZE);
|
||||||
|
pango_attr_list_insert (attrs, pango_attr_font_desc_new (desc));
|
||||||
|
pango_font_description_free (desc);
|
||||||
|
str = g_strconcat (item->name, " 0", NULL);
|
||||||
|
attr = pango_attr_font_features_new (str);
|
||||||
|
attr->start_index = 0;
|
||||||
|
attr->end_index = strlen (input);
|
||||||
|
pango_attr_list_insert (attrs, attr);
|
||||||
|
str = g_strconcat (item->name, " 1", NULL);
|
||||||
|
attr = pango_attr_font_features_new (str);
|
||||||
|
attr->start_index = strlen (input) + strlen (" ⟶ ");
|
||||||
|
attr->end_index = attr->start_index + strlen (input);
|
||||||
|
pango_attr_list_insert (attrs, attr);
|
||||||
|
|
||||||
|
gtk_label_set_text (GTK_LABEL (item->example), text);
|
||||||
|
gtk_label_set_attributes (GTK_LABEL (item->example), attrs);
|
||||||
|
|
||||||
|
g_free (text);
|
||||||
|
pango_attr_list_unref (attrs);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
gtk_label_set_markup (GTK_LABEL (item->example), "");
|
||||||
|
g_free (input);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
add_check_group (GtkFontChooserWidget *fontchooser,
|
add_check_group (GtkFontChooserWidget *fontchooser,
|
||||||
const char *title,
|
const char *title,
|
||||||
@ -1901,7 +2077,7 @@ add_check_group (GtkFontChooserWidget *fontchooser,
|
|||||||
int i;
|
int i;
|
||||||
|
|
||||||
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
||||||
gtk_widget_set_halign (group, GTK_ALIGN_START);
|
gtk_widget_set_halign (group, GTK_ALIGN_FILL);
|
||||||
|
|
||||||
label = gtk_label_new (title);
|
label = gtk_label_new (title);
|
||||||
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
||||||
@ -1919,12 +2095,13 @@ add_check_group (GtkFontChooserWidget *fontchooser,
|
|||||||
GtkWidget *feat;
|
GtkWidget *feat;
|
||||||
FeatureItem *item;
|
FeatureItem *item;
|
||||||
GtkGesture *gesture;
|
GtkGesture *gesture;
|
||||||
|
GtkWidget *box;
|
||||||
|
GtkWidget *example;
|
||||||
|
|
||||||
tag = hb_tag_from_string (tags[i], -1);
|
tag = hb_tag_from_string (tags[i], -1);
|
||||||
|
|
||||||
feat = gtk_check_button_new_with_label (get_feature_display_name (tag));
|
feat = gtk_check_button_new_with_label (get_feature_display_name (tag));
|
||||||
set_inconsistent (GTK_CHECK_BUTTON (feat), TRUE);
|
set_inconsistent (GTK_CHECK_BUTTON (feat), TRUE);
|
||||||
|
|
||||||
g_signal_connect_swapped (feat, "notify::active", G_CALLBACK (update_font_features), fontchooser);
|
g_signal_connect_swapped (feat, "notify::active", G_CALLBACK (update_font_features), fontchooser);
|
||||||
g_signal_connect_swapped (feat, "notify::inconsistent", G_CALLBACK (update_font_features), fontchooser);
|
g_signal_connect_swapped (feat, "notify::inconsistent", G_CALLBACK (update_font_features), fontchooser);
|
||||||
g_signal_connect (feat, "clicked", G_CALLBACK (feat_clicked), NULL);
|
g_signal_connect (feat, "clicked", G_CALLBACK (feat_clicked), NULL);
|
||||||
@ -1935,12 +2112,22 @@ add_check_group (GtkFontChooserWidget *fontchooser,
|
|||||||
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (gesture), GDK_BUTTON_SECONDARY);
|
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (gesture), GDK_BUTTON_SECONDARY);
|
||||||
g_signal_connect (gesture, "pressed", G_CALLBACK (feat_pressed), feat);
|
g_signal_connect (gesture, "pressed", G_CALLBACK (feat_pressed), feat);
|
||||||
|
|
||||||
gtk_container_add (GTK_CONTAINER (group), feat);
|
example = gtk_label_new ("");
|
||||||
|
gtk_label_set_selectable (GTK_LABEL (example), TRUE);
|
||||||
|
gtk_widget_set_halign (example, GTK_ALIGN_START);
|
||||||
|
|
||||||
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
|
||||||
|
gtk_box_set_homogeneous (GTK_BOX (box), TRUE);
|
||||||
|
gtk_container_add (GTK_CONTAINER (box), feat);
|
||||||
|
gtk_container_add (GTK_CONTAINER (box), example);
|
||||||
|
gtk_container_add (GTK_CONTAINER (group), box);
|
||||||
|
|
||||||
item = g_new (FeatureItem, 1);
|
item = g_new (FeatureItem, 1);
|
||||||
item->name = tags[i];
|
item->name = tags[i];
|
||||||
item->tag = tag;
|
item->tag = tag;
|
||||||
|
item->top = box;
|
||||||
item->feat = feat;
|
item->feat = feat;
|
||||||
|
item->example = example;
|
||||||
|
|
||||||
priv->feature_items = g_list_prepend (priv->feature_items, item);
|
priv->feature_items = g_list_prepend (priv->feature_items, item);
|
||||||
}
|
}
|
||||||
@ -1961,7 +2148,7 @@ add_radio_group (GtkFontChooserWidget *fontchooser,
|
|||||||
PangoAttrList *attrs;
|
PangoAttrList *attrs;
|
||||||
|
|
||||||
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
group = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
||||||
gtk_widget_set_halign (group, GTK_ALIGN_START);
|
gtk_widget_set_halign (group, GTK_ALIGN_FILL);
|
||||||
|
|
||||||
label = gtk_label_new (title);
|
label = gtk_label_new (title);
|
||||||
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
||||||
@ -1979,6 +2166,8 @@ add_radio_group (GtkFontChooserWidget *fontchooser,
|
|||||||
GtkWidget *feat;
|
GtkWidget *feat;
|
||||||
FeatureItem *item;
|
FeatureItem *item;
|
||||||
const char *name;
|
const char *name;
|
||||||
|
GtkWidget *box;
|
||||||
|
GtkWidget *example;
|
||||||
|
|
||||||
tag = hb_tag_from_string (tags[i], -1);
|
tag = hb_tag_from_string (tags[i], -1);
|
||||||
name = get_feature_display_name (tag);
|
name = get_feature_display_name (tag);
|
||||||
@ -1991,12 +2180,22 @@ add_radio_group (GtkFontChooserWidget *fontchooser,
|
|||||||
g_signal_connect_swapped (feat, "notify::active", G_CALLBACK (update_font_features), fontchooser);
|
g_signal_connect_swapped (feat, "notify::active", G_CALLBACK (update_font_features), fontchooser);
|
||||||
g_object_set_data (G_OBJECT (feat), "default", group_button);
|
g_object_set_data (G_OBJECT (feat), "default", group_button);
|
||||||
|
|
||||||
gtk_container_add (GTK_CONTAINER (group), feat);
|
example = gtk_label_new ("");
|
||||||
|
gtk_label_set_selectable (GTK_LABEL (example), TRUE);
|
||||||
|
gtk_widget_set_halign (example, GTK_ALIGN_START);
|
||||||
|
|
||||||
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
|
||||||
|
gtk_box_set_homogeneous (GTK_BOX (box), TRUE);
|
||||||
|
gtk_container_add (GTK_CONTAINER (box), feat);
|
||||||
|
gtk_container_add (GTK_CONTAINER (box), example);
|
||||||
|
gtk_container_add (GTK_CONTAINER (group), box);
|
||||||
|
|
||||||
item = g_new (FeatureItem, 1);
|
item = g_new (FeatureItem, 1);
|
||||||
item->name = tags[i];
|
item->name = tags[i];
|
||||||
item->tag = tag;
|
item->tag = tag;
|
||||||
|
item->top = box;
|
||||||
item->feat = feat;
|
item->feat = feat;
|
||||||
|
item->example = example;
|
||||||
|
|
||||||
priv->feature_items = g_list_prepend (priv->feature_items, item);
|
priv->feature_items = g_list_prepend (priv->feature_items, item);
|
||||||
}
|
}
|
||||||
@ -2012,7 +2211,11 @@ gtk_font_chooser_widget_populate_features (GtkFontChooserWidget *fontchooser)
|
|||||||
const char *number_case[] = { "xxxx", "lnum", "onum", NULL };
|
const char *number_case[] = { "xxxx", "lnum", "onum", NULL };
|
||||||
const char *number_spacing[] = { "xxxx", "pnum", "tnum", NULL };
|
const char *number_spacing[] = { "xxxx", "pnum", "tnum", NULL };
|
||||||
const char *number_formatting[] = { "zero", "nalt", NULL };
|
const char *number_formatting[] = { "zero", "nalt", NULL };
|
||||||
const char *char_variants[] = { "swsh", "cswh", "calt", "falt", "hist", "salt", "jalt", "titl", "rand", NULL };
|
const char *char_variants[] = {
|
||||||
|
"swsh", "cswh", "calt", "falt", "hist", "salt", "jalt", "titl", "rand",
|
||||||
|
"ss01", "ss02", "ss03", "ss04", "ss05", "ss06", "ss07", "ss08", "ss09", "ss10",
|
||||||
|
"ss11", "ss12", "ss13", "ss14", "ss15", "ss16", "ss17", "ss18", "ss19", "ss20",
|
||||||
|
NULL };
|
||||||
|
|
||||||
add_check_group (fontchooser, _("Ligatures"), ligatures);
|
add_check_group (fontchooser, _("Ligatures"), ligatures);
|
||||||
add_check_group (fontchooser, _("Letter Case"), letter_case);
|
add_check_group (fontchooser, _("Letter Case"), letter_case);
|
||||||
@ -2044,8 +2247,8 @@ gtk_font_chooser_widget_update_font_features (GtkFontChooserWidget *fontchooser)
|
|||||||
for (l = priv->feature_items; l; l = l->next)
|
for (l = priv->feature_items; l; l = l->next)
|
||||||
{
|
{
|
||||||
FeatureItem *item = l->data;
|
FeatureItem *item = l->data;
|
||||||
gtk_widget_hide (item->feat);
|
gtk_widget_hide (item->top);
|
||||||
gtk_widget_hide (gtk_widget_get_parent (item->feat));
|
gtk_widget_hide (gtk_widget_get_parent (item->top));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((priv->level & GTK_FONT_CHOOSER_LEVEL_FEATURES) == 0)
|
if ((priv->level & GTK_FONT_CHOOSER_LEVEL_FEATURES) == 0)
|
||||||
@ -2093,14 +2296,16 @@ gtk_font_chooser_widget_update_font_features (GtkFontChooserWidget *fontchooser)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
has_feature = TRUE;
|
has_feature = TRUE;
|
||||||
gtk_widget_show (item->feat);
|
gtk_widget_show (item->top);
|
||||||
gtk_widget_show (gtk_widget_get_parent (item->feat));
|
gtk_widget_show (gtk_widget_get_parent (item->top));
|
||||||
gtk_widget_show (priv->feature_language_combo);
|
gtk_widget_show (priv->feature_language_combo);
|
||||||
|
|
||||||
|
update_feature_example (item, hb_face, script_tag, lang_tag, priv->font_desc);
|
||||||
|
|
||||||
if (GTK_IS_RADIO_BUTTON (item->feat))
|
if (GTK_IS_RADIO_BUTTON (item->feat))
|
||||||
{
|
{
|
||||||
GtkWidget *def = GTK_WIDGET (g_object_get_data (G_OBJECT (item->feat), "default"));
|
GtkWidget *def = GTK_WIDGET (g_object_get_data (G_OBJECT (item->feat), "default"));
|
||||||
gtk_widget_show (def);
|
gtk_widget_show (gtk_widget_get_parent (def));
|
||||||
}
|
}
|
||||||
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
else if (GTK_IS_CHECK_BUTTON (item->feat))
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user