2017-02-20 16:11:52 -06:00
|
|
|
// 12 february 2017
|
|
|
|
#include "uipriv_unix.h"
|
|
|
|
|
2017-02-20 21:25:49 -06:00
|
|
|
// TODO pango alpha attributes turn 0 into 65535 :|
|
|
|
|
|
2017-02-20 16:11:52 -06:00
|
|
|
// we need to collect all the OpenType features and background blocks and add them all at once
|
2017-02-20 19:41:14 -06:00
|
|
|
// TODO this is the wrong approach; it causes Pango to end runs early, meaning attributes like the ligature attributes never get applied properly
|
2017-02-20 16:11:52 -06:00
|
|
|
// TODO rename this struct to something that isn't exclusively foreach-ing?
|
|
|
|
struct foreachParams {
|
2017-02-20 18:51:00 -06:00
|
|
|
const char *s;
|
2017-02-20 16:11:52 -06:00
|
|
|
PangoAttrList *attrs;
|
|
|
|
// keys are pointers to size_t maintained by g_new0()/g_free()
|
2017-05-17 22:37:16 -05:00
|
|
|
// values are strings
|
2017-02-20 16:11:52 -06:00
|
|
|
GHashTable *features;
|
2017-02-20 21:24:02 -06:00
|
|
|
// TODO use pango's built-in background attribute?
|
|
|
|
GPtrArray *backgroundClosures;
|
2017-02-20 16:11:52 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
static gboolean featurePosEqual(gconstpointer a, gconstpointer b)
|
|
|
|
{
|
|
|
|
size_t *sa = (size_t *) a;
|
|
|
|
size_t *sb = (size_t *) b;
|
|
|
|
|
|
|
|
return *sa == *sb;
|
|
|
|
}
|
|
|
|
|
|
|
|
static guint featurePosHash(gconstpointer n)
|
|
|
|
{
|
|
|
|
size_t *sn = (size_t *) n;
|
|
|
|
|
|
|
|
return (guint) (*sn);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void freeFeatureString(gpointer s)
|
|
|
|
{
|
|
|
|
g_string_free((GString *) s, TRUE);
|
|
|
|
}
|
|
|
|
|
2017-02-20 18:51:00 -06:00
|
|
|
#define isCodepointStart(b) (((uint8_t) (b)) <= 0x7F || ((uint8_t) (b)) >= 0xC0)
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
static void setFeaturesInRange(struct foreachParams *p, size_t start, size_t end, uiOpenTypeFeatures *otf)
|
2017-02-20 16:11:52 -06:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
size_t *key;
|
|
|
|
GString *new;
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
new = otfToPangoCSSString(otf);
|
2017-02-20 16:11:52 -06:00
|
|
|
for (i = start; i < end; i++) {
|
2017-02-20 18:51:00 -06:00
|
|
|
// don't create redundant entries; see below
|
|
|
|
if (!isCodepointStart(p->s[i]))
|
|
|
|
continue;
|
2017-02-20 16:11:52 -06:00
|
|
|
key = g_new0(size_t, 1);
|
|
|
|
*key = i;
|
2017-05-17 22:37:16 -05:00
|
|
|
g_hash_table_replace(p->features, key, g_strdup(new->str));
|
2017-02-20 16:11:52 -06:00
|
|
|
}
|
2017-05-17 22:37:16 -05:00
|
|
|
g_string_free(new, TRUE);
|
2017-02-20 16:11:52 -06:00
|
|
|
}
|
|
|
|
|
2017-02-20 21:24:02 -06:00
|
|
|
struct closureParams {
|
|
|
|
size_t start;
|
|
|
|
size_t end;
|
|
|
|
double r;
|
|
|
|
double g;
|
|
|
|
double b;
|
|
|
|
double a;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void backgroundClosure(uiDrawContext *c, uiDrawTextLayout *layout, double x, double y, gpointer data)
|
|
|
|
{
|
|
|
|
struct closureParams *p = (struct closureParams *) data;
|
|
|
|
uiDrawBrush brush;
|
|
|
|
|
|
|
|
brush.Type = uiDrawBrushTypeSolid;
|
|
|
|
brush.R = p->r;
|
|
|
|
brush.G = p->g;
|
|
|
|
brush.B = p->b;
|
|
|
|
brush.A = p->a;
|
|
|
|
drawTextBackground(c, x, y, layout, p->start, p->end, &brush, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void freeClosureParams(gpointer data, GClosure *closure)
|
|
|
|
{
|
|
|
|
uiFree((struct closureParams *) data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GClosure *mkBackgroundClosure(size_t start, size_t end, double r, double g, double b, double a)
|
2017-02-20 16:11:52 -06:00
|
|
|
{
|
2017-02-20 21:24:02 -06:00
|
|
|
struct closureParams *p;
|
|
|
|
GClosure *closure;
|
2017-02-20 16:11:52 -06:00
|
|
|
|
2017-02-20 21:24:02 -06:00
|
|
|
p = uiNew(struct closureParams);
|
|
|
|
p->start = start;
|
|
|
|
p->end = end;
|
|
|
|
p->r = r;
|
|
|
|
p->g = g;
|
|
|
|
p->b = b;
|
|
|
|
p->a = a;
|
|
|
|
closure = (GClosure *) g_cclosure_new(G_CALLBACK(backgroundClosure), p, freeClosureParams);
|
|
|
|
// TODO write a specific marshaler
|
|
|
|
// TODO or drop the closure stuff entirely
|
|
|
|
g_closure_set_marshal(closure, g_cclosure_marshal_generic);
|
|
|
|
return closure;
|
2017-02-20 16:11:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
static void addattr(struct foreachParams *p, size_t start, size_t end, PangoAttribute *attr)
|
|
|
|
{
|
|
|
|
if (attr == NULL) // in case of a future attribute
|
|
|
|
return;
|
|
|
|
attr->start_index = start;
|
|
|
|
attr->end_index = end;
|
|
|
|
pango_attr_list_insert(p->attrs, attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int processAttribute(uiAttributedString *s, uiAttributeSpec *spec, size_t start, size_t end, void *data)
|
|
|
|
{
|
|
|
|
struct foreachParams *p = (struct foreachParams *) data;
|
2017-02-20 21:24:02 -06:00
|
|
|
GClosure *closure;
|
2017-02-20 16:11:52 -06:00
|
|
|
PangoUnderline underline;
|
|
|
|
|
|
|
|
switch (spec->Type) {
|
|
|
|
case uiAttributeFamily:
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_family_new((const char *) (spec->Value)));
|
|
|
|
break;
|
|
|
|
case uiAttributeSize:
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_size_new(cairoToPango(spec->Double)));
|
|
|
|
break;
|
|
|
|
case uiAttributeWeight:
|
2017-02-20 19:41:14 -06:00
|
|
|
// TODO reverse the misalignment from drawtext.c if it is corrected
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_weight_new((PangoWeight) (spec->Value)));
|
2017-02-20 16:11:52 -06:00
|
|
|
break;
|
|
|
|
case uiAttributeItalic:
|
2017-02-20 19:41:14 -06:00
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_style_new(pangoItalics[(uiDrawTextItalic) (spec->Value)]));
|
2017-02-20 16:11:52 -06:00
|
|
|
break;
|
|
|
|
case uiAttributeStretch:
|
2017-02-20 19:41:14 -06:00
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_stretch_new(pangoStretches[(uiDrawTextStretch) (spec->Value)]));
|
2017-02-20 16:11:52 -06:00
|
|
|
break;
|
|
|
|
case uiAttributeColor:
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_foreground_new(
|
|
|
|
(guint16) (spec->R * 65535.0),
|
|
|
|
(guint16) (spec->G * 65535.0),
|
|
|
|
(guint16) (spec->B * 65535.0)));
|
|
|
|
addattr(p, start, end,
|
|
|
|
FUTURE_pango_attr_foreground_alpha_new(
|
|
|
|
(guint16) (spec->A * 65535.0)));
|
|
|
|
break;
|
|
|
|
case uiAttributeBackground:
|
2017-02-20 21:24:02 -06:00
|
|
|
closure = mkBackgroundClosure(start, end,
|
2017-02-20 16:11:52 -06:00
|
|
|
spec->R, spec->G, spec->B, spec->A);
|
2017-02-20 21:24:02 -06:00
|
|
|
g_ptr_array_add(p->backgroundClosures, closure);
|
2017-02-20 16:11:52 -06:00
|
|
|
break;
|
|
|
|
case uiAttributeUnderline:
|
|
|
|
switch (spec->Value) {
|
|
|
|
case uiDrawUnderlineStyleNone:
|
|
|
|
underline = PANGO_UNDERLINE_NONE;
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineStyleSingle:
|
|
|
|
underline = PANGO_UNDERLINE_SINGLE;
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineStyleDouble:
|
|
|
|
underline = PANGO_UNDERLINE_DOUBLE;
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineStyleSuggestion:
|
|
|
|
underline = PANGO_UNDERLINE_ERROR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_underline_new(underline));
|
|
|
|
break;
|
|
|
|
case uiAttributeUnderlineColor:
|
|
|
|
switch (spec->Value) {
|
|
|
|
case uiDrawUnderlineColorCustom:
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_underline_color_new(
|
|
|
|
(guint16) (spec->R * 65535.0),
|
|
|
|
(guint16) (spec->G * 65535.0),
|
|
|
|
(guint16) (spec->B * 65535.0)));
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineColorSpelling:
|
|
|
|
// TODO GtkTextView style property error-underline-color
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_underline_color_new(65535, 0, 0));
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineColorGrammar:
|
|
|
|
// TODO find a more appropriate color
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_underline_color_new(0, 65535, 0));
|
|
|
|
break;
|
|
|
|
case uiDrawUnderlineColorAuxiliary:
|
|
|
|
// TODO find a more appropriate color
|
|
|
|
addattr(p, start, end,
|
|
|
|
pango_attr_underline_color_new(0, 0, 65535));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2017-05-17 22:37:16 -05:00
|
|
|
case uiAttributeFeatures:
|
|
|
|
// TODO ensure the parentheses around spec->Value are provided on Windows
|
|
|
|
setFeaturesInRange(p, start, end, (uiOpenTypeFeatures *) (spec->Value));
|
2017-02-20 16:11:52 -06:00
|
|
|
break;
|
2017-05-17 22:37:16 -05:00
|
|
|
default:
|
|
|
|
// TODO complain
|
|
|
|
;
|
2017-02-20 16:11:52 -06:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean applyFeatures(gpointer key, gpointer value, gpointer data)
|
|
|
|
{
|
|
|
|
struct foreachParams *p = (struct foreachParams *) data;
|
|
|
|
size_t *pos = (size_t *) key;
|
2017-05-17 22:37:16 -05:00
|
|
|
char *s = (char *) value;
|
2017-02-20 18:51:00 -06:00
|
|
|
size_t n;
|
2017-02-20 16:11:52 -06:00
|
|
|
|
2017-02-20 18:51:00 -06:00
|
|
|
// make sure we cover an entire code point
|
|
|
|
// otherwise Pango will break apart multi-byte characters, spitting out U+FFFD characters at the invalid points
|
|
|
|
n = 1;
|
|
|
|
while (!isCodepointStart(p->s[*pos + n]))
|
|
|
|
n++;
|
|
|
|
addattr(p, *pos, *pos + n,
|
2017-05-17 22:37:16 -05:00
|
|
|
FUTURE_pango_attr_font_features_new(s));
|
2017-02-20 16:11:52 -06:00
|
|
|
return TRUE; // always delete; we're emptying the map
|
|
|
|
}
|
|
|
|
|
|
|
|
static void applyAndFreeFeatureAttributes(struct foreachParams *p)
|
|
|
|
{
|
|
|
|
g_hash_table_foreach_remove(p->features, applyFeatures, p);
|
|
|
|
g_hash_table_destroy(p->features);
|
|
|
|
}
|
|
|
|
|
2017-02-20 21:24:02 -06:00
|
|
|
static void unrefClosure(gpointer data)
|
|
|
|
{
|
|
|
|
g_closure_unref((GClosure *) data);
|
|
|
|
}
|
|
|
|
|
|
|
|
PangoAttrList *attrstrToPangoAttrList(uiDrawTextLayoutParams *p, GPtrArray **backgroundClosures)
|
2017-02-20 16:11:52 -06:00
|
|
|
{
|
|
|
|
struct foreachParams fep;
|
|
|
|
|
2017-02-20 18:51:00 -06:00
|
|
|
fep.s = uiAttributedStringString(p->String);
|
2017-02-20 16:11:52 -06:00
|
|
|
fep.attrs = pango_attr_list_new();
|
|
|
|
fep.features = g_hash_table_new_full(
|
|
|
|
featurePosHash, featurePosEqual,
|
2017-05-17 22:37:16 -05:00
|
|
|
g_free, g_free);
|
2017-02-20 21:24:02 -06:00
|
|
|
fep.backgroundClosures = g_ptr_array_new_with_free_func(unrefClosure);
|
2017-02-20 16:11:52 -06:00
|
|
|
uiAttributedStringForEachAttribute(p->String, processAttribute, &fep);
|
|
|
|
applyAndFreeFeatureAttributes(&fep);
|
2017-02-20 21:24:02 -06:00
|
|
|
*backgroundClosures = fep.backgroundClosures;
|
2017-02-20 16:11:52 -06:00
|
|
|
return fep.attrs;
|
|
|
|
}
|
2017-02-20 21:24:02 -06:00
|
|
|
|
|
|
|
void invokeBackgroundClosure(GClosure *closure, uiDrawContext *c, uiDrawTextLayout *layout, double x, double y)
|
|
|
|
{
|
|
|
|
GValue values[4] = {
|
|
|
|
// the zero-initialization is needed for g_value_init() to work
|
|
|
|
G_VALUE_INIT,
|
|
|
|
G_VALUE_INIT,
|
|
|
|
G_VALUE_INIT,
|
|
|
|
G_VALUE_INIT,
|
|
|
|
};
|
|
|
|
|
|
|
|
g_value_init(values + 0, G_TYPE_POINTER);
|
|
|
|
g_value_set_pointer(values + 0, c);
|
|
|
|
g_value_init(values + 1, G_TYPE_POINTER);
|
|
|
|
g_value_set_pointer(values + 1, layout);
|
|
|
|
g_value_init(values + 2, G_TYPE_DOUBLE);
|
|
|
|
g_value_set_double(values + 2, x);
|
|
|
|
g_value_init(values + 3, G_TYPE_DOUBLE);
|
|
|
|
g_value_set_double(values + 3, y);
|
|
|
|
g_closure_invoke(closure,
|
|
|
|
NULL,
|
|
|
|
4, values,
|
|
|
|
NULL);
|
|
|
|
}
|