1.下拉刷新控件

2.下拉加载更多控件

下拉刷新控件

@property(nonatomic,strong) VRefreshHeadView *vrefresh;

[self vrefresh];

-(VRefreshHeadView *)vrefresh{

__weak typeof(self) weakSelf = self;

if (!_vrefresh) {

_vrefresh=[[VRefreshHeadView alloc] initWithScrollView:weakSelf.tbView beginRefreshBlock:^(VRefreshHeadView *vrefresh) {

NSLog(@"进入刷新回调");

[self performSelector:@selector(endVrefresh) withObject:nil afterDelay:3.0f];

}];

}

return _vrefresh;

}

-(void)endVrefresh{

[_vrefresh endRefresh];

}

//

//  VRefreshHeadView.h

//  下拉刷新控件

//

//  Created by Vie on 2016/10/10.

//  Copyright © 2016年 Vie. All rights reserved.

//

#import <UIKit/UIKit.h>

typedef NS_ENUM(NSUInteger, VRefreshType) {

VRefreshTypeDefault = 0,//默认下拉刷新样式

};

@interface VRefreshHeadView : UIView

typedef void (^beginRefreshBlock) (VRefreshHeadView *vrefresh);//刷新事件回调

@property(nonatomic,assign) NSUInteger type;//刷新视图样式

/**

创建下拉刷新视图

@param scrollView        滚动视图

@param beginRefreshBlock 开始刷新回调

@return 刷新视图对象

*/

-(instancetype)initWithScrollView:(UIScrollView *)scrollView beginRefreshBlock:(beginRefreshBlock)beginRefreshBlock;

//停止刷新

-(void)endRefresh;

@end

//

//  VRefreshHeadView.m

//  下拉刷新控件

//

//  Created by Vie on 2016/10/10.

//  Copyright © 2016年 Vie. All rights reserved.

//

#import "VRefreshHeadView.h"

#define vRefreshHeadViewHeight 40

#define navBarHeight  64  //竖屏导航栏和控制器高度(竖屏导航栏高度44,竖屏状态栏高度为20、打电话或者某些情况为40)

@interface VRefreshHeadView ()

@property (nonatomic,copy) beginRefreshBlock beginRefreshBlock;

@property(nonatomic,weak) UIScrollView *scrollView;

/**

*头部提示语

*/

@property(nonatomic,weak) UILabel *headTipLable;

/**

*  加载提示

*/

@property (nonatomic, weak) UIActivityIndicatorView *indicatorView;

/**

*是否处于刷新状态

*/

@property(nonatomic,assign) BOOL isRefresh;

@end

@implementation VRefreshHeadView

#pragma mark  - init方法

-(instancetype)initWithScrollView:(UIScrollView *)scrollView beginRefreshBlock:(beginRefreshBlock)beginRefreshBlock{

self=[super initWithFrame:CGRectMake(0, -vRefreshHeadViewHeight, scrollView.frame.size.width, vRefreshHeadViewHeight)];

if (self) {

_scrollView=scrollView;

_beginRefreshBlock=beginRefreshBlock;

//把当前视图加载到scrollView的UI

[_scrollView addSubview:self];

//设置背景颜色

self.backgroundColor=[UIColor clearColor];

//默认不在刷新状态

self.isRefresh=false;

//注册观察者,监听下拉刷新改变

[_scrollView addObserver:self forKeyPath:@"contentOffset" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];

}

return self;

}

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{

if ([keyPath isEqualToString:@"contentOffset"]) {

if (_scrollView.contentOffset.y<(-navBarHeight-vRefreshHeadViewHeight)&&!self.isRefresh&&_scrollView.dragging) {

[self.indicatorView startAnimating];

self.isRefresh=true;

self.headTipLable.text=@"加载中..";

_scrollView.contentInset = UIEdgeInsetsMake(vRefreshHeadViewHeight+navBarHeight, 0, 0, 0);

if (_beginRefreshBlock) {

_beginRefreshBlock(self);

}

}

}

}

//停止刷新

-(void)endRefresh{

[UIView  animateWithDuration:0.3 animations:^{

//        _scrollView.contentInset = UIEdgeInsetsZero;

_scrollView.contentInset = UIEdgeInsetsMake(navBarHeight, 0, 0, 0);

} completion:^(BOOL finished) {

[self.indicatorView stopAnimating];

self.isRefresh=false;

_headTipLable.text=@"下拉刷新数据";

}];

}

/*

// Only override drawRect: if you perform custom drawing.

// An empty implementation adversely affects performance during animation.

*/

- (void)drawRect:(CGRect)rect {

// Drawing code

if (_type) {

}else{

//没有指定type用默认视图

[self setDefaultView];

}

}

//默认样式

-(void)setDefaultView{

[self headTipLable];

}

#pragma mark - 懒加载创建控件

- (UIActivityIndicatorView *)indicatorView

{

if (!_indicatorView)

{

UIActivityIndicatorView *act = [[UIActivityIndicatorView alloc] initWithFrame:CGRectMake(50, 0, 30, self.frame.size.height)];

act.color = [UIColor grayColor];

[self addSubview:act];

_indicatorView = act;

}

return _indicatorView;

}

-(UILabel *)headTipLable{

if (!_headTipLable) {

UILabel *lb = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, self.frame.size.width,self.frame.size.height)];

lb.textAlignment = NSTextAlignmentCenter;

lb.textColor = [UIColor grayColor];

lb.font = [UIFont systemFontOfSize:13];

lb.text = @"下拉刷新数据";

[self addSubview:lb];

_headTipLable = lb;

}

return _headTipLable;

}

-(void)dealloc{

[_scrollView removeObserver:self forKeyPath:@"contentOffset"];

}

@end

下拉加载更多控件

@property(nonatomic,strong) VLoadMoreFootView *vload;

[self vload];

-(VLoadMoreFootView *)vload{

__weak typeof(self) weakSelf=self;

if (!_vload) {

_vload=[[VLoadMoreFootView alloc] initWithScrollView:weakSelf.tbView beginLoadMoreBlock:^(VLoadMoreFootView *vrefresh) {

NSLog(@"进入上拉加载");

[self performSelector:@selector(endVload) withObject:nil afterDelay:3.0f];

}];

}

return _vload;

}

-(void)endVload{

[_vload endLoadMore];

}

//

//  VLoadMoreFootView.h

//  上拉加载更多控件

//

//  Created by Vie on 2016/10/17.

//  Copyright © 2016年 Vie. All rights reserved.

//

#import <UIKit/UIKit.h>

typedef NS_ENUM(NSUInteger, VLoadMoreType) {

VLoadMoreTypeDefault = 0,//默认上拉加载样式

};

@interface VLoadMoreFootView : UIView

typedef void (^beginLoadMoreBlock) (VLoadMoreFootView *vrefresh);//加载更多事件回调

@property(nonatomic,assign) NSUInteger type;//加载更多视图样式

/**

创建上拉加载新视图

@param scrollView         滚动视图

@param beginLoadMoreBlock 开始加载更多回调

@return 加载视图对象

*/

-(instancetype)initWithScrollView:(UIScrollView *)scrollView beginLoadMoreBlock:(beginLoadMoreBlock)beginLoadMoreBlock;

//停止加载

-(void)endLoadMore;

@end

//

//  VLoadMoreFootView.m

//  上拉加载更多控件

//

//  Created by Vie on 2016/10/17.

//  Copyright © 2016年 Vie. All rights reserved.

//

#import "VLoadMoreFootView.h"

#define  vLoadMoreBottomViewHeight  40

#define navBarHeight  64  //竖屏导航栏和控制器高度(竖屏导航栏高度44,竖屏状态栏高度为20、打电话或者某些情况为40)

@interface VLoadMoreFootView()

@property(nonatomic,copy) beginLoadMoreBlock beginLoadMoreBlock;

@property(nonatomic,weak) UIScrollView *scrollView;

/**

*底部提示语

*/

@property(nonatomic,weak) UILabel *bottomTipLable;

/**

*加载提示

*/

@property(nonatomic,weak) UIActivityIndicatorView *indicatorView;

/**

*是否处于加载状态

*/

@property(nonatomic,assign) BOOL isLoad;

@end

@implementation VLoadMoreFootView

#pragma mark - init方法

-(instancetype)initWithScrollView:(UIScrollView *)scrollView beginLoadMoreBlock:(beginLoadMoreBlock)beginLoadMoreBlock{

self=[super initWithFrame:CGRectMake(0, 0, scrollView.frame.size.width, vLoadMoreBottomViewHeight)];

if (self) {

_scrollView=scrollView;

_beginLoadMoreBlock=beginLoadMoreBlock;

//把当前视图加载到scrollView的UI

[_scrollView addSubview:self];

//设置背景颜色

self.backgroundColor=[UIColor clearColor];

//默认不在加载状态

self.isLoad=false;

//注册观察这,监听上拉加载改变

[_scrollView addObserver:self forKeyPath:@"contentOffset" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];

}

return self;

}

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{

if ([keyPath isEqualToString:@"contentOffset"]) {

if (_scrollView.contentOffset.y>(_scrollView.contentSize.height-_scrollView.frame.size.height)&&!self.isLoad&&_scrollView.dragging) {

[self.indicatorView startAnimating];

self.isLoad=true;

self.bottomTipLable.text=@"加载中..";

_scrollView.contentInset = UIEdgeInsetsMake(0, 0, vLoadMoreBottomViewHeight, 0);

if (_beginLoadMoreBlock) {

_beginLoadMoreBlock(self);

}

}

}

}

//停止加载

-(void)endLoadMore{

[UIView animateWithDuration:0.3 animations:^{

//        _scrollView.contentInset = UIEdgeInsetsZero;

_scrollView.contentInset = UIEdgeInsetsMake(navBarHeight, 0, 0, 0);

} completion:^(BOOL finished) {

[self.indicatorView stopAnimating];

self.isLoad=false;

_bottomTipLable.text = @"上拉加载更多数据";

}];

}

/*

// Only override drawRect: if you perform custom drawing.

// An empty implementation adversely affects performance during animation.

*/

- (void)drawRect:(CGRect)rect {

// Drawing code

if (_type) {

}else{

//没有指定type用默认视图

[self setDefaultView];

}

}

//默认样式

-(void)setDefaultView{

[self bottomTipLable];

}

-(UILabel *)bottomTipLable{

if (!_bottomTipLable) {

UILabel *lb = [[UILabel alloc] initWithFrame:CGRectMake(0, _scrollView.contentSize.height, self.frame.size.width,self.frame.size.height)];

lb.textAlignment = NSTextAlignmentCenter;

lb.textColor = [UIColor grayColor];

lb.font = [UIFont systemFontOfSize:13];

lb.text = @"上拉加载更多数据";

[self addSubview:lb];

_bottomTipLable = lb;

}

return _bottomTipLable;

}

#pragma mark - 懒加载创建控件

- (UIActivityIndicatorView *)indicatorView

{

if (!_indicatorView)

{

UIActivityIndicatorView *act = [[UIActivityIndicatorView alloc] initWithFrame:CGRectMake(50, _scrollView.contentSize.height, 30, self.frame.size.height)];

act.color = [UIColor grayColor];

[self addSubview:act];

_indicatorView = act;

}

return _indicatorView;

}

-(void)dealloc{

[_scrollView removeObserver:self forKeyPath:@"contentOffset"];

}

@end

iOS,自定义控件的更多相关文章

  1. iOS 自定义控件开发(中)

    <iOS 自定义控件开发(上)> <iOS 自定义控件开发(中)> 接上篇iOS自定义控件开发之后,我们尝试另外一种. 在Xcode的右边,会看到如下的图 其中,上面有一个:C ...

  2. iOS 自定义控件开发(上)

    工作需要,最近在进行iOS方面的图表工作.找了很多第三方库都无法实现效果,所以决定自己写一个控件. <iOS 自定义控件开发(上)> <iOS 自定义控件开发(中)> #0 目 ...

  3. iOS自定义控件教程:制作一个可重用的旋钮

    当你的APP需要一些新功能时,自定义UI控件会十分有用,尤其是这些自定义控件可以在其他APP里面很好的重用.Colin Eberhart写过一篇很棒的介绍自定义UI控件的教程.这个教程涉及的是一个继承 ...

  4. 分享一下我封装iOS自定义控件的体会,附上三个好用的控件Demo <时间选择器&多行输入框&日期选择器>

    前段时间有小伙伴问到我:"这样的控件该怎么做呢?",我感觉是个比较简单的控件,可能对于入行不久的同志思路没有很清晰吧.趁着最近工作不忙,就来这里分享一下我封装自定义控件的几点体会吧 ...

  5. 关于iOS自定义控件:在view上实现事件和代理

    自定义控件.h #import <UIKit/UIKit.h> #import "PPViewtouchesBeginDelegate.h" @interface PP ...

  6. iOS自定义控件创建原理(持续更新)

    前言 因为如果要创建各种自定义控件根据需求的不同会有很多的差别,所以我就在这里,分析一些自定义控件的创建实现方法 弹出视图 1.把要弹出的视图装在一个控制器里面,自定义转场动画 2.创建一个弹出视图, ...

  7. iOS自定义控件开发详解

    http://blog.csdn.net/zhangao0086/article/details/45622875

  8. IOS知识小记

    iOS开发 小知识点 http://www.cnblogs.com/tangbinblog/archive/2012/07/20/2601324.html Objective-C中的instancet ...

  9. Android开发之自定义组件和接口回调

    说到自定义控件不得不提的就是接口回调,在Android开发中接口回调用的还是蛮多的.在这篇博客开始的时候呢,我想聊一下iOS的自定义控件.在iOS中自定义控件的思路是继承自UIView, 在UIVie ...

随机推荐

  1. 借鉴dubbo实现自定义缓存

    自定义缓存一般基于ConcurrentMap实现,实现缓存需要注意的点是缓存容器对象 本身依赖于 static final去存储对象,样例: ConcurrentMap<String, Gene ...

  2. Flask 教程

    官方文档 推荐教程 环境 pip install virtualenv cd proj_fold virtualenv venv . venv/bin/activate for *unix or ve ...

  3. Linux下定时执行任务的几种方式

    如果说我说如果,你的某一个目录下会经常的生成一些垃圾文件,比如访问日志.错误日志.core文件,而你又不想过几分钟就去手动检查一下,那么可以使用定时执行任务的方式来解决.目前我所知道的可以执行定时任务 ...

  4. OpenGL光照和颜色

    OpenGL光照和颜色 转自:http://www.cnblogs.com/kekec/archive/2011/08/16/2140789.html OpenGL场景中模型颜色的产生,大致为如下的流 ...

  5. *cf.4 贪心

    D. Kostya the Sculptor time limit per test 3 seconds memory limit per test 256 megabytes input stand ...

  6. bzoj1057: [ZJOI2007]棋盘制作--最大子矩阵

    既然要求最大01子矩阵,那么把应该为0的位置上的数取反,这样就变成求最大子矩阵 最大子矩阵可以用单调栈 #include<stdio.h> #include<string.h> ...

  7. 谢欣伦 - OpenDev原创教程 - 通信开发库libComm

    libComm是一个免费的简单的通信库,其中的接口类与函数大都以小写的x打头,来源于我的姓氏首字母(谢欣伦). 下载 Sample using libComm - v1.1 For WinXP lib ...

  8. UI控件闪烁及刷新

    我们常常在一个窗口上放置很多控件,在改变窗口大小时,控件会跟着一起闪烁... 此时,可以将窗口添加WS_CLIPCHILDREN属性即可.(如果包含多层,都需要WS_CLIPCHILDREN属性) 默 ...

  9. 实战Java虚拟机之三“G1的新生代GC”

    今天开始实战Java虚拟机之三:“G1的新生代GC”. 总计有5个系列 实战Java虚拟机之一“堆溢出处理” 实战Java虚拟机之二“虚拟机的工作模式” 实战Java虚拟机之三“G1的新生代GC” 实 ...

  10. Jackson

    Jackson可以轻松的将Java对象转换成json对象和xml文档,同样也可以将json.xml转换成Java对象. 1. 下载依赖库jar包 Jackson的jar all下载地址:http:// ...