IOS, rounded corners, shadows, borders

The main function

  1. Rounded corner, controllable rounded corner direction.

  2. Frame related parameters, controllable frame direction.

  3. Shadow related parameters, controllable shadow direction.

  4. Images are generated using CoreGraphics and can be executed asynchronously.

  5. Super small cache footprint. Image size independent of view size, 1000 random values rounded corner shadow border images, 29MB.

Making the address

rendering

Use the sample

// UIAppearance [SFCSBView appearance].cornerRadius = 20; [SFCSBView appearance].rectCornner = UIRectCornerAllCorners; [SFCSBView appearance].shadowPosition = UIShadowPostionAll; [SFCSBView appearance].shadowRadius = 20; [SFCSBView appearance].borderColor = [UIColor systemBlueColor]; [SFCSBView appearance].borderWidth = 5; [SFCSBView appearance].borderPosition = UIBorderPostionAll; SFCSBView * shadowView = [SFCSBView new]; shadowView.cornerRadius = 20; shadowView.rectCornner = UIRectCornerAllCorners; shadowView.shadowPosition = UIShadowPostionAll; ShadowView. ShadowColor = [[UIColor blackColor] colorWithAlphaComponent: 0.6]; shadowView.shadowRadius = 20; shadowView.borderColor = UIColor.systemBlueColor; shadowView.borderWidth = 5; shadowView.borderPosition = UIBorderPostionAll;Copy the code

SFCSBView main implementation code

SFShadowImageMaker *shadowMaker = self.shadowProcessor; SFCornerImageMaker *cornerMaker = self.cornerProcessor; SFBorderImageMaker *borderMaker = self.borderProcessor; SFColorImageMaker *colorMaker = self.colorProcessor; if (self.handleMakers) self.handleMakers(@[colorMaker,cornerMaker,borderMaker,shadowMaker]); NSString *identifier = [NSString stringWithFormat:@"%@%@%@%@",colorMaker.identifier,cornerMaker.identifier,borderMaker.identifier,shadowMaker.identifier] ; CGRect backImageViewFrame = self.bounds; if (shadowMaker.isEnable){ backImageViewFrame = [shadowMaker viewRectForSize:self.bounds.size]; CGFloat insertValue = -1; backImageViewFrame = CGRectInset(backImageViewFrame, insertValue, insertValue); } // Every time the subView frame is modified, the view calls the layoutSubviews method. // Objective: To reduce frame updates in the case of highly reusable UICornerShadowView that is updated every time. // If the current frame is the same as the required frame, do not update the frame if (! [identifier isEqualToString:self.lastBackGroundImageIdentifer] || ! CGRectEqualToRect(self.backGroundImageView.frame, backImageViewFrame)) { self.backGroundImageView.frame = backImageViewFrame; } UIImage *cacheImage = [[SFCSBViewImageCache shared] objectForKey:identifier]; if (cacheImage) { self.backGroundImageView.image = cacheImage; self.backgroundColor = UIColor.clearColor; self.lastBackGroundImageIdentifer = identifier; return; } __weak typeof(self) weakSelf = self; dispatch_async(dispatch_get_global_queue(0, 0), ^{ UIImage *image = [SFImageMakerManager.shared startWithGenerator:colorMaker processors:@[cornerMaker,borderMaker,shadowMaker]]; if (shadowMaker.isEnable) { UIEdgeInsets inset = shadowMaker.convasEdgeInsets; CGFloat x = (image.size.width - inset.left - inset.right) / 2; CGFloat y = (image.size.height - inset.top - inset.bottom) / 2; image = [image resizableImageWithCapInsets:UIEdgeInsetsMake(y + inset.top, x + inset.left, y + inset.bottom, x + inset.right)]; }else{ image = [image resizableImageWithCapInsets:UIEdgeInsetsMake(image.size.height / 2, image.size.width / 2, image.size.height / 2, image.size.width / 2)]; } [SFCSBViewImageCache.shared setObject:image forKey:identifier]; dispatch_async(dispatch_get_main_queue(), ^{ if (! weakSelf) { return; } weakSelf.lastBackGroundImageIdentifer = identifier; weakSelf.backGroundImageView.image = image; }); });Copy the code

The performance test

// The following code:
// The simulator generates 1000 images, takes 1.3s, and the cache size is about 50MB.
// real machine iPhone7, generate 1000 times, time 0.9 seconds, cache size about 50MB.
// Each image is about 50KB. Randomised testing in projects yielded better results.
func testPerformanceExample(a) {
    // This is an example of a performance test case.
   self.measure {
        var cost: Int = 0
        for _ in (0.1000){
            rectCorner.radius = CGFloat(Int.random(in: 0..<10))
            rectCorner.position = .allCorners
            shadow.position = .all
            shadow.shadowColor = UIColor.black.withAlphaComponent(0.6)
            shadow.shadowBlurRadius = CGFloat(Int.random(in: 8..<20))
            border.color = UIColor.systemBlue
            border.width = CGFloat(Int.random(in: 0..<20))
            border.position = .all
            var maxValue = rectCorner.radius > (border.width + 1) && rectCorner.isEnable ? rectCorner.radius : border.width + 1
            maxValue = shadow.shadowBlurRadius > maxValue ? shadow.shadowBlurRadius : maxValue
            let size = CGSize.init(width: maxValue * 2, height: maxValue * 2)
            var image = SFColorImage.init(color: UIColor.white, size: size).general()
            image = rectCorner.process(image)
            image = border.process(image, rectCorner: rectCorner)
            image = shadow.process(image)
            if let cgimg = image.cgImage{
                cost += cgimg.height * cgimg.width * (cgimg.bitsPerPixel / cgimg.bitsPerComponent)
            }
        }
        print("\(cost / (1024 * 1024))MB")
        cost = 0}}Copy the code

IPhone7 iOS 13.3.1, UITableView.

Each cell has a different rounded corner shadow border (instant generation) 1000, quickly slide in the TableView. The FPS stayed around 60FPS and the CPU peaked at 140%.

This kind of:

In certain cases, there is reuse, but it needs to be judged by index, and it needs to be refreshed every time.

For example, only five cells are shown here. At 1000 cells, the peak value of the main line is 40% in the case of fast sliding.

A hundred thousand whys

  1. Why ultra-small cache occupancy

  2. Shadow direction control, border direction control, how to achieve?

    This is the shadow implementation with the same border.

  3. Suitable for what scenarios:

    • A large number of parameters in the same case, rest assured to eat. Don’t worry about the size of the View, use stretch implementation.
    • The CPU usage is still within the acceptable range when some parameters are the same.
    • All parameters are almost different, and I give up. High occupancy, unacceptable power consumption.