2017-01-02 22:53:31 -06:00
|
|
|
// 2 january 2017
|
2016-01-12 00:12:35 -06:00
|
|
|
#import "uipriv_darwin.h"
|
2017-01-06 22:53:23 -06:00
|
|
|
#import "draw.h"
|
2016-01-12 00:12:35 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
struct uiDrawTextLayout {
|
|
|
|
CFAttributedStringRef attrstr;
|
|
|
|
double width;
|
2017-01-03 22:59:23 -06:00
|
|
|
CTFramesetterRef framesetter;
|
2017-01-05 20:36:07 -06:00
|
|
|
// note: technically, metrics returned from frame are relative to CGPathGetPathBoundingBox(tl->path)
|
|
|
|
// however, from what I can gather, for a path created by CGPathCreateWithRect(), like we do (with a NULL transform), CGPathGetPathBoundingBox() seems to just return the standardized form of the rect used to create the path
|
|
|
|
// (this I confirmed through experimentation)
|
|
|
|
// so we can just use tl->size for adjustments
|
|
|
|
// we don't need to adjust coordinates by any origin since our rect origin is (0, 0)
|
2017-01-03 22:59:23 -06:00
|
|
|
CGSize size;
|
|
|
|
CGPathRef path;
|
2017-01-04 22:50:08 -06:00
|
|
|
CTFrameRef frame;
|
|
|
|
CFArrayRef lines;
|
2016-01-12 00:12:35 -06:00
|
|
|
};
|
|
|
|
|
2017-01-03 11:18:17 -06:00
|
|
|
static CTFontRef fontdescToCTFont(uiDrawFontDescriptor *fd)
|
|
|
|
{
|
|
|
|
CTFontDescriptorRef desc;
|
|
|
|
CTFontRef font;
|
|
|
|
|
2017-01-03 22:59:23 -06:00
|
|
|
desc = fontdescToCTFontDescriptor(fd);
|
|
|
|
font = CTFontCreateWithFontDescriptor(desc, fd->Size, NULL);
|
2017-01-03 11:18:17 -06:00
|
|
|
CFRelease(desc); // TODO correct?
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CFAttributedStringRef attrstrToCoreFoundation(uiAttributedString *s, uiDrawFontDescriptor *defaultFont)
|
2016-01-12 00:12:35 -06:00
|
|
|
{
|
|
|
|
CFStringRef cfstr;
|
2017-01-03 11:18:17 -06:00
|
|
|
CFMutableDictionaryRef defaultAttrs;
|
|
|
|
CTFontRef defaultCTFont;
|
2017-01-02 22:53:31 -06:00
|
|
|
CFAttributedStringRef base;
|
|
|
|
CFMutableAttributedStringRef mas;
|
2016-01-12 00:12:35 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
cfstr = CFStringCreateWithCharacters(NULL, attrstrUTF16(s), attrstrUTF16Len(s));
|
|
|
|
if (cfstr == NULL) {
|
|
|
|
// TODO
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
2017-01-03 11:18:17 -06:00
|
|
|
defaultAttrs = CFDictionaryCreateMutable(NULL, 1,
|
2017-01-02 22:53:31 -06:00
|
|
|
&kCFCopyStringDictionaryKeyCallBacks,
|
|
|
|
&kCFTypeDictionaryValueCallBacks);
|
|
|
|
if (defaultAttrs == NULL) {
|
|
|
|
// TODO
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
2017-01-03 11:18:17 -06:00
|
|
|
defaultCTFont = fontdescToCTFont(defaultFont);
|
|
|
|
CFDictionaryAddValue(defaultAttrs, kCTFontAttributeName, defaultCTFont);
|
|
|
|
CFRelease(defaultCTFont);
|
2016-01-12 00:12:35 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
base = CFAttributedStringCreate(NULL, cfstr, defaultAttrs);
|
|
|
|
if (base == NULL) {
|
|
|
|
// TODO
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
2017-01-02 22:53:31 -06:00
|
|
|
CFRelease(cfstr);
|
|
|
|
CFRelease(defaultAttrs);
|
|
|
|
mas = CFAttributedStringCreateMutableCopy(NULL, 0, base);
|
|
|
|
CFRelease(base);
|
2016-01-12 00:46:28 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
CFAttributedStringBeginEditing(mas);
|
|
|
|
// TODO copy in the attributes
|
|
|
|
CFAttributedStringEndEditing(mas);
|
2016-01-12 00:12:35 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
return mas;
|
2016-01-12 00:46:28 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
uiDrawTextLayout *uiDrawNewTextLayout(uiAttributedString *s, uiDrawFontDescriptor *defaultFont, double width)
|
2016-01-12 00:46:28 -06:00
|
|
|
{
|
2017-01-03 22:59:23 -06:00
|
|
|
uiDrawTextLayout *tl;
|
|
|
|
CGFloat cgwidth;
|
|
|
|
CFRange range, unused;
|
|
|
|
CGRect rect;
|
|
|
|
|
|
|
|
tl = uiNew(uiDrawTextLayout);
|
|
|
|
tl->attrstr = attrstrToCoreFoundation(s, defaultFont);
|
|
|
|
range.location = 0;
|
|
|
|
range.length = CFAttributedStringGetLength(tl->attrstr);
|
|
|
|
tl->width = width;
|
|
|
|
|
|
|
|
// TODO CTFrameProgression for RTL/LTR
|
|
|
|
// TODO kCTParagraphStyleSpecifierMaximumLineSpacing, kCTParagraphStyleSpecifierMinimumLineSpacing, kCTParagraphStyleSpecifierLineSpacingAdjustment for line spacing
|
|
|
|
tl->framesetter = CTFramesetterCreateWithAttributedString(tl->attrstr);
|
|
|
|
if (tl->framesetter == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
|
|
|
cgwidth = (CGFloat) width;
|
|
|
|
if (cgwidth < 0)
|
|
|
|
cgwidth = CGFLOAT_MAX;
|
|
|
|
// TODO these seem to be floor()'d or truncated?
|
|
|
|
// TODO double check to make sure this TODO was right
|
|
|
|
tl->size = CTFramesetterSuggestFrameSizeWithConstraints(tl->framesetter,
|
|
|
|
range,
|
|
|
|
// TODO kCTFramePathWidthAttributeName?
|
|
|
|
NULL,
|
|
|
|
CGSizeMake(cgwidth, CGFLOAT_MAX),
|
|
|
|
&unused); // not documented as accepting NULL
|
|
|
|
|
|
|
|
rect.origin = CGZeroPoint;
|
|
|
|
rect.size = tl->size;
|
|
|
|
tl->path = CGPathCreateWithRect(rect, NULL);
|
2017-01-04 22:50:08 -06:00
|
|
|
tl->frame = CTFramesetterCreateFrame(tl->framesetter,
|
|
|
|
range,
|
|
|
|
tl->path,
|
|
|
|
// TODO kCTFramePathWidthAttributeName?
|
|
|
|
NULL);
|
|
|
|
if (tl->frame == NULL) {
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
|
|
|
tl->lines = CTFrameGetLines(tl->frame);
|
|
|
|
|
2017-01-03 22:59:23 -06:00
|
|
|
return tl;
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawFreeTextLayout(uiDrawTextLayout *tl)
|
2016-01-12 00:12:35 -06:00
|
|
|
{
|
2017-01-04 22:50:08 -06:00
|
|
|
// TODO release tl->lines?
|
|
|
|
CFRelease(tl->frame);
|
2017-01-03 22:59:23 -06:00
|
|
|
CFRelease(tl->path);
|
|
|
|
CFRelease(tl->framesetter);
|
2017-01-03 11:18:17 -06:00
|
|
|
CFRelease(tl->attrstr);
|
|
|
|
uiFree(tl);
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
|
|
|
|
2017-01-06 22:53:23 -06:00
|
|
|
// TODO double-check helvetica
|
|
|
|
// TODO document that (x,y) is the top-left corner of the *entire frame*
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawText(uiDrawContext *c, uiDrawTextLayout *tl, double x, double y)
|
2016-01-15 19:18:53 -06:00
|
|
|
{
|
2017-01-06 22:53:23 -06:00
|
|
|
CGContextSaveGState(c->c);
|
|
|
|
|
|
|
|
// Core Text doesn't draw onto a flipped view correctly; we have to pretend it was unflipped
|
|
|
|
// see the iOS bits of the first example at https://developer.apple.com/library/mac/documentation/StringsTextFonts/Conceptual/CoreText_Programming/LayoutOperations/LayoutOperations.html#//apple_ref/doc/uid/TP40005533-CH12-SW1 (iOS is naturally flipped)
|
|
|
|
// TODO how is this affected by a non-identity CTM?
|
|
|
|
CGContextTranslateCTM(c->c, 0, cheight);
|
|
|
|
CGContextScaleCTM(c->c, 1.0, -1.0);
|
|
|
|
CGContextSetTextMatrix(c->c, CGAffineTransformIdentity);
|
|
|
|
|
|
|
|
// wait, that's not enough; we need to offset y values to account for our new flipping
|
|
|
|
y = c->height - y;
|
|
|
|
|
|
|
|
// CTFrameDraw() draws in the path we specified when creating the frame
|
|
|
|
// this means that in our usage, CTFrameDraw() will draw at (0,0)
|
|
|
|
// so move the origin to be at (x,y) instead
|
|
|
|
// TODO are the signs correct?
|
|
|
|
CGContextTranslateCTM(c->c, x, y);
|
|
|
|
|
|
|
|
CTFrameDraw(tl->frame, c->c);
|
|
|
|
|
|
|
|
CGContextRestoreGState(c->c);
|
2016-01-15 19:18:53 -06:00
|
|
|
}
|
|
|
|
|
2017-01-05 16:56:47 -06:00
|
|
|
// TODO document that the width and height of a layout is not necessarily the sum of the widths and heights of its constituent lines; this is definitely untrue on OS X, where lines are placed in such a way that the distance between baselines is always integral
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawTextLayoutExtents(uiDrawTextLayout *tl, double *width, double *height)
|
2016-01-16 12:34:22 -06:00
|
|
|
{
|
2017-01-05 16:55:05 -06:00
|
|
|
*width = tl->size.width;
|
|
|
|
*height = tl->size.height;
|
2016-01-16 12:34:22 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
int uiDrawTextLayoutNumLines(uiDrawTextLayout *tl)
|
2016-01-16 12:34:22 -06:00
|
|
|
{
|
2017-01-04 22:50:08 -06:00
|
|
|
return CFArrayGetCount(tl->lines);
|
2016-01-16 12:34:22 -06:00
|
|
|
}
|
2016-01-15 21:48:38 -06:00
|
|
|
|
2017-01-04 22:50:08 -06:00
|
|
|
// TODO release when done?
|
|
|
|
#define getline(tl, line) ((CTLineRef) CFArrayGetValueAtIndex(tl->lines, line))
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawTextLayoutLineByteRange(uiDrawTextLayout *tl, int line, size_t *start, size_t *end)
|
2016-01-15 19:18:53 -06:00
|
|
|
{
|
2017-01-04 22:50:08 -06:00
|
|
|
CTLineRef lr;
|
|
|
|
CFRange range;
|
|
|
|
|
|
|
|
lr = getline(tl, line);
|
|
|
|
range = CTLineGetStringRange(lr);
|
|
|
|
// TODO set start and end
|
2016-01-15 19:18:53 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawTextLayoutLineGetMetrics(uiDrawTextLayout *tl, int line, uiDrawTextLayoutLineMetrics *m)
|
2016-01-12 00:12:35 -06:00
|
|
|
{
|
2017-01-04 22:50:08 -06:00
|
|
|
CTLineRef lr;
|
|
|
|
CFRange range;
|
|
|
|
CGPoint origin;
|
|
|
|
CGFloat ascent, descent, leading;
|
|
|
|
|
|
|
|
range.location = line;
|
|
|
|
range.length = 1;
|
|
|
|
CTFrameGetLineOrigins(tl->frame, range, &origin);
|
|
|
|
m->X = origin.x;
|
2017-01-05 20:36:07 -06:00
|
|
|
// and remember that the frame is flipped
|
|
|
|
m->BaselineY = tl->size.height - origin.y;
|
2017-01-04 22:50:08 -06:00
|
|
|
|
|
|
|
lr = getline(tl, line);
|
|
|
|
// though CTLineGetTypographicBounds() returns 0 on error, it also returns 0 on an empty string, so we can't reasonably check for error
|
2017-01-05 16:55:05 -06:00
|
|
|
m->Width = CTLineGetTypographicBounds(lr, &ascent, &descent, &leading);
|
2017-01-04 22:50:08 -06:00
|
|
|
m->Ascent = ascent;
|
|
|
|
m->Descent = descent;
|
|
|
|
m->Leading = leading;
|
2016-01-12 00:12:35 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawTextLayoutByteIndexToGraphemeRect(uiDrawTextLayout *tl, size_t pos, int *line, double *x, double *y, double *width, double *height)
|
2016-01-12 00:12:35 -06:00
|
|
|
{
|
|
|
|
}
|
2016-01-13 14:17:49 -06:00
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
uiDrawTextLayoutHitTestResult uiDrawTextLayoutHitTest(uiDrawTextLayout *tl, double x, double y, size_t *byteIndex, int *line)
|
2016-05-23 00:11:43 -05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-01-02 22:53:31 -06:00
|
|
|
void uiDrawTextLayoutByteRangeToRectangle(uiDrawTextLayout *tl, size_t start, size_t end, uiDrawTextLayoutByteRangeRectangle *r)
|
2016-04-19 15:06:50 -05:00
|
|
|
{
|
|
|
|
}
|