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. Andriod学习笔记1:代码优化总结1

    多行变一行 比如说开发一个简单的计算器应用程序,需要定义0-9的数字按钮,第一次就习惯性地写出了如下代码: Button btn0; Button btn1; Button btn2; Button ...

  2. 本机tomcat的server.xml被还原的问题及解决办法

    将tomcat的server.xml进行修改,但当eclipse发布站点后,发布tomcat中的server.xml会被还原. 原因是eclipse会将自己的tomcat配置文件对tomcat覆盖,解 ...

  3. RStudio技巧01_美化RStudio的帮助页面

    R中的package及其函数实在太多,经常遇到不会使用或者忘记如何使用的的package和函数,所以总会查阅帮助文档,在Rstudio中提供了专门的help面板,当遇到不懂的package或者函数时只 ...

  4. Socket聊天室-TcpListener,TcpClient

    参考自:http://blog.csdn.net/liguo9860/article/details/6148614 服务端:

  5. Bean不同配置方式的比较

    在<Spring3.x 企业应用开发实战>上学习了Bean的三种不同配置方法,下图是我从书中截取的图片,比较了一下这三种配置的异同 ps:发现图片不能完全显示(右侧有一块不显示),解决方法 ...

  6. Offline.js - 自动判断网络连接状态并提醒用户

    http://www.cnblogs.com/lhb25/p/offline-js-alert-users-when-no-internet-connectivity.html 使用 jslint/j ...

  7. Mysql5.7修改默认密码

    由于 Mysql5.7的默认密码是随机生成的,所以需要修改成我们自己常用的密码 1.修改 my.ini,在 [mysqld] 小节下添加一行:skip-grant-tables=1 这一行配置让 my ...

  8. Python3中使用PyMySQL连接Mysql

    Python3中使用PyMySQL连接Mysql 在Python2中连接Mysql数据库用的是MySQLdb,在Python3中连接Mysql数据库用的是PyMySQL,因为MySQLdb不支持Pyt ...

  9. jquery数组(排序)

    sort(); html: <h3>字符串数组排序前</h3> <div id="show1"></div> <h3>排 ...

  10. AI PRO I 第4章

    Behavior Selection Algorithms An Overview Michael Dawe, Steve Gargolinski, Luke Dicken, Troy Humphre ...