SoFunction
Updated on 2025-04-03

iOS Liangya Technology Solutions for insufficient UILabel performance

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

YYLabelDrawing 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 &lt; 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 &amp;&amp; !view &amp;&amp; !layer) continue;
        if (image &amp;&amp; !context) continue;
        if (view &amp;&amp; !targetView) continue;
        if (layer &amp;&amp; !targetLayer) continue;
        if (cancel &amp;&amp; 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!