关于UIPageControl 自定义样式定制的正确姿势

本文介绍了一种定制UIPageControl样式的实现方法,包括修改颜色、图标样式等,并提供了一个安全替换系统默认图标的解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

近来,项目中需要定制UIPageControl样式,目前知道的方式大概如下,这里简单概括:

一、修改颜色,最简单的方式:

_pageControl.pageIndicatorTintColor = [UIColor redColor];// 设置非选中页的圆点颜色
_pageControl.currentPageIndicatorTintColor = [UIColor blueColor]; // 设置选中页的圆点颜色

二、修改静态图标样式,可以采用KVO方式,这种弊端是只能静态显示,无法动态替换,且私有,危险:

[_pageControl setValue:[UIImage imageNamed:@"image1"] forKeyPath:@"_pageImage"];

[_pageControl setValue:[UIImage imageNamed:@"image2"] forKeyPath:@"_currentPageImage"];

三、替换系统默认图标,自定义样式:

常规方式便是创建一个类,继承于UIPageControl,重新set方式,这里写下核心代码,也是目前网络上的方式:

-(void) updateDots {            
     for (int i = 0; i < [self.subviews count]; i++)  {
       UIImageView *dot = [self.subviews objectAtIndex:i];
        if (i == self.currentPage) dot.image = activeImage;
       else dot.image = inactiveImage;
    }
 }
结果是程序崩溃,归咎原因:
这种方式及其危险的,因为[self.subviews objectAtIndex:i] class 为UIView, 而 UIImageView *dot = [self.subviews objectAtIndex:i]; 这句是将一个UIView转换为UIImageView, 故容易crash 掉,特此放弃如上国内方式;


四、完善上面第三种方式,转为安全的替换方式,核心代码如下:

for (UIView *subview in view.subviews) {
            if ([subview isKindOfClass:[UIImageView class]]) {
                dot = (UIImageView *)subview;
                break;
            }
        }
        

采用如上安全方式,可有效防止崩溃,至此完成项目需求,效果如下:


五、这里把整个代码放出来,帮助大家少走弯路,具体代码如下:

1、.h文件

#import <UIKit/UIKit.h>

@interface CustomPageControl : UIPageControl
@property (nonatomic, strong) UIImage *currentImage;
@property (nonatomic, strong) UIImage *inactiveImage;

@property (nonatomic, assign) CGSize currentImageSize;
@property (nonatomic, assign) CGSize inactiveImageSize;
@end

2、.m文件

#import "CustomPageControl.h"

@implementation CustomPageControl

- (instancetype)init{
    self = [super init];
    if (self) {
        self.userInteractionEnabled = NO;
    }
    return self;
}

- (void)setCurrentPage:(NSInteger)currentPage{
    [super setCurrentPage:currentPage];
 
    [self updateDots];
}


- (void)updateDots{
    for (int i = 0; i < [self.subviews count]; i++) {
        UIImageView *dot = [self imageViewForSubview:[self.subviews objectAtIndex:i] currPage:i];
        if (i == self.currentPage){
            dot.image = self.currentImage;
            dot.size = self.currentImageSize;
        }else{
            dot.image = self.inactiveImage;
            dot.size = self.inactiveImageSize;
        }
    }
}


- (UIImageView *)imageViewForSubview:(UIView *)view currPage:(int)currPage{
    UIImageView *dot = nil;
    if ([view isKindOfClass:[UIView class]]) {
        for (UIView *subview in view.subviews) {
            if ([subview isKindOfClass:[UIImageView class]]) {
                dot = (UIImageView *)subview;
                break;
            }
        }
        
        if (dot == nil) {
            dot = [[UIImageView alloc] initWithFrame:CGRectMake(0.0f, 0.0f, view.frame.size.width, view.frame.size.height)];
           
            [view addSubview:dot];
        }
    }else {
        dot = (UIImageView *)view;
    }
    
    return dot;
}



@end

3、使用方式:

CustomPageControl *pageCtrl = [[CustomPageControl alloc] initWithFrame:CGRectMake(0, 10, mScreenWidth, 5)]; 
pageCtrl.numberOfPages = cardSwitchViewAry.count;
pageCtrl.currentPage = 0; 
pageCtrl.userInteractionEnabled = NO;
pageCtrl.inactiveImage = [UIImage imageNamed:@"ex_page_1"];
pageCtrl.inactiveImageSize = CGSizeMake(5, 5);
pageCtrl.currentImage = [UIImage imageNamed:@"ex_page_2"];
pageCtrl.currentImageSize = CGSizeMake(10, 5);


    //去掉系统自带样式
pageCtrl.currentPageIndicatorTintColor = [UIColor clearColor];
pageCtrl.pageIndicatorTintColor = [UIColor clearColor];    
至此,定制UIPageControl的正确姿势已了解,希望对大家有用。


本Demo使用UICollectionView实现自动无限轮播功能。 主要功能: 1.实现自动轮播,可修改轮播的时间 2.轮播图片可以来自本地,也可来自网络,通过单独的方法进行设置即可。对于加载网络图片时,Demo中使用了YYWebImage,也可自行替换成SDWebImage。 3.重写了和系统UIPageControl一样的功能,可用图片代替PageControl上的点点,也可自定义其颜色以及切换动画。 使用方法:使用方法比较简单。 /** * 加载本地图片Banner */ - (void)setupLocalBannerImageView { NSArray *array = @[@"1.png", @"2.png", @"3.png", @"4.png", @"5.png"]; FFBannerView *bannerVew = [FFBannerView bannerViewWithFrame:CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, 200) locationImageArray:array]; bannerVew.timeInterval = 2.0; [self.view addSubview:bannerVew]; } /** * 加载网络图片Banner */ - (void)setupNetWorkBannerImageView { NSArray *array = @[@"http://i3.download.fd.pchome.net/t_960x600/g1/M00/07/09/oYYBAFMv8q2IQHunACi90oB0OHIAABbUQAAXO4AKL3q706.jpg", @"http://images.weiphone.net/attachments/photo/Day_120308/118871_91f6133116504086ed1b82e0eb951.jpg", @"http://benyouhuifile.it168.com/forum/macos/attachments/month_1104/110425215921926a173e0f728e.jpg", @"http://benyouhuifile.it168.com/forum/macos/attachments/month_1104/1104241737046031b3a754f783.jpg"]; FFBannerView *bannerVew = [FFBannerView bannerViewWithFrame:CGRectMake(0, 250, [UIScreen mainScreen].bounds.size.width, 200) netWorkImageArray:array placeHolderImage:nil]; bannerVew.timeInterval = 2.0; bannerVew.pageControlStyle = FFPageControlStyleMiddle; bannerVew.delegate = self; [self.view addSubview:bannerVew]; } 以上方式即可简单使用,如需自定义PageControl也可继承FFAbstractDotView,做些基本的设置即可。 gitHub下载地址:喜欢的朋友请给个星呗! 欢迎各位一起来讨论,有问题请发邮箱270452746@qq.com或者直接加我QQ:270452746进行讨论。谢谢!
最新SWIFT 4.0版自定义PageControl,椭圆,空心圆,图片点 刚开始做swift项目,可用资源少而且每个swift版本变化太大,以前的都不能拿来直接用,现在我参考一个object-C的PageControl自己做了一个swift版的, 参考OC资源链接:https://github.com/hackxhj/EllipsePageControl, 非常感谢原作者。 本项目在原OC的功能基础上进行的改进,增加了自定义点的宽度,点的layer,不是当前点的图片等功能 基本能满足大部分的需求,写的很简单,大家一看就懂,欢迎大家使用 由于水平有限,项目中有改进之处忘各位大神给与指点,以求不断完善 本人联系方式:wei287030375@sina.com 如果觉得还可以的话给个star吧,也是对以后进行创作的一种鼓励,谢谢 效果图: image 以下是部分代码: class WEIPageControl: UIControl { var localNumberOfPages = NSInteger()//分页数量 var localCurrentPage = NSInteger()//当前点所在下标 var localPointSize = CGSize()//点的大小 var localPointSpace = CGFloat()//点之间的间距 var localOtherColor = UIColor()//未选中点的颜色 var localCurrentColor = UIColor()//当前点的颜色 var localOtherImage: UIImage?//未选中点的图片 var localCurrentImage: UIImage?//当前点的图片 var localIsSquare = Bool()//是否是方形点 var localCurrentWidthMultiple = CGFloat()//当前选中点宽度与未选中点的宽度的倍数 var localOtherBorderColor: UIColor?//未选中点的layerColor var localOtherBorderWidth: CGFloat?//未选中点的layer宽度 var localCurrentBorderColor: UIColor?//未选中点的layerColor var localCurrentBorderWidth: CGFloat?//未选中点的layer宽度 var clickIndex: ((_ result: NSInteger?) -> ())? 在ViewController中使用 //方形点举例 class ViewController: UIViewController, UIScrollViewDelegate { pageC pageControl3.frame = CGRect.init(x: left, y: scrollView3.frame.maxY, width: width, height: 20) pageControl3.numberOfPages = pageCount//总页数 pageControl3.isSquare = true//设置为方型点 pageControl3.currentWidthMultiple = 2.5//当前点的宽度为其他点的2.5倍 pageControl3.currentColor = UIColor.red pageControl3.otherColor = UIColor.blue pageControl3.pointSize = CGSize.init(width: 14, height: 6)//方点的size pageControl3.clickPoint { (index) in//方点的点击事件 self.scrollView3.setContentOffset(CGPoint.init(x: width * CGFloat(index!), y: 0), animated: true) } self.view.addSubview(pageControl3) 代码用起来就是这么简单,欢迎大家使用, 本项目github地址:https://github.com/wei287030375/WEIPageControl
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

群鸿

感谢认可,多谢打赏。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值