// 2 january 2017 #import "uipriv_darwin.h" #import "draw.h" @interface lineInfo : NSObject @property NSRange glyphRange; @property NSRange characterRange; @property NSRect lineRect; @property CGFloat baselineOffset; @end @implementation lineInfo @end struct uiDrawTextLayout { // NSTextStorage is subclassed from NSMutableAttributedString NSTextStorage *attrstr; NSTextContainer *container; NSLayoutManager *layoutManager; // the width as passed into uiDrawTextLayout constructors double width; #if 0 /* TODO */ // the *actual* size of the frame // 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) CGSize size; #endif NSMutableArray *lineInfo; // for converting CFAttributedString indices to byte offsets size_t *u16tou8; size_t nu16tou8; // TODO I don't like the casing of this name }; static NSFont *fontdescToNSFont(uiDrawFontDescriptor *fd) { NSFontDescriptor *desc; NSFont *font; desc = fontdescToNSFontDescriptor(fd); font = [NSFont fontWithDescriptor:desc size:fd->Size]; [desc release]; return font; } static NSTextStorage *attrstrToTextStorage(uiAttributedString *s, uiDrawFontDescriptor *defaultFont) { NSString *nsstr; NSMutableDictionary *defaultAttrs; NSTextStorage *attrstr; nsstr = [[NSString alloc] initWithCharacters:attrstrUTF16(s) length:attrstrUTF16Len(s)]; defaultAttrs = [NSMutableDictionary new]; [defaultAttrs setObject:fontdescToNSFont(defaultFont) forKey:NSFontAttributeName]; attrstr = [[NSTextStorage alloc] initWithString:nsstr attributes:defaultAttrs]; [defaultAttrs release]; [nsstr release]; [attrstr beginEditing]; // TODO copy in the attributes [attrstr endEditing]; return attrstr; } // TODO fine-tune all the properties uiDrawTextLayout *uiDrawNewTextLayout(uiAttributedString *s, uiDrawFontDescriptor *defaultFont, double width) { uiDrawTextLayout *tl; CGFloat cgwidth; // TODO correct type? NSUInteger index; tl = uiNew(uiDrawTextLayout); tl->attrstr = attrstrToTextStorage(s, defaultFont); tl->width = width; // TODO the documentation on the size property implies this might not be necessary? cgwidth = (CGFloat) width; if (cgwidth < 0) cgwidth = CGFLOAT_MAX; // TODO rename to tl->textContainer tl->container = [[NSTextContainer alloc] initWithSize:NSMakeSize(cgwidth, CGFLOAT_MAX)]; // TODO pull the reference for this [tl->container setLineFragmentPadding:0]; tl->layoutManager = [[NSLayoutManager alloc] init]; [tl->layoutManager addTextContainer:tl->container]; [tl->attrstr addLayoutManager:tl->layoutManager]; // and force a re-layout (TODO get source [tl->layoutManager glyphRangeForTextContainer:tl->container]; // TODO equivalent of CTFrameProgression for RTL/LTR? // now collect line information; see https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/TextLayout/Tasks/CountLines.html tl->lineInfo = [NSMutableArray new]; index = 0; while (index < [tl->layoutManager numberOfGlyphs]) { NSRange glyphRange; NSRect lineRect; lineInfo *li; lineRect = [tl->layoutManager lineFragmentRectForGlyphAtIndex:index effectiveRange:&glyphRange]; li = [lineInfo new]; li.glyphRange = glyphRange; li.characterRange = [tl->layoutManager characterRangeForGlyphRange:li.glyphRange actualGlyphRange:NULL]; li.lineRect = lineRect; // and this is from http://www.cocoabuilder.com/archive/cocoa/308568-how-to-get-baseline-info.html and http://www.cocoabuilder.com/archive/cocoa/199283-height-and-location-of-text-within-line-in-nslayoutmanager-ignoring-spacing.html li.baselineOffset = [[tl->layoutManager typesetter] baselineOffsetInLayoutManager:tl->layoutManager glyphIndex:index]; [tl->lineInfo addObject:li]; [li release]; index = glyphRange.location + glyphRange.length; } // and finally copy the UTF-16 to UTF-8 index conversion table tl->u16tou8 = attrstrCopyUTF16ToUTF8(s, &(tl->nu16tou8)); return tl; } void uiDrawFreeTextLayout(uiDrawTextLayout *tl) { uiFree(tl->u16tou8); [tl->lineInfo release]; [tl->layoutManager release]; [tl->container release]; [tl->attrstr release]; uiFree(tl); } // TODO document that (x,y) is the top-left corner of the *entire frame* void uiDrawText(uiDrawContext *c, uiDrawTextLayout *tl, double x, double y) { NSGraphicsContext *gc; CGContextFlush(c->c); // just to be safe [NSGraphicsContext saveGraphicsState]; gc = [NSGraphicsContext graphicsContextWithGraphicsPort:c->c flipped:YES]; [NSGraphicsContext setCurrentContext:gc]; // TODO is this the right point? // TODO does this draw with the proper default styles? [tl->layoutManager drawGlyphsForGlyphRange:[tl->layoutManager glyphRangeForTextContainer:tl->container] atPoint:NSMakePoint(x, y)]; [gc flushGraphics]; // just to be safe [NSGraphicsContext restoreGraphicsState]; // TODO release gc? } // TODO update all of these { // 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 // TODO width doesn't include trailing whitespace... // TODO figure out how paragraph spacing should play into this // } void uiDrawTextLayoutExtents(uiDrawTextLayout *tl, double *width, double *height) { NSRect r; // see https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/TextLayout/Tasks/StringHeight.html r = [tl->layoutManager usedRectForTextContainer:tl->container]; *width = r.size.width; *height = r.size.height; } int uiDrawTextLayoutNumLines(uiDrawTextLayout *tl) { return [tl->lineInfo count]; } void uiDrawTextLayoutLineByteRange(uiDrawTextLayout *tl, int line, size_t *start, size_t *end) { lineInfo *li; li = (lineInfo *) [tl->lineInfo objectAtIndex:line]; *start = tl->u16tou8[li.characterRange.location]; *end = tl->u16tou8[li.characterRange.location + li.characterRange.length]; } void uiDrawTextLayoutLineGetMetrics(uiDrawTextLayout *tl, int line, uiDrawTextLayoutLineMetrics *m) { lineInfo *li; li = (lineInfo *) [tl->lineInfo objectAtIndex:line]; m->X = li.lineRect.origin.x; m->Y = li.lineRect.origin.y; m->Width = li.lineRect.size.width; m->Height = li.lineRect.size.height; // TODO is this correct? m->BaselineY = (m->X + m->Height) - li.baselineOffset; // TODO m->Ascent = 10000; m->Descent = 10000; m->Leading = 10000; // TODO m->ParagraphSpacingBefore = 0; m->LineHeightSpace = 0; m->LineSpacing = 0; m->ParagraphSpacing = 0; } void uiDrawTextLayoutHitTest(uiDrawTextLayout *tl, double x, double y, uiDrawTextLayoutHitTestResult *result) { #if 0 /* TODO */ CFIndex i; CTLineRef line; CFIndex pos; if (y >= 0) { for (i = 0; i < tl->nLines; i++) { double ltop, lbottom; ltop = tl->lineMetrics[i].Y; lbottom = ltop + tl->lineMetrics[i].Height; if (y >= ltop && y < lbottom) break; } result->YPosition = uiDrawTextLayoutHitTestPositionInside; if (i == tl->nLines) { i--; result->YPosition = uiDrawTextLayoutHitTestPositionAfter; } } else { i = 0; result->YPosition = uiDrawTextLayoutHitTestPositionBefore; } result->Line = i; result->XPosition = uiDrawTextLayoutHitTestPositionInside; if (x < tl->lineMetrics[i].X) { result->XPosition = uiDrawTextLayoutHitTestPositionBefore; // and forcibly return the first character x = tl->lineMetrics[i].X; } else if (x > (tl->lineMetrics[i].X + tl->lineMetrics[i].Width)) { result->XPosition = uiDrawTextLayoutHitTestPositionAfter; // and forcibly return the last character x = tl->lineMetrics[i].X + tl->lineMetrics[i].Width; } line = (CTLineRef) CFArrayGetValueAtIndex(tl->lines, i); // TODO copy the part from the docs about this point pos = CTLineGetStringIndexForPosition(line, CGPointMake(x, 0)); if (pos == kCFNotFound) { // TODO } result->Pos = tl->u16tou8[pos]; #endif } void uiDrawTextLayoutByteRangeToRectangle(uiDrawTextLayout *tl, size_t start, size_t end, uiDrawTextLayoutByteRangeRectangle *r) { }