IOS UI-自定义UIColectionView布局
ViewController.m
//
// ViewController.m
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/26.
// Copyright © 2016年 博文科技. All rights reserved.
// #import "ViewController.h"
#import "ImgCell.h"
#import "LineLayout.h"
#import "FoldLayout.h"
#import "CircleLayout.h" @interface ViewController ()<UICollectionViewDataSource,UICollectionViewDelegate> @property (nonatomic, strong) NSMutableArray *imgsArray;
@property (nonatomic, weak) UICollectionView *collectionView; @end @implementation ViewController static NSString *ID = @"image"; - (NSMutableArray *)imgsArray
{
if (!_imgsArray) {
_imgsArray = [NSMutableArray array]; for (int i=; i<; i++) {
[_imgsArray addObject:[NSString stringWithFormat:@"%d.jpg",i]]; }
}
return _imgsArray;
} - (void)viewDidLoad {
[super viewDidLoad];
[self createUI];
} - (void)createUI
{
CGRect rect = CGRectMake(, , , );
UICollectionView *collection = [[UICollectionView alloc] initWithFrame:rect collectionViewLayout:[[FoldLayout alloc] init]];
collection.dataSource = self;
collection.delegate = self;
// [collection registerClass:[UICollectionViewCell class] forCellWithReuseIdentifier:ID];
[collection registerNib:[UINib nibWithNibName:@"ImgCell" bundle:nil] forCellWithReuseIdentifier:ID];
[self.view addSubview:collection];
self.collectionView = collection; } - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
if ([self.collectionView.collectionViewLayout isKindOfClass:[FoldLayout class]]) {
[self.collectionView setCollectionViewLayout:[[CircleLayout alloc] init] animated:YES];
}
else{
[self.collectionView setCollectionViewLayout:[[FoldLayout alloc] init] animated:YES];
}
} #pragma mark - UICollectionViewDataSource - (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section
{
return self.imgsArray.count;
} - (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
ImgCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:ID forIndexPath:indexPath]; cell.image = self.imgsArray[indexPath.item]; return cell;
} #pragma mark - UICollectionViewDelegate - (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath
{
//删除模型数据
[self.imgsArray removeObjectAtIndex:indexPath.item];
//刷新UI
[collectionView deleteItemsAtIndexPaths:@[indexPath]];
} @end
LineLayout.m(线性流水布局)
//
// LineLayout.m
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/26.
// Copyright © 2016年 博文科技. All rights reserved.
// #import "LineLayout.h" static const CGFloat lineLayoutSize = ; @implementation LineLayout //初始化
- (void)prepareLayout
{
[super prepareLayout];
//设置item大小
self.itemSize = CGSizeMake(lineLayoutSize, lineLayoutSize);
//设置两端居中
CGFloat inset = (self.collectionView.frame.size.width - lineLayoutSize) * 0.5;
self.sectionInset = UIEdgeInsetsMake(, inset, , inset); //设置水平滚动
self.scrollDirection = UICollectionViewScrollDirectionHorizontal;
//设置间距
self.minimumLineSpacing = lineLayoutSize; //每一个item都有自己的UICollectionViewLayoutAttributes
//每一indexPath都有自己的UICollectionViewLayoutAttributes
} //只要显示的边界发生变化就重新布局:内部会重新调用prepareLayout,layoutAttributesForElementsInRect方法获得所有item的布局属性
- (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds
{
return YES;
} - (NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect
{
//NSLog(@"layoutAttributesForElementsInRect");
// 0.计算可见的矩形框
CGRect visibleRect;
visibleRect.size = self.collectionView.frame.size;
visibleRect.origin = self.collectionView.contentOffset;
// 1.获得默认的item的UICollectionViewLayoutAttributes
NSArray *array = [super layoutAttributesForElementsInRect:rect];
NSArray * attributes = [[NSArray alloc] initWithArray:array copyItems:YES]; // 2.计算屏幕最中间的x
CGFloat centerX = self.collectionView.contentOffset.x + self.collectionView.frame.size.width * 0.5; for (UICollectionViewLayoutAttributes *attr in attributes) { if (!CGRectIntersectsRect(visibleRect, attr.frame)) continue; // 每一个item中点x
CGFloat itemCenterX = attr.center.x;
//计算item离屏幕中间的距离
CGFloat distance = ABS(itemCenterX - centerX);
//距离越小,缩放比例越大
CGFloat zoom = - (distance / (self.collectionView.frame.size.width * 0.5));
//NSLog(@"%f",zoom);
//缩放比例
CGFloat scale = + 0.5 * zoom;
//缩放
attr.transform3D = CATransform3DMakeScale(scale, scale, );
} return attributes;
} //用来设置UICollectionView停止滚动那一刻的位置
- (CGPoint)targetContentOffsetForProposedContentOffset:(CGPoint)proposedContentOffset withScrollingVelocity:(CGPoint)velocity
{
// 1.计算UICollectionView最终停止的范围
CGRect lastRect;
lastRect.origin = proposedContentOffset;
lastRect.size = self.collectionView.frame.size;
// 计算屏幕中间的x
CGFloat centerX = proposedContentOffset.x + self.collectionView.frame.size.width * 0.5;
// 2.取出这个范围内的所有属性
NSArray *array = [super layoutAttributesForElementsInRect:lastRect];
NSArray * attributes = [[NSArray alloc] initWithArray:array copyItems:YES]; // 3.遍历所有属性
CGFloat adjustOffsetX = MAXFLOAT;
for (UICollectionViewLayoutAttributes *attr in attributes) {
if (ABS(attr.center.x - centerX) < ABS(adjustOffsetX)) {
adjustOffsetX = attr.center.x - centerX;
}
}
return CGPointMake(proposedContentOffset.x + adjustOffsetX, proposedContentOffset.y);
} @end
FoldLayout.m(折叠布局)
//
// FoldLayout.m
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/27.
// Copyright © 2016年 博文科技. All rights reserved.
// #import "FoldLayout.h" #define Random0_1 (arc4random_uniform(100)/100) @implementation FoldLayout - (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds
{
return YES;
} - (CGSize)collectionViewContentSize
{
return CGSizeMake(, );
} - (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath
{
NSArray *angles = @[@, @(-0.2), @(-0.5), @, @(0.2), @(0.5)]; UICollectionViewLayoutAttributes *attrs = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:indexPath];
attrs.size = CGSizeMake(, );
// attrs.center = CGPointMake(arc4random_uniform(self.collectionView.frame.size.width), arc4random_uniform(self.collectionView.frame.size.height));
attrs.center = CGPointMake(self.collectionView.frame.size.width * 0.5, self.collectionView.frame.size.height * 0.5);
if (indexPath.item >= ) {
attrs.hidden = YES;
} else {
attrs.transform = CGAffineTransformMakeRotation([angles[indexPath.item] floatValue]);
//zIndex越大,就越在上面
attrs.zIndex = [self.collectionView numberOfItemsInSection:indexPath.section] - indexPath.item;
}
return attrs;
} - (NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect
{
NSMutableArray *array = [NSMutableArray array];
NSInteger count = [self.collectionView numberOfItemsInSection:]; for (int i=; i<count; i++) {
// UICollectionViewLayoutAttributes *attrs = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:[NSIndexPath indexPathForItem:i inSection:0]];
// attrs.size = CGSizeMake(100, 100);
// attrs.center = CGPointMake(self.collectionView.frame.size.width * 0.5, self.collectionView.frame.size.height * 0.5);
// if (i >= 5) {
// attrs.hidden = YES;
// } else {
//
// NSArray *angles = @[@0, @(-0.2), @(-0.5), @(0.2), @(0.5)];
// attrs.transform = CGAffineTransformMakeRotation([angles[i] floatValue]);
// //zIndex越大,就越在上面
// attrs.zIndex = count - i;
NSIndexPath *indexPath = [NSIndexPath indexPathForItem:i inSection:];
UICollectionViewLayoutAttributes *attrs = [self layoutAttributesForItemAtIndexPath:indexPath];
[array addObject:attrs]; }
return array;
} @end
CircleLayout.m(环形布局)
//
// CircleLayout.m
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/27.
// Copyright © 2016年 博文科技. All rights reserved.
// #import "CircleLayout.h" @implementation CircleLayout - (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds
{
return YES;
} - (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath
{ UICollectionViewLayoutAttributes *attrs = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:indexPath];
attrs.size = CGSizeMake(, ); //圆的半径
CGFloat radius = ;
CGPoint cireclecenter = CGPointMake(self.collectionView.frame.size.width * 0.5, self.collectionView.frame.size.height * 0.5);
//每个item间角度
CGFloat angleDetla = M_PI * / [self.collectionView numberOfItemsInSection:indexPath.section]; //计算当前item角度
CGFloat angle = indexPath.item * angleDetla;
attrs.center = CGPointMake(cireclecenter.x + radius * cosf(angle), cireclecenter.y + radius * sinf(angle)); attrs.zIndex = indexPath.item;
return attrs;
} - (NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect
{
NSMutableArray *array = [NSMutableArray array];
NSInteger count = [self.collectionView numberOfItemsInSection:]; for (int i=; i<count; i++) {
NSIndexPath *indexPath = [NSIndexPath indexPathForItem:i inSection:];
UICollectionViewLayoutAttributes *attrs = [self layoutAttributesForItemAtIndexPath:indexPath];
[array addObject:attrs]; }
return array;
} @end
ImgCell.h(自定义UICollectionViewCell)
//
// ImgCell.h
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/26.
// Copyright © 2016年 博文科技. All rights reserved.
// #import <UIKit/UIKit.h> @interface ImgCell : UICollectionViewCell @property (weak, nonatomic) IBOutlet UIImageView *imgView;
@property (nonatomic, copy) NSString *image; @end //
// ImgCell.m
// IOS_0226_自定义UIColectionView布局
//
// Created by ma c on 16/2/26.
// Copyright © 2016年 博文科技. All rights reserved.
// #import "ImgCell.h" @interface ImgCell () @end @implementation ImgCell - (void)setImage:(NSString *)image
{
_image = [image copy];
self.imgView.image = [UIImage imageNamed:_image];
} - (void)awakeFromNib {
self.imgView.layer.borderWidth = ;
self.imgView.layer.borderColor = [UIColor whiteColor].CGColor;
self.imgView.layer.cornerRadius = ;
self.imgView.clipsToBounds = YES;
} @end
IOS UI-自定义UIColectionView布局的更多相关文章
- iOS开发自定义流水布局
//集成UICollectionViewFlowLayout 自己写的布局 // SJBFlowLayout.m // 自定义流水布局 // // Created by zyyt on 16/7 ...
- Android开发1:基本UI界面设计——布局和组件
前言 啦啦啦~本学期要开始学习Android开发啦~ 博主在开始学习前是完完全全的小白,只有在平时完成老师要求的实验的过程中一步一步学习~从此篇博文起,博主将开始发布Android开发有关的博文,希望 ...
- [前端]使用JQuery UI Layout Plug-in布局 - wolfy
引言 使用JQuery UI Layout Plug-in布局框架实现快速布局,用起来还是挺方便的,稍微研究了一下,就能上手,关于该布局框架的材料,网上也挺多的.在项目中也使用到了,不过那是前端的工作 ...
- JQuery UI Layout Plug-in布局
端]使用JQuery UI Layout Plug-in布局 引言 使用JQuery UI Layout Plug-in布局框架实现快速布局,用起来还是挺方便的,稍微研究了一下,就能上手,关于该布 ...
- 【详细】Android入门到放弃篇-YES OR NO-》各种UI组件,布局管理器,单元Activity
问:达叔,你放弃了吗? 答:不,放弃是不可能的,丢了Android,你会心疼吗?如果别人把你丢掉,你是痛苦呢?还是痛苦呢?~ 引导语 有人说,爱上一个人是痛苦的,有人说,喜欢一个人是幸福的. 人与人之 ...
- [前端]使用JQuery UI Layout Plug-in布局
引言 使用JQuery UI Layout Plug-in布局框架实现快速布局,用起来还是挺方便的,稍微研究了一下,就能上手,关于该布局框架的材料,网上也挺多的.在项目中也使用到了,不过那是前端的工作 ...
- iOS 如何自定义UISearchBar 中textField的高度
iOS 如何自定义UISearchBar 中textField的高度 只需设置下边的方法就可以 [_searchBar setSearchFieldBackgroundImage:[UIImage i ...
- 原生HTML5 input type=file按钮UI自定义
原生<input type="file" name="file" />长得太丑 提升一下颜值 实现方案一.设置input[type=file]透明度 ...
- iOS 隐藏自定义tabbar
iOS 隐藏自定义tabbar -(void)viewWillAppear:(BOOL)animated { NSArray *array=self.tabBarController.view.su ...
随机推荐
- php 单线程 (http://bbs.csdn.net/topics/390778072)
以前想php单线程,网站肯定是用于多人访问的,如果访问量大,那岂不是出现排队问题? apache+php是阻塞型处理,nginx+php是异步非阻塞的,php有进程管理器,fpm fcgi什么的.ph ...
- linux上mysql访问:Access denied for user 'agtipay'@'iZm5ebiyb4f90ga9xiycgsZ' (using password: YES)
公司的聚合支付测试环境出了一个问题(agtipay用户访问数据的时候出现如题错误),快搞死我两天时间(原谅技术不才),如题.首先明确一下问题: 1.访问拒绝,说明数据库连接这里有问题,数据库连接访问拒 ...
- 手写Bind
Function.prototype.bind2 = function(context){ var self = this; var args = [].slice.call(arguments,1) ...
- python3_ftp文件传输
Python中的ftplib模块 Python中默认安装的ftplib模块定义了FTP类,其中函数有限,可用来实现简单的ftp客户端,用于上传或下载文件 FTP的工作流程及基本操作可参考协议RFC95 ...
- 将std::array转换成std::tuple
template<typename Array, std::size_t... Index> decltype(auto) array2tuple_impl(const Array& ...
- [转]tomcat之一:指定tomcat运行时JDK版本
今天在做项目的时候,主管让我在本机上启动多个tomcat且指定不同的jdk环境.因为在企业的项目中个,对于同一个服务器中有多个jdk和tomcat,所以就需要手动指定不同的jdk. 在网上找了很多资料 ...
- 20145310《Java程序设计》第4次实验报告
20145310<Java程序设计>第4次实验报告 实验内容 搭建Android环境 运行Android 修改代码并输出自己的学号 实验步骤 搭建Android环境 安装Android S ...
- 20145312 《Java程序设计》第三周学习总结
20145312 <Java程序设计>第三周学习总结 学习笔记 Chapter 4 4.1类与对象 4.1.1 定义类 1.以服饰设计为例,类定义用class关键词,建立衣服实例要用new ...
- 20145313 《Java程序设计》第十周学习总结
网络编程 网络编程就是在两个或两个以上的设备(例如计算机)之间传输数据.程序员所作的事情就是把数据发送到指定的位置,或者接收到指定的数据,这个就是狭义的网络编程范畴.在发送和接收数据时,大部分的程序设 ...
- openwrt下定义软件包的依赖关系类型
在openwrt下软件包的依赖关系由DEPENDS:=来指定 第一种依赖关系类型为只有将依赖的软件包手动选上,当前的软件包就会自动被选中,用法为DEPENDS:=package_name 第二种依赖关 ...