DWIntrosPage 简单定制引导页

源码在底下

前言:

该框架代码非常简单,仅需几行代码即可用于项目。 1、目前支持动态图(GIF)、静态图,未来还会支持视频、缓存、横向等。 2、页面切换时带渐变效果,体验感更强 3、支持最后一页右移消失和前面页面直接跳过 4、支持额外自定义,显示页面底下dots以及skip按钮与否。 5、更低的内存占用、CPU占用 6、支持直接使用图片名字,框架自动识别图片类型 7、支持cocopods安装以及手动拖进项目

1、使用方式:

- (UIViewController *)setupDynamicVC {
    
    DWIntrosPageContentViewController *page1 = [DWIntrosPageContentViewController introsPageWithBackgroundImageWithName:@"gif01"];
    DWIntrosPageContentViewController *page2 = [DWIntrosPageContentViewController introsPageWithBackgroundImageWithName:@"gif02"];
    DWIntrosPageContentViewController *page3 = [DWIntrosPageContentViewController introsPageWithBackgroundImageWithName:@"gif03"];
    DWIntrosPagesViewController *introsPage = [DWIntrosPagesViewController dwIntrosPagesWithPageArray:@[page1, page2, page3]];
    __weak typeof(self) weakSelf = self;
    introsPage.skipButtonClickedBlock = ^{
        NSLog(@"clicked skip button");
        [weakSelf setupHomeVC];
    };
    return introsPage;
}

上面为使用静态图方式,动态图方式与此大同小异,故不多说。

2、框架目录

该框架主要有两大类:

  • DWIntrosPageContentViewController引导页的每一个详情页面
  • DWIntrosPagesViewController相当于引导页的容器,承载引导页的每一个详情页面。

下面摘取部分代码 DWIntrosPageContentViewController

+ (instancetype)introsPageWithBackgroundImageWithName:(NSString *)imgName {
    if ([imgName.pathExtension isEqualToString:@"jpg"] || [imgName.pathExtension isEqualToString:@"png"]) {
        return [[self alloc] initWithBackgroundImageWithName:imgName];
    }
    NSData *data = [self gainFullImageWithName:imgName];
    if ([_imagePathExtension((__bridge CFDataRef)(data)) isEqualToString:@"GIF"]) {
        return [[self alloc] initWithBackgroundGIFWithData:data];
    }else if([_imagePathExtension((__bridge CFDataRef)(data)) isEqualToString:@"JPG"]) {
        return [[self alloc] initWithBackgroundNormalImageWithData:data];
    }
    return [[self alloc] initWithBackgroundImageWithName:imgName];
}

设置每一页的详情,不同类型不同设置方式

+ (NSData *)gainFullImageWithName:(NSString *)name {
    if (name.length == 0) return nil;
    
    NSString *res = name.stringByDeletingPathExtension;
    NSString *ext = name.pathExtension;
    NSString *path = nil;
    NSArray *exts = ext.length > 0 ? @[ext] : @[@"", @"png", @"jpeg", @"jpg", @"gif", @"webp", @"apng"];
    for (NSString *e in exts) {
        path = [[NSBundle mainBundle] pathForResource:res ofType:e];
        if (path) break;
    }
    NSData *data = [NSData dataWithContentsOfFile:path];
    if (data.length == 0) return nil;
    return data;

}

判断图片是否有后缀,有后缀直接使用,没有后缀则尝试加上后缀名去[NSBundle mainBundle]寻找

DWIntrosPagesViewController

#pragma mark ------ UIPageViewControllerDelegate && DataSource ------
- (UIViewController *)pageViewController:(UIPageViewController *)pageViewController viewControllerBeforeViewController:(UIViewController *)viewController {
   
    if (viewController == [self.viewControllers firstObject]) {
        return nil; // Return 'nil' to indicate that no more progress can be made in the given direction. (表示已经不能继续往左)
    }else {
        NSInteger lastPageIndex = [self.viewControllers indexOfObject:viewController] - 1;
        NSLog(@"%ld", (long)lastPageIndex);
        return [self.viewControllers objectAtIndex:lastPageIndex];
    }
}

- (UIViewController *)pageViewController:(UIPageViewController *)pageViewController viewControllerAfterViewController:(UIViewController *)viewController {
    if (viewController == [self.viewControllers lastObject]) {
        return nil; // (表示已经不能继续往右)
    }else {
        NSInteger nextPageIndex = [self.viewControllers indexOfObject:viewController] + 1;
//        NSLog(@"%ld", (long)nextPageIndex);
        return self.viewControllers[nextPageIndex];
    }
}

判断控制器数组中的控制器,如果是首页或者尾页,则返回nil

- (void)pageViewController:(UIPageViewController *)pageViewController didFinishAnimating:(BOOL)finished previousViewControllers:(NSArray<UIViewController *> *)previousViewControllers transitionCompleted:(BOOL)completed {
    // We need to confirm the transition completed and to do other things.
    // 确保转换完成才改变底下的小圆点
    if (completed) {
        _currentPage = [pageViewController.viewControllers lastObject];
        NSInteger currentIndex = [self.viewControllers indexOfObject:_currentPage];
        [self.pageControl setCurrentPage:currentIndex];
    }
}

等页面切换完成后才切换底下的dots

#pragma mark ------ changeAlpha ------
- (void)changeAlphaWithRatio:(CGFloat)ratio scrollView:(UIScrollView *)scrollView {

    if (ratio == 0) {
        return;
    }
   
    // figure out alpha
    // next page alpha equals to the ratio
    CGFloat nextPageAlpha = ratio;
    CGFloat currentPageAlpha = 1 - ratio;
    
    // warning: don't mix up the order of the two sentences
    //注意: 不要弄乱下面两句顺序
    [self.nextPage updateAlpha:nextPageAlpha];
    [self.currentPage updateAlpha:currentPageAlpha];
    
    // change the alpha of skip button and pageControl dots
    if (_nextPage == [self.viewControllers lastObject] ) {
        self.skipButton.alpha = currentPageAlpha;
        self.pageControl.alpha = currentPageAlpha;
    }
    if (_currentPage == [self.viewControllers lastObject]) {
        self.skipButton.alpha = nextPageAlpha;
        self.pageControl.alpha = nextPageAlpha;
        // The last page will not show the skill button and page dots.
        // 最后一页往右也不能出现下面两个按钮
        if ((scrollView.contentOffset.x > self.historyX)) {
            self.skipButton.alpha = 0;
            self.pageControl.alpha = 0;
            if (scrollView.contentOffset.x - self.historyX > 45) {
                [self skipButtonCliked];
            }
        }
    }

}

根据移动的偏移值,对切换中的前后两个页面进行透明度的改变,从而达到渐变效果。

最后

整体而言,代码非常简单,请各位大佬轻拍。

代码地址

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏陈满iOS

[iOS源码笔记]·第三方网络图片处理框架:SDWebImage网络下载及缓存管理策略

typedef void(^SDExternalCompletionBlock)(UIImage * _Nullable image, NSError * _N...

15510
来自专栏一“技”之长

iOS获取通讯录联系人信息 原

随着apple对用户隐私的越来越重视,IOS系统的权限设置也更加严格,在获取系统通讯录之前,我们必须获得用户的授权。权限申请代码示例如下:

25230
来自专栏慎独

UIViewController生命周期分析

18040
来自专栏哈雷彗星撞地球

iOS 中如何判断当前是2G/3G/4G/5G/WiFi

5G 什么的,还得等苹果API更新啊,不过将来还是这个处理过程就是了。 关于判断当前的网络环境是2G/3G/4G,这个问题以前经常看到,最近在一工程里看到了如...

25820
来自专栏雨尘分享

ReactiveCocoa 入门知识——归总篇

23740
来自专栏iOS开发笔记

iOS开发-搜索栏UISearchBar和UISearchController

最近项目中用到了搜索栏,所以在网上搜了一些相关的资料学习了一下,现在记录一下,iOS中的搜索栏实现起来相对简单一点,网上也有很多参考资料,不过靠谱的不是很多...

63870
来自专栏進无尽的文章

多媒体 -获取本地图片和视频文件

很多自定义播放器,和自定义多选相册的图片都是需要先获取系统图片库中的所有图片资源或者视屏资源 ,所使用的核心方法就是AssetsLibrary框架里的ALAss...

10220
来自专栏Alice

(whh仅供自己参考)进行ip网络请求的步骤

这个过程大致是这个样子: 1 添加通知 2 发送网络请求 里边有一个发送通知的操作 3 执行发送通知的具体操作 代码如下: 1 在VC添加通知 [[NSNo...

19860
来自专栏美团技术团队

ReactiveCocoa中潜在的内存泄漏及解决方案

ReactiveCocoa是GitHub开源的一个函数响应式编程框架,目前在美团App中大量使用。用过它的人都知道很好用,也确实为我们的生活带来了很多便利,特别...

54380
来自专栏向治洪

RCTEventEmitter使用

在0.27版本之前,RN的Native端向js端发射消息主要通过sendDeviceEventWithName的方式,相关代码如下。 @synthesize b...

48670

扫码关注云+社区

领取腾讯云代金券