Main reference YYKit
YYKit is profound and profound, just like Shaolin martial arts
Async View
For asynchronous + runloop draw when idle,
Because Apple's UILabel performance is not enough 6
Async Layer
Idea: UI operations must be placed in the main thread.
However, graphics processing can be placed in child threads.
(Develop the graphic context, draw, and take out the picture)
The last step is to put it in the main thread, it's fine
= image
In Custom View, the layer class is re-formed as an asynchronous layer
+ (Class)layerClass { return ; }
Create drawing tasks
Create a drawing task,YYAsyncLayerDisplayTask
The key is the drawing method insidedisplay
Get the graph context created by the asynchronous layer layer,
Adjust the coordinate system, (the origin of Core Text, at the bottom left)
The text map is rich text,
Rich text map is one frame,
One frame is split into many CTLines,
Display one by one
- (YYAsyncLayerDisplayTask *)newAsyncDisplayTask { // capture current state to display task NSString *text = _text; UIFont *fontX = _font; YYAsyncLayerDisplayTask *task = [YYAsyncLayerDisplayTask new]; CGFloat h_h = ; CGFloat w_w = ; = ^(CGContextRef context, CGSize size, BOOL(^isCancelled)(void)) { if (isCancelled()) return; //Here, the text will be reversed due to the drawing [[NSOperationQueue mainQueue] addOperationWithBlock:^{ CGContextSetTextMatrix(context, CGAffineTransformIdentity); CGContextTranslateCTM(context, 0, h_h); CGContextScaleCTM(context, 1.0, -1.0); }]; NSAttributedString* str = [[NSAttributedString alloc] initWithString:text attributes:@{NSFontAttributeName: fontX, NSForegroundColorAttributeName: }]; CTFramesetterRef ref = CTFramesetterCreateWithAttributedString((__bridge CFAttributedStringRef)str); CGPathRef path = CGPathCreateWithRect(CGRectMake(0, 0, w_w, 3000), nil); CTFrameRef pic = CTFramesetterCreateFrame(ref, CFRangeMake(0, 0), path, nil); CFArrayRef arr = CTFrameGetLines(pic); NSArray *array = (__bridge NSArray*)arr; int i = 0; int cnt = (int); CGPoint originsArray[cnt]; CTFrameGetLineOrigins(pic, CFRangeMake(0, 0), originsArray); CGFloat y_y = h_h - 60; while (i < cnt) { NSLog(@"%f", originsArray[i].y); CTLineRef line = (__bridge CTLineRef)(array[i]); CGContextSetTextPosition(context, 0, y_y - i * 30); CTLineDraw(line, context); i += 1; } }; return task; }
In Async Layer, start the drawing task,
First handle the inheritance relationship.
Then execute the drawing task mentioned above
- (void)display { = ; [self _displayAsync]; }
Perform drawing tasks,
Get the task and don't draw the content, forget it
Then judge whether it is its size (size), compliant or not
sizeCGSize(1, 1)
, just continue,
The child thread, first create the graphical context.
Then process the background color,
If it goes well, perform the drawing steps above.
From the graph context, take out the image and hand it over to
- (void)_displayAsync{ __strong id<YYAsyncLayerDelegate> delegate = (id); YYAsyncLayerDisplayTask *task = [delegate newAsyncDisplayTask]; if (!) { = nil; return; } CGSize size = ; BOOL opaque = ; CGFloat scale = ; CGColorRef backgroundColor = (opaque && ) ? CGColorRetain() : NULL; if ( < 1 || < 1) { CGImageRef image = (__bridge_retained CGImageRef)(); = nil; if (image) { dispatch_async(YYAsyncLayerGetReleaseQueue(), ^{ CFRelease(image); }); } CGColorRelease(backgroundColor); return; } dispatch_async(YYAsyncLayerGetDisplayQueue(), ^{ if (isCancelled()) { CGColorRelease(backgroundColor); return; } UIGraphicsBeginImageContextWithOptions(size, opaque, scale); CGContextRef context = UIGraphicsGetCurrentContext(); if (opaque) { CGContextSaveGState(context); { if (!backgroundColor || CGColorGetAlpha(backgroundColor) < 1) { CGContextSetFillColorWithColor(context, [UIColor whiteColor].CGColor); CGContextAddRect(context, CGRectMake(0, 0, * scale, * scale)); CGContextFillPath(context); } if (backgroundColor) { CGContextSetFillColorWithColor(context, backgroundColor); CGContextAddRect(context, CGRectMake(0, 0, * scale, * scale)); CGContextFillPath(context); } } CGContextRestoreGState(context); CGColorRelease(backgroundColor); } (context, size, isCancelled); if (isCancelled()) { UIGraphicsEndImageContext(); return; } UIImage *image = UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext(); if (isCancelled()) { return; } dispatch_async(dispatch_get_main_queue(), ^{ if (isCancelled() == NO) { = (__bridge id)(); } }); }); }
RunLoop
trigger
After setting the style, it will not be triggered immediately, repaint
Save it first
- (void)setText:(NSString *)text { _text = ; [[YYTransaction transactionWithTarget:self selector:@selector(contentsNeedUpdated)] commit]; }
Calling the drawing task of asynchronous layer
- (void)contentsNeedUpdated { // do update [ setNeedsDisplay]; }
Save events
First save the target and action of the method call as an object
YYTransactionSetup();
Singleton method, initialization
Add the method call object to the collection
@implementation YYTransaction + (YYTransaction *)transactionWithTarget:(id)target selector:(SEL)selector{ if (!target || !selector) return nil; YYTransaction *t = [YYTransaction new]; = target; = selector; return t; } - (void)commit { if (!_target || !_selector) return; YYTransactionSetup(); [transactionSet addObject:self]; }
When you are free, you will get things done without affecting the frame rate
The following singleton method initializes the event task collection,
Run loop callback, execute
No interference, Lordrunloop
static NSMutableSet *transactionSet = nil; static void YYRunLoopObserverCallBack(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info) { if ( == 0) return; NSSet *currentSet = transactionSet; transactionSet = [NSMutableSet new]; [currentSet enumerateObjectsUsingBlock:^(YYTransaction *transaction, BOOL *stop) { [ performSelector:]; }]; } static void YYTransactionSetup() { static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ transactionSet = [NSMutableSet new]; CFRunLoopRef runloop = CFRunLoopGetMain(); CFRunLoopObserverRef observer; observer = CFRunLoopObserverCreate(CFAllocatorGetDefault(), kCFRunLoopBeforeWaiting | kCFRunLoopExit, true, 0xFFFFFF, YYRunLoopObserverCallBack, NULL); CFRunLoopAddObserver(runloop, observer, kCFRunLoopCommonModes); CFRelease(observer); }); }
YYLabel
Very powerful rendering tool,
Based on the asynchronous rendering above
Supports various styles
Added an abstractionYYTextLayout
YYLabel
Drawing tasks in
If updates are required, create a new layout layout.
If centered / bottomed it, handle the layout
Then layout draw
@implementation YYLabel - (YYTextAsyncLayerDisplayTask *)newAsyncDisplayTask { // create display task YYTextAsyncLayerDisplayTask *task = [YYTextAsyncLayerDisplayTask new]; = ^(CGContextRef context, CGSize size, BOOL (^isCancelled)(void)) { if (isCancelled()) return; if ( == 0) return; YYTextLayout *drawLayout = layout; if (layoutNeedUpdate) { layout = [YYTextLayout layoutWithContainer:container text:text]; shrinkLayout = [YYLabel _shrinkLayoutWithLayout:layout]; if (isCancelled()) return; layoutUpdated = YES; drawLayout = shrinkLayout ? shrinkLayout : layout; } CGSize boundingSize = ; CGPoint point = CGPointZero; if (verticalAlignment == YYTextVerticalAlignmentCenter) { if () { = -( - ) * 0.5; } else { = ( - ) * 0.5; } } else if (verticalAlignment == YYTextVerticalAlignmentBottom) { if () { = -( - ); } else { = ( - ); } } point = YYTextCGPointPixelRound(point); [drawLayout drawInContext:context size:size point:point view:nil layer:nil debug:debug cancel:isCancelled]; }; return task; } @end
Draw various
Draw the background first,
Next, draw shadows,
Underline,
Word,
picture
frame
@implementation YYTextLayout - (void)drawInContext:(CGContextRef)context size:(CGSize)size point:(CGPoint)point view:(UIView *)view layer:(CALayer *)layer debug:(YYTextDebugOption *)debug cancel:(BOOL (^)(void))cancel{ @autoreleasepool { if ( && context) { if (cancel && cancel()) return; YYTextDrawBlockBorder(self, context, size, point, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawBorder(self, context, size, point, YYTextBorderTypeBackgound, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawShadow(self, context, size, point, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawDecoration(self, context, size, point, YYTextDecorationTypeUnderline, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawText(self, context, size, point, cancel); } if ( && (context || view || layer)) { if (cancel && cancel()) return; YYTextDrawAttachment(self, context, size, point, view, layer, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawInnerShadow(self, context, size, point, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawDecoration(self, context, size, point, YYTextDecorationTypeStrikethrough, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawBorder(self, context, size, point, YYTextBorderTypeNormal, cancel); } if ( && context) { if (cancel && cancel()) return; YYTextDrawDebug(self, context, size, point, debug); } } }
Enter to draw text
And pictures
The drawing granularity here, compare the above,
More finer particle size
The above is CTLine,
Here is CTRun
// Pay attention to the conditional judgment,// with Save/Restore Graphic Contextstatic void YYTextDrawAttachment(YYTextLayout *layout, CGContextRef context, CGSize size, CGPoint point, UIView *targetView, CALayer *targetLayer, BOOL (^cancel)(void)) { BOOL isVertical = ; CGFloat verticalOffset = isVertical ? ( - ) : 0; for (NSUInteger i = 0, max = ; i < max; i++) { YYTextAttachment *a = [i]; if (!) continue; UIImage *image = nil; UIView *view = nil; CALayer *layer = nil; if ([ isKindOfClass:[UIImage class]]) { image = ; } else if ([ isKindOfClass:[UIView class]]) { view = ; } else if ([ isKindOfClass:[CALayer class]]) { layer = ; } if (!image && !view && !layer) continue; if (image && !context) continue; if (view && !targetView) continue; if (layer && !targetLayer) continue; if (cancel && cancel()) break; CGSize asize = image ? : view ? : ; CGRect rect = ((NSValue *)[i]).CGRectValue; if (isVertical) { rect = UIEdgeInsetsInsetRect(rect, UIEdgeInsetRotateVertical()); } else { rect = UIEdgeInsetsInsetRect(rect, ); } rect = YYTextCGRectFitWithContentMode(rect, asize, ); rect = YYTextCGRectPixelRound(rect); rect = CGRectStandardize(rect); += + verticalOffset; += ; if (image) { CGImageRef ref = ; if (ref) { CGContextSaveGState(context); CGContextTranslateCTM(context, 0, CGRectGetMaxY(rect) + CGRectGetMinY(rect)); CGContextScaleCTM(context, 1, -1); CGContextDrawImage(context, rect, ref); CGContextRestoreGState(context); } } else if (view) { = rect; [targetView addSubview:view]; } else if (layer) { = rect; [targetLayer addSublayer:layer]; } } }
This article, the last question:
How do you obtain the drawing information of layout above?
Take the text first, create a CTFrame, and get the CTLine array in the CTFrame
Then iterate through each row and calculate
@implementation YYTextLayout + (YYTextLayout *)layoutWithContainer:(YYTextContainer *)container text:(NSAttributedString *)text range:(NSRange)range { // ... ctSetter = CTFramesetterCreateWithAttributedString((CFTypeRef)text); if (!ctSetter) goto fail; ctFrame = CTFramesetterCreateFrame(ctSetter, YYTextCFRangeFromNSRange(range), cgPath, (CFTypeRef)frameAttrs); if (!ctFrame) goto fail; lines = [NSMutableArray new]; ctLines = CTFrameGetLines(ctFrame); // ... for (NSUInteger i = 0, max = ; i < max; i++) { YYTextLine *line = lines[i]; if (truncatedLine && == ) line = truncatedLine; if ( > 0) { [attachments addObjectsFromArray:]; [attachmentRanges addObjectsFromArray:]; [attachmentRects addObjectsFromArray:]; for (YYTextAttachment *attachment in ) { if () { [attachmentContentsSet addObject:]; } } } } // ... }
github repo
This is the end of this article about the solution to insufficient UILabel performance in iOS Liangya Technology. For more related iOS UILabe content, please search for my previous articles or continue browsing the related articles below. I hope everyone will support me in the future!