美文网首页
iOS Core Animation Advanced Tech

iOS Core Animation Advanced Tech

作者: Helly1024 | 来源:发表于2016-04-13 17:45 被阅读62次

    专用图层

    CAShapeLayer

    CAShapeLayer的优点:

    • 不会出现像素化
    • 不会被图层边界剪裁掉
    • 渲染快速,使用了硬件加速
    • 高效使用内存,不需要像普通CALayer一样创建一个寄宿图形

    CAShapeLayer可以用来绘制所有能够通过CGPath来表示的形状。这个形状不一定要闭合,图层路径也不一定要不可破,事实上可以在一个图层上绘制好几个不同的形状。你可以控制一些属性比如lineWith(线宽,用点表示单位),lineCap(线条结尾的样子),和lineJoin(线条之间的结合点的样子).但是在图层层面只有一次机会设置这些属性。如果想用不同颜色或风格来绘制多个形状,就必须每个形状准备一个图层了。

    示例代码:

    @implementation ShapeLayerViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
    
        // 绘制火柴人
        UIBezierPath *path = [[UIBezierPath alloc] init];
        [path moveToPoint:CGPointMake(174, 100)];              
        [path addArcWithCenter:CGPointMake(150, 100)
                        radius:25
                    startAngle:0
                      endAngle:2*M_PI
                     clockwise:YES];
        
        [path moveToPoint:CGPointMake(150, 125)];
        [path addLineToPoint:CGPointMake(150, 175)];
        [path addLineToPoint:CGPointMake(125, 225)];
        
        [path moveToPoint:CGPointMake(150, 175)];
        [path addLineToPoint:CGPointMake(175, 225)];
        
        [path moveToPoint:CGPointMake(100, 145)];
        [path addLineToPoint:CGPointMake(200, 145)];
        
        CAShapeLayer *shapeLayer = [[CAShapeLayer alloc] init];
        shapeLayer.strokeColor = [UIColor greenColor].CGColor;
        shapeLayer.fillColor = [UIColor clearColor].CGColor;
        shapeLayer.lineWidth = 5;
        shapeLayer.lineJoin = kCALineJoinRound;
        
        shapeLayer.path = path.CGPath;
        
        [self.view.layer addSublayer:shapeLayer]; 
    }
    

    使用CAShapeLayer可以单独指定矩形的一个或多个角为圆角:

    CGRect rect = CGRectMake(50, 50, 100, 100);
    CGSize radii = CGSizeMake(20, 20);
    
    UIRectCorner corners = UIRectCornerTopLeft | UIRectCornerBottomRight;
    
    UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:rect 
                                                byRoundingCorners:corners 
                                                      cornerRadii:radii];
    

    CATextLayer

    示例代码:

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        CATextLayer *layer = [CATextLayer layer];
        layer.frame = self.view.bounds;
        [self.view.layer addSublayer:layer];
        
        layer.foregroundColor = [UIColor greenColor].CGColor;
        layer.alignmentMode = kCAAlignmentCenter;
        layer.wrapped = YES;
        
        layer.contentsScale = [UIScreen mainScreen].scale;
        
        UIFont *font = [UIFont systemFontOfSize:28.0f];
        
        CFStringRef fontName = (__bridge CFStringRef)font.fontName;
        CGFontRef fontRef = CGFontCreateWithFontName(fontName);
        layer.font = fontRef;
        layer.fontSize = font.pointSize;
        CGFontRelease(fontRef);
        
        NSString *text = @"Hellow World!";
        
        layer.string = text;
    }
    

    富文本

    示例代码:

    NSString *text = @"\n\n\n\nLorem ipsum dolor sit amet, consectetur adipiscing \n elit. Quisque massa arcu, eleifend vel varius in, facilisis pulvinar \n leo. Nunc quis nunc at mauris pharetra condimentum ut ac neque. Nunc \n elementum, libero ut porttitor dictum, diam odio congue lacus, vel \n fringilla sapien diam at purus. Etiam suscipit pretium nunc sit amet \n lobortis";
    
    
    //create attributed string
    NSMutableAttributedString *string = nil;
    string = [[NSMutableAttributedString alloc] initWithString:text];
    
    //convert UIFont to a CTFont
    CFStringRef fontName = (__bridge CFStringRef)font.fontName;
    CGFloat fontSize = font.pointSize;
    CTFontRef fontRef = CTFontCreateWithName(fontName, fontSize, NULL);
    
    //set text attributes
    NSDictionary *attribs = @{
                              (__bridge id)kCTForegroundColorAttributeName:(__bridge id)[UIColor blackColor].CGColor,
                              (__bridge id)kCTFontAttributeName: (__bridge id)fontRef
                              };
    
    [string setAttributes:attribs range:NSMakeRange(0, [text length])];
    attribs = @{
                (__bridge id)kCTForegroundColorAttributeName: (__bridge id)[UIColor redColor].CGColor,
                (__bridge id)kCTUnderlineStyleAttributeName: @(kCTUnderlineStyleSingle),
                (__bridge id)kCTFontAttributeName: (__bridge id)fontRef
                };
    [string setAttributes:attribs range:NSMakeRange(6, 5)];
    
    //release the CTFont we created earlier
    CFRelease(fontRef);
    
    //    layer.fontSize = font.pointSize;
    layer.string = string;
    

    CATransformLayer

    通过CATransformLayer我们可以在构造复杂的3D事物时,组织独立的元素,并且为每个元素添加不同的3D转换效果。

    示例代码:

    @interface ViewController ()
    
    @property (nonatomic, weak) IBOutlet UIView *containerView;
    
    @end
    
    @implementation ViewController
    
    - (CALayer *)faceWithTransform:(CATransform3D)transform
    {
      //create cube face layer
      CALayer *face = [CALayer layer];
      face.frame = CGRectMake(-50, -50, 100, 100);
    
      //apply a random color
      CGFloat red = (rand() / (double)INT_MAX);
      CGFloat green = (rand() / (double)INT_MAX);
      CGFloat blue = (rand() / (double)INT_MAX);
      face.backgroundColor = [UIColor colorWithRed:red green:green blue:blue alpha:1.0].CGColor;
    
      //apply the transform and return
      face.transform = transform;
      return face;
    }
    
    - (CALayer *)cubeWithTransform:(CATransform3D)transform
    {
      //create cube layer
      CATransformLayer *cube = [CATransformLayer layer];
    
      //add cube face 1
      CATransform3D ct = CATransform3DMakeTranslation(0, 0, 50);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //add cube face 2
      ct = CATransform3DMakeTranslation(50, 0, 0);
      ct = CATransform3DRotate(ct, M_PI_2, 0, 1, 0);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //add cube face 3
      ct = CATransform3DMakeTranslation(0, -50, 0);
      ct = CATransform3DRotate(ct, M_PI_2, 1, 0, 0);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //add cube face 4
      ct = CATransform3DMakeTranslation(0, 50, 0);
      ct = CATransform3DRotate(ct, -M_PI_2, 1, 0, 0);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //add cube face 5
      ct = CATransform3DMakeTranslation(-50, 0, 0);
      ct = CATransform3DRotate(ct, -M_PI_2, 0, 1, 
      ct = CATransform3DRotate(ct, -M_PI_2, 0, 1, 0);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //add cube face 6
      ct = CATransform3DMakeTranslation(0, 0, -50);
      ct = CATransform3DRotate(ct, M_PI, 0, 1, 0);
      [cube addSublayer:[self faceWithTransform:ct]];
    
      //center the cube layer within the container
      CGSize containerSize = self.containerView.bounds.size;
      cube.position = CGPointMake(containerSize.width / 2.0, containerSize.height / 2.0);
    
      //apply the transform and return
      cube.transform = transform;
      return cube;
    }
    
    - (void)viewDidLoad
    {
      [super viewDidLoad];
    
      //set up the perspective transform
      CATransform3D pt = CATransform3DIdentity;
      pt.m34 = -1.0 / 500.0;
      self.containerView.layer.sublayerTransform = pt;
    
      //set up the transform for cube 1 and add it
      CATransform3D c1t = CATransform3DIdentity;
      c1t = CATransform3DTranslate(c1t, -100, 0, 0);
      CALayer *cube1 = [self cubeWithTransform:c1t];
      [self.containerView.layer addSublayer:cube1];
    
      //set up the transform for cube 2 and add it
      CATransform3D c2t = CATransform3DIdentity;
      c2t = CATransform3DTranslate(c2t, 100, 0, 0);
      c2t = CATransform3DRotate(c2t, -M_PI_4, 1, 0, 0);
      c2t = CATransform3DRotate(c2t, -M_PI_4, 0, 1, 0);
      CALayer *cube2 = [self cubeWithTransform:c2t];
      [self.containerView.layer addSublayer:cube2];
    }
    @end
    

    CAGradientLayer

    CAGradientLayer是用来生成两种或更多颜色平滑渐变的。

    示例代码:

    @implementation GradientLayerViewController
    
    - (void)viewDidLoad {
        
        [super viewDidLoad];
        
        CAGradientLayer *gradientLayer = [CAGradientLayer layer];
        
        gradientLayer.frame = self.view.bounds;
        
        // Set Colors
        gradientLayer.colors = @[
                                 (__bridge id)[UIColor redColor].CGColor,
                                 (__bridge id)[UIColor blueColor].CGColor,
                                 (__bridge id)[UIColor grayColor].CGColor,
                                 (__bridge id)[UIColor greenColor].CGColor,
                                 (__bridge id)[UIColor cyanColor].CGColor,
                                 (__bridge id)[UIColor yellowColor].CGColor];
    
        // Set the range of each color
        gradientLayer.locations = @[@0, @0.2, @0.29, @0.38, @0.57, @1];
        
        // Set the startPoint and endPoint
        gradientLayer.startPoint = CGPointMake(0, 0);
        gradientLayer.endPoint = CGPointMake(1, 1);
        
        [self.view.layer addSublayer:gradientLayer];
    }
    
    
    @end
    

    CAReplicatorLayer

    CAReplicatorLayer的目的是为了高效生成许多相似的图层。它会绘制一个或多个图层的子图层,并在每个复制体上应用不同的变换。

    重复图层示例代码:

    @interface ViewController ()
    
    @property (nonatomic, weak) IBOutlet UIView *containerView;
    
    @end
    
    @implementation ViewController
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        //create a replicator layer and add it to our view
        CAReplicatorLayer *replicator = [CAReplicatorLayer layer];
        replicator.frame = self.containerView.bounds;
        [self.containerView.layer addSublayer:replicator];
    
        //configure the replicator
        replicator.instanceCount = 10;
    
        //apply a transform for each instance
        CATransform3D transform = CATransform3DIdentity;
        transform = CATransform3DTranslate(transform, 0, 200, 0);
        transform = CATransform3DRotate(transform, M_PI / 5.0, 0, 0, 1);
        transform = CATransform3DTranslate(transform, 0, -200, 0);
        replicator.instanceTransform = transform;
    
        //apply a color shift for each instance
        replicator.instanceBlueOffset = -0.1;
        replicator.instanceGreenOffset = -0.1;
    
        //create a sublayer and place it inside the replicator
        CALayer *layer = [CALayer layer];
        layer.frame = CGRectMake(100.0f, 100.0f, 100.0f, 100.0f);
        layer.backgroundColor = [UIColor whiteColor].CGColor;
        [replicator addSublayer:layer];
    }
    @end
    

    带倒影效果的VIew示例代码:

    #import "ReflectionView.h"
    
    @implementation ReflectionView
    
    + (Class)layerClass
    {
        return [CAReplicatorLayer class];
    }
    
    - (void)setUp
    {
        //configure replicator
        CAReplicatorLayer *layer = (CAReplicatorLayer *)self.layer;
        layer.instanceCount = 2;
    
        //move reflection instance below original and flip vertically
        CATransform3D transform = CATransform3DIdentity;
        CGFloat verticalOffset = self.bounds.size.height + 2;
        transform = CATransform3DTranslate(transform, 0, verticalOffset, 0);
        transform = CATransform3DScale(transform, 1, -1, 0);
        layer.instanceTransform = transform;
    
        //reduce alpha of reflection layer
        layer.instanceAlphaOffset = -0.6;
    }
    
    - (id)initWithFrame:(CGRect)frame
    {
        //this is called when view is created in code
        if ((self = [super initWithFrame:frame])) {
            [self setUp];
        }
        return self;
    }
    
    - (void)awakeFromNib
    {
        //this is called when view is created from a nib
        [self setUp];
    }
    @end
    

    CAScrollLayer

    自动适应图层的bounds使图层上指定点得内容出现在图层原点:

    [(CAScrollLayer *)self.layer scrollToPoint:offset];
    

    从图层树中查找并找到第一个可用的CAScrollLayer,然后滑动它使得指定点/矩形区域成为可视的:

    - (void)scrollPoint:(CGPoint)point;
    - (void)scrollRectToVisible:(CGRect)rect;
    

    通过CAScrollLayer实现滑动视图代码清单:

    #import "ScrollView.h"
    @implementation ScrollView
    + (Class)layerClass
    {
        return [CAScrollLayer class];
    }
    
    - (void)setUp
    {
        //enable clipping
        self.layer.masksToBounds = YES;
    
        //attach pan gesture recognizer
        UIPanGestureRecognizer *recognizer = nil;
        recognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(pan:)];
        [self addGestureRecognizer:recognizer];
    }
    
    - (id)initWithFrame:(CGRect)frame
    {
        //this is called when view is created in code
        if ((self = [super initWithFrame:frame])) {
            [self setUp];
        }
        return self;
    }
    
    - (void)awakeFromNib {
        //this is called when view is created from a nib
        [self setUp];
    }
    
    - (void)pan:(UIPanGestureRecognizer *)recognizer
    {
        //get the offset by subtracting the pan gesture
        //translation from the current bounds origin
        CGPoint offset = self.bounds.origin;
        offset.x -= [recognizer translationInView:self].x;
        offset.y -= [recognizer translationInView:self].y;
    
        //scroll the layer
        [(CAScrollLayer *)self.layer scrollToPoint:offset];
    
        //reset the pan gesture translation
        [recognizer setTranslation:CGPointZero inView:self];
    }
    @end
    

    AVPlayerLayer

    示例代码:

    - (void)viewDidLoad
    {
        [super viewDidLoad];
        //get video URL
        NSURL *URL = [[NSBundle mainBundle] URLForResource:@"Ship" withExtension:@"mp4"];
    
        //create player and player layer
        AVPlayer *player = [AVPlayer playerWithURL:URL];
        AVPlayerLayer *playerLayer = [AVPlayerLayer playerLayerWithPlayer:player];
    
        //set player layer frame and attach it to our view
        playerLayer.frame = self.containerView.bounds;
        [self.containerView.layer addSublayer:playerLayer];
    
        //play the video
        [player play];
    }
    

    给视频增加变换,边框和圆角,示例代码:

    - (void)viewDidLoad
    {
        ...
        //set player layer frame and attach it to our view
        playerLayer.frame = self.containerView.bounds;
        [self.containerView.layer addSublayer:playerLayer];
    
        //transform layer
        CATransform3D transform = CATransform3DIdentity;
        transform.m34 = -1.0 / 500.0;
        transform = CATransform3DRotate(transform, M_PI_4, 1, 1, 0);
        playerLayer.transform = transform;
        
        //add rounded corners and border
        playerLayer.masksToBounds = YES;
        playerLayer.cornerRadius = 20.0;
        playerLayer.borderColor = [UIColor redColor].CGColor;
        playerLayer.borderWidth = 5.0;
    
        //play the video
        [player play];
    }

    相关文章

      网友评论

          本文标题:iOS Core Animation Advanced Tech

          本文链接:https://www.haomeiwen.com/subject/hrxglttx.html