骨架图
相比大家现在听到这个词应该不会感到陌生,如今流行的App的内容页都是采用骨架图loading,比之间的小菊花看上去高大上许多。那么作为一个iOS开发者如何快速做出一个骨架图呢?当然有两种方式:一种是站在巨人的肩膀上开发,另外一种是自己摸索。其实这个东西并不难,只是用到了CAGradientLayer,平时开发过程中可能很少用的这个,所以有些人可能不是很了解。只需要看一下苹果的官方文档,我相信没有人不会用这个。废话不多说,我们先来看一下效果图。
效果图
最简单的方式就是创建多个view进行排版,然后自定义view的layer层,在加上动画。直接上代码
代码
static NSInteger maskViewTag = 100111;
@implementation UIView (WBBLDetailSkeletonLoading)
- (void)addBLDetailSkeletonLoadMasking{
UIView *maskView = [[UIView alloc] initWithFrame:self.bounds];
[maskView setBackgroundColor:[UIColor wb_lightColorWithHex:0xffffff darkColorWithHex:0x000000]];
[maskView setTag:maskViewTag];
[self addSubview:maskView];
UIView *titleTopView = [[UIView alloc] initWithFrame:CGRectMake(15, 20, kScreenWidth - 30, 24)];
titleTopView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
titleTopView.layer.masksToBounds = YES;
titleTopView.layer.cornerRadius = 3;
[maskView addSubview:titleTopView];
UIView *titleBottomView = [[UIView alloc] initWithFrame:CGRectMake(15, CGRectGetMaxY(titleTopView.frame) + 7, 130, 24)];
titleBottomView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
titleBottomView.layer.masksToBounds = YES;
titleBottomView.layer.cornerRadius = 3;
[maskView addSubview:titleBottomView];
UIView *userHeaderView = [[UIView alloc] initWithFrame:CGRectMake(15, CGRectGetMaxY(titleBottomView.frame) + 20, 34, 34)];
userHeaderView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
userHeaderView.layer.masksToBounds = YES;
userHeaderView.layer.cornerRadius = 34 / 2.0;
[maskView addSubview:userHeaderView];
UIView *userNameView = [[UIView alloc] initWithFrame:CGRectMake(CGRectGetMaxX(userHeaderView.frame) + 8, CGRectGetMaxY(titleBottomView.frame) + 20, 70, 14)];
userNameView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
userNameView.layer.masksToBounds = YES;
userNameView.layer.cornerRadius = 3;
[maskView addSubview:userNameView];
UIView *userInfoView = [[UIView alloc] initWithFrame:CGRectMake(CGRectGetMaxX(userHeaderView.frame) + 8, CGRectGetMaxY(userNameView.frame) + 5.5, 94, 14)];
userInfoView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
userInfoView.layer.masksToBounds = YES;
userInfoView.layer.cornerRadius = 3;
[maskView addSubview:userInfoView];
UIView *rightButtonView = [[UIView alloc] initWithFrame:CGRectMake(kScreenWidth - 70 - 15, CGRectGetMaxY(titleBottomView.frame) + 25.5, 70, 24)];
rightButtonView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
rightButtonView.layer.masksToBounds = YES;
rightButtonView.layer.cornerRadius = 3;
[maskView addSubview:rightButtonView];
NSInteger lineMsgNum = 5;
CGFloat lineMsgViewWidth = kScreenWidth - 30;
for (NSInteger i=0; i<=lineMsgNum; i++) {
if (i==lineMsgNum) {
lineMsgViewWidth = 145;
}
UIView *lineMsgView = [[UIView alloc] initWithFrame:CGRectMake(15, CGRectGetMaxY(userInfoView.frame) + 26 + i*28, lineMsgViewWidth, 17)];
lineMsgView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
lineMsgView.layer.masksToBounds = YES;
lineMsgView.layer.cornerRadius = 3;
[maskView addSubview:lineMsgView];
}
CGFloat bottomLineMsgViewY = CGRectGetMaxY(userInfoView.frame) + 26 + lineMsgNum*28 + 17;
UIView *contentView = [[UIView alloc] initWithFrame:CGRectMake(15, bottomLineMsgViewY + 25.5, kScreenWidth - 30, 229)];
contentView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
contentView.layer.masksToBounds = YES;
contentView.layer.cornerRadius = 3;
[maskView addSubview:contentView];
UIView *lastView = [[UIView alloc] initWithFrame:CGRectMake(15, CGRectGetMaxY(contentView.frame) + 11, 167, 24)];
lastView.backgroundColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
lastView.layer.masksToBounds = YES;
lastView.layer.cornerRadius = 3;
[maskView addSubview:lastView];
[self addLoadingAnimationWithSuperView:maskView];
}
- (void)addLoadingAnimationWithSuperView:(UIView *)superView{
for (UIView *view in superView.subviews) {
[self startAnimationWithView:view];
}
}
- (void)dismissBLDetailSkeletonLoadMasking{
if ([self viewWithTag:100111]) {
[[self viewWithTag:100111] removeFromSuperview];//移除view
}
}
- (void)startAnimationWithView:(UIView *)view{
CAGradientLayer *bgViewLayer = [[CAGradientLayer alloc] init];
bgViewLayer.backgroundColor = [UIColor clearColor].CGColor;
bgViewLayer.frame = CGRectMake(-view.bounds.size.width, 0, view.bounds.size.width * 3, view.bounds.size.height);
bgViewLayer.startPoint = CGPointMake(0, 0.5);
bgViewLayer.endPoint = CGPointMake(1, 0.5);
UIColor *startColor = [UIColor wb_lightColorWithHex:0xf6f8f9 darkColorWithHex:0x171717];
UIColor *endColor = [UIColor wb_lightColorWithHex:0xe9ebee darkColorWithHex:0x121212];;
bgViewLayer.colors = [NSArray arrayWithObjects:(__bridge id)startColor.CGColor,(__bridge id)endColor.CGColor,(__bridge id)startColor.CGColor, nil];
bgViewLayer.locations = @[@0.25,@0.5,@0.75];
CABasicAnimation* rotationAnimation;
rotationAnimation = [CABasicAnimation animationWithKeyPath:@"locations"];
rotationAnimation.fromValue = @[@0.0, @0.0, @.25];
rotationAnimation.toValue = @[@0.75, @1.0, @1.0];
rotationAnimation.duration = 3;
rotationAnimation.repeatCount = 10000;
rotationAnimation.removedOnCompletion = NO;
[bgViewLayer addAnimation:rotationAnimation forKey:nil];
[view.layer addSublayer:bgViewLayer];
}
@end
优势
- 轻量级,灵活可控,可以快速升级迭代
- 不用引入三方文件,减小包大小
特别鸣谢
iOS - 骨架屏自动生成方案
iOS-用户体验之骨架屏的实践
CAGradientLayer
CAGradientLayer苹果官网