2017-02-12 00:05:27 -06:00
|
|
|
// 12 february 2017
|
|
|
|
#import "uipriv_darwin.h"
|
|
|
|
|
|
|
|
// unlike the other systems, Core Text rolls family, size, weight, italic, width, AND opentype features into the "font" attribute
|
2017-02-12 12:41:52 -06:00
|
|
|
// TODO opentype features and AAT fvar table info is lost, so a handful of fonts in the font panel ("Titling" variants of some fonts and Skia and possibly others but those are the examples I know about) cannot be represented by uiDrawFontDescriptor; what *can* we do about this since this info is NOT part of the font on other platforms?
|
2017-02-12 00:05:27 -06:00
|
|
|
// TODO see if we could use NSAttributedString?
|
|
|
|
// TODO consider renaming this struct and the fep variable(s)
|
|
|
|
struct foreachParams {
|
|
|
|
CFMutableAttributedStringRef mas;
|
|
|
|
NSMutableDictionary *converted;
|
|
|
|
uiDrawFontDescriptor *defaultFont;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void ensureFontInRange(struct foreachParams *p, size_t start, size_t end)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
NSNumber *n;
|
|
|
|
uiDrawFontDescriptor *new;
|
|
|
|
|
|
|
|
for (i = start; i < end; i++) {
|
|
|
|
n = [NSNumber numberWithInteger:i];
|
|
|
|
if ([p->converted objectForKey:n] != nil)
|
|
|
|
continue;
|
|
|
|
new = uiNew(uiDrawFontDescriptor);
|
|
|
|
*new = *(p->defaultFont);
|
|
|
|
[p->converted setObject:[NSValue valueWithPointer:new] forKey:n];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void adjustFontInRange(struct foreachParams *p, size_t start, size_t end, void (^adj)(uiDrawFontDescriptor *desc))
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
NSNumber *n;
|
|
|
|
NSValue *v;
|
|
|
|
|
|
|
|
for (i = start; i < end; i++) {
|
|
|
|
n = [NSNumber numberWithInteger:i];
|
|
|
|
v = (NSValue *) [p->converted objectForKey:n];
|
|
|
|
adj((uiDrawFontDescriptor *) [v pointerValue]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-12 13:11:25 -06:00
|
|
|
static int processAttribute(uiAttributedString *s, uiAttributeSpec *spec, size_t start, size_t end, void *data)
|
2017-02-12 00:05:27 -06:00
|
|
|
{
|
|
|
|
struct foreachParams *p = (struct foreachParams *) data;
|
2017-02-12 19:27:47 -06:00
|
|
|
CFRange range;
|
|
|
|
CGColorSpaceRef colorspace;
|
|
|
|
CGColorRef color;
|
|
|
|
CGFloat components[4];
|
2017-02-12 00:05:27 -06:00
|
|
|
|
|
|
|
start = attrstrUTF8ToUTF16(s, start);
|
|
|
|
end = attrstrUTF8ToUTF16(s, end);
|
2017-02-12 19:27:47 -06:00
|
|
|
range.location = start;
|
|
|
|
range.length = end - start;
|
2017-02-12 13:11:25 -06:00
|
|
|
switch (spec->Type) {
|
2017-02-12 00:05:27 -06:00
|
|
|
case uiAttributeFamily:
|
|
|
|
ensureFontInRange(p, start, end);
|
|
|
|
adjustFontInRange(p, start, end, ^(uiDrawFontDescriptor *desc) {
|
2017-02-12 13:11:25 -06:00
|
|
|
desc->Family = (char *) (spec->Value);
|
2017-02-12 00:05:27 -06:00
|
|
|
});
|
|
|
|
break;
|
2017-02-12 12:41:52 -06:00
|
|
|
case uiAttributeSize:
|
|
|
|
ensureFontInRange(p, start, end);
|
|
|
|
adjustFontInRange(p, start, end, ^(uiDrawFontDescriptor *desc) {
|
2017-02-12 13:11:25 -06:00
|
|
|
desc->Size = spec->Double;
|
2017-02-12 12:41:52 -06:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case uiAttributeWeight:
|
|
|
|
ensureFontInRange(p, start, end);
|
|
|
|
adjustFontInRange(p, start, end, ^(uiDrawFontDescriptor *desc) {
|
2017-02-12 13:11:25 -06:00
|
|
|
desc->Weight = (uiDrawTextWeight) (spec->Value);
|
2017-02-12 12:41:52 -06:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case uiAttributeItalic:
|
|
|
|
ensureFontInRange(p, start, end);
|
|
|
|
adjustFontInRange(p, start, end, ^(uiDrawFontDescriptor *desc) {
|
2017-02-12 13:11:25 -06:00
|
|
|
desc->Italic = (uiDrawTextItalic) (spec->Value);
|
2017-02-12 12:41:52 -06:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case uiAttributeStretch:
|
|
|
|
ensureFontInRange(p, start, end);
|
|
|
|
adjustFontInRange(p, start, end, ^(uiDrawFontDescriptor *desc) {
|
2017-02-12 13:11:25 -06:00
|
|
|
desc->Stretch = (uiDrawTextStretch) (spec->Value);
|
2017-02-12 12:41:52 -06:00
|
|
|
});
|
|
|
|
break;
|
2017-02-12 19:27:47 -06:00
|
|
|
case uiAttributeColor:
|
|
|
|
colorspace = CGColorSpaceCreateWithName(kCGColorSpaceSRGB);
|
|
|
|
if (colorspace == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
components[0] = spec->R;
|
|
|
|
components[1] = spec->G;
|
|
|
|
components[2] = spec->B;
|
|
|
|
components[3] = spec->A;
|
|
|
|
color = CGColorCreate(colorspace, components);
|
|
|
|
CFRelease(colorspace);
|
|
|
|
CFAttributedStringSetAttribute(p->mas, range, kCTForegroundColorAttributeName, color);
|
|
|
|
CFRelease(color);
|
|
|
|
break;
|
2017-02-12 00:05:27 -06:00
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CTFontRef fontdescToCTFont(uiDrawFontDescriptor *fd)
|
|
|
|
{
|
|
|
|
CTFontDescriptorRef desc;
|
|
|
|
CTFontRef font;
|
|
|
|
|
|
|
|
desc = fontdescToCTFontDescriptor(fd);
|
|
|
|
font = CTFontCreateWithFontDescriptor(desc, fd->Size, NULL);
|
|
|
|
CFRelease(desc); // TODO correct?
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void applyAndFreeFontAttributes(struct foreachParams *p)
|
|
|
|
{
|
|
|
|
[p->converted enumerateKeysAndObjectsUsingBlock:^(NSNumber *key, NSValue *val, BOOL *stop) {
|
|
|
|
uiDrawFontDescriptor *desc;
|
|
|
|
CTFontRef font;
|
|
|
|
CFRange range;
|
|
|
|
|
|
|
|
desc = (uiDrawFontDescriptor *) [val pointerValue];
|
|
|
|
font = fontdescToCTFont(desc);
|
|
|
|
range.location = [key integerValue];
|
|
|
|
range.length = 1;
|
|
|
|
CFAttributedStringSetAttribute(p->mas, range, kCTFontAttributeName, font);
|
|
|
|
CFRelease(font);
|
|
|
|
uiFree(desc);
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
|
|
|
static const CTTextAlignment ctaligns[] = {
|
|
|
|
[uiDrawTextLayoutAlignLeft] = kCTTextAlignmentLeft,
|
|
|
|
[uiDrawTextLayoutAlignCenter] = kCTTextAlignmentCenter,
|
|
|
|
[uiDrawTextLayoutAlignRight] = kCTTextAlignmentRight,
|
|
|
|
};
|
|
|
|
|
|
|
|
static CTParagraphStyleRef mkParagraphStyle(uiDrawTextLayoutParams *p)
|
|
|
|
{
|
|
|
|
CTParagraphStyleRef ps;
|
|
|
|
CTParagraphStyleSetting settings[16];
|
|
|
|
size_t nSettings = 0;
|
|
|
|
|
|
|
|
settings[nSettings].spec = kCTParagraphStyleSpecifierAlignment;
|
|
|
|
settings[nSettings].valueSize = sizeof (CTTextAlignment);
|
|
|
|
settings[nSettings].value = ctaligns + p->Align;
|
|
|
|
nSettings++;
|
|
|
|
|
|
|
|
ps = CTParagraphStyleCreate(settings, nSettings);
|
|
|
|
if (ps == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
return ps;
|
|
|
|
}
|
|
|
|
|
|
|
|
CFAttributedStringRef attrstrToCoreFoundation(uiDrawTextLayoutParams *p)
|
|
|
|
{
|
|
|
|
CFStringRef cfstr;
|
|
|
|
CFMutableDictionaryRef defaultAttrs;
|
|
|
|
CTFontRef defaultCTFont;
|
|
|
|
CTParagraphStyleRef ps;
|
|
|
|
CFAttributedStringRef base;
|
|
|
|
CFMutableAttributedStringRef mas;
|
|
|
|
struct foreachParams fep;
|
|
|
|
|
|
|
|
cfstr = CFStringCreateWithCharacters(NULL, attrstrUTF16(p->String), attrstrUTF16Len(p->String));
|
|
|
|
if (cfstr == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
defaultAttrs = CFDictionaryCreateMutable(NULL, 0,
|
|
|
|
&kCFCopyStringDictionaryKeyCallBacks,
|
|
|
|
&kCFTypeDictionaryValueCallBacks);
|
|
|
|
if (defaultAttrs == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
defaultCTFont = fontdescToCTFont(p->DefaultFont);
|
|
|
|
CFDictionaryAddValue(defaultAttrs, kCTFontAttributeName, defaultCTFont);
|
|
|
|
CFRelease(defaultCTFont);
|
|
|
|
ps = mkParagraphStyle(p);
|
|
|
|
CFDictionaryAddValue(defaultAttrs, kCTParagraphStyleAttributeName, ps);
|
|
|
|
CFRelease(ps);
|
|
|
|
|
|
|
|
base = CFAttributedStringCreate(NULL, cfstr, defaultAttrs);
|
|
|
|
if (base == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
CFRelease(cfstr);
|
|
|
|
CFRelease(defaultAttrs);
|
|
|
|
mas = CFAttributedStringCreateMutableCopy(NULL, 0, base);
|
|
|
|
CFRelease(base);
|
|
|
|
|
|
|
|
CFAttributedStringBeginEditing(mas);
|
|
|
|
fep.mas = mas;
|
|
|
|
fep.converted = [NSMutableDictionary new];
|
|
|
|
fep.defaultFont = p->DefaultFont;
|
|
|
|
uiAttributedStringForEachAttribute(p->String, processAttribute, &fep);
|
|
|
|
applyAndFreeFontAttributes(&fep);
|
|
|
|
[fep.converted release];
|
|
|
|
CFAttributedStringEndEditing(mas);
|
|
|
|
|
|
|
|
return mas;
|
|
|
|
}
|