UIScrollview 技巧

设置UIScrollView的contentSize


如果使用自动布局,那么它会自动帮你基于这个scrollview的子视图的约束来计算这个内容大小。在非自动布局情况下,如果app旋转导致scrollview 的bounds改变,不会影响到scrollview的contentSize,而如果重新设置contentSize,也不会影响scrollview的子视图,这个contentSize仅仅是决定了滚动的范围。

下面我们用代码创建一个UIScrollView,UILabel在y轴上顺序排列:

240753254335316.png

UIScrollView* sv = [UIScrollView new];
sv.backgroundColor = [UIColor whiteColor];
sv.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:sv];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[sv]|"options:0 metrics:nilviews:@{@"sv":sv}]];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[sv]|"options:0 metrics:nilviews:@{@"sv":sv}]];
UILabel* previousLab = nil;
for (int i=0; i<30; i++) {UILabel* lab = [UILabel new];lab.translatesAutoresizingMaskIntoConstraints = NO;lab.text = [NSString stringWithFormat:@"This is label %d", i+1];[sv addSubview:lab];[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab}]];if (!previousLab) { // first one, pin to top[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab}]];} else { // all others, pin to previous[sv addConstraints:[NSLayoutConstraintconstraintsWithVisualFormat:@"V:[prev]-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab, @"prev":previousLab}]];}previousLab = lab;
}

运行代码,你会发现label 都放置在正确的位置,但是scrollView 却不能滚动。而且,即使你手动设置了contentSize也无法滚动。原因就是在页面布局时,scrollview的contentSize会自动根据它和子视图之间的约束来计算得出。解决方法就是,再添加多一个约束,告诉scroll view 它的contentSize的高度应该是多少:

// 给底部最后一个label设置一个y轴方向的约束,这样高度就可以确定了
[sv addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:[lab]-(10)-|"options:0 metrics:nilviews:@{@"lab":previousLab}]];

可以看到,我们可以在垂直方向上滚动scrollview了,在水平方向上仍然不能滚动(contentSize的宽度默认就是0,正好是我们需要的)。

使用一个Content View

我们一般不会把scrollview 的子视图直接添加到scrollview上,而是添加到UIView上,然后把这个视图再添加到scrollview上,这个视图就是这个Content View。这样可以更加方便的组织和使用。

在自动布局下,我们可以有两种设置scroll view的contentSize 方法:

  • 设置content view 的translatesAutoresizingMaskIntoConstraints为YES,然后手动设置这个scroll view的 contentSize为这个content view的大小。

  • 设置content view 的translatesAutoresizingMaskIntoConstraints为NO,然后给这个content view 设置宽度和高度约束。

设置content view的大小或者约束这种方式是与它的子视图怎么定位(设置frame,还是实用约束)无关。有四种可能的组合,这四种组合开头的代码都是一样的:

UIScrollView* sv = [UIScrollView new];
sv.backgroundColor = [UIColor whiteColor];
sv.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:sv];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[sv]|"options:0 metrics:nilviews:@{@"sv":sv}]];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[sv]|"options:0 metrics:nilviews:@{@"sv":sv}]];
UIView* v = [UIView new]; // content view
[sv addSubview: v];

第一种组合不使用约束:

CGFloat y = 10;
for (int i=0; i<30; i++) {UILabel* lab = [UILabel new];lab.text = [NSString stringWithFormat:@"This is label %d", i+1];[lab sizeToFit];CGRect f = lab.frame;f.origin = CGPointMake(10,y);lab.frame = f;[v addSubview:lab]; // add to content view, not scroll viewy += lab.bounds.size.height + 10;
}
// set content view frame and content size explicitly
v.frame = CGRectMake(0,0,0,y);
sv.contentSize = v.frame.size;

第二种组合,content view使用约束,但它的子视图不使用:

CGFloat y = 10;
for (int i=0; i<30; i++) {// ... same as before, create labels, keep incrementing y
}
// configure the content view using constraints
v.translatesAutoresizingMaskIntoConstraints = NO;
[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[v(y)]|"options:0 metrics:@{@"y":@(y)} views:@{@"v":v}]];
[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[v(0)]|"options:0 metrics:nil views:@{@"v":v}]];

第三种组合,content view 和它的子视图都使用约束:

UILabel* previousLab = nil;
for (int i=0; i<30; i++) {UILabel* lab = [UILabel new];lab.translatesAutoresizingMaskIntoConstraints = NO;lab.text = [NSString stringWithFormat:@"This is label %d", i+1];[v addSubview:lab];[v addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab}]];if (!previousLab) { // first one, pin to top[v addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab}]];} else { // all others, pin to previous[v addConstraints:[NSLayoutConstraintconstraintsWithVisualFormat:@"V:[prev]-(10)-[lab]"options:0 metrics:nilviews:@{@"lab":lab, @"prev":previousLab}]];
}previousLab = lab;
}
// last one, pin to bottom, this dictates content size height
[v addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:[lab]-(10)-|"options:0 metrics:nilviews:@{@"lab":previousLab}]];
// configure the content view using constraints
v.translatesAutoresizingMaskIntoConstraints = NO;
[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[v]|"options:0 metrics:nil views:@{@"v":v}]];
[sv addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[v(0)]|"options:0 metrics:nil views:@{@"v":v}]];

第四种组合是很奇怪的组合,content view的子视图使用约束布局,但是content view 和 scrollview 不使用:

UILabel* previousLab = nil;
// ... same as before, add subviews and constraints to content view
// autolayout helps us learn the consequences of those constraints
CGSize minsz = [v systemLayoutSizeFittingSize:UILayoutFittingCompressedSize];
// set content view frame and content size explicitly
v.frame = CGRectMake(0,0,0,minsz.height);
sv.contentSize = v.frame.size;

滚动


在iOS 7 下,由于app都是全屏的,status bar,navigation bar 都是半透明的,所以scroll view 也会在 status bar 下面,那么有可能导致scrollview 的内容被status bar遮盖,我们可以设置:

sv.contentInset = UIEdgeInsetsMake(20, 0, 0, 0);

当我们设置了contentInset,我们通常也会设置scrollIndicatorInsets,让滚动条也适应:

sv.contentInset = UIEdgeInsetsMake(20, 0, 0, 0);
sv.scrollIndicatorInsets = sv.contentInset;

这样,即可以在status bar 看到半透明的scrollview 内容,又不会滚动不下导致的显示不全,但是我们硬编码这个20的值好像不是很优雅,我们可以:

- (void) viewWillLayoutSubviews {if (self.sv) {CGFloat top = self.topLayoutGuide.length;CGFloat bot = self.bottomLayoutGuide.length;self.sv.contentInset = UIEdgeInsetsMake(top, 0, bot, 0);self.sv.scrollIndicatorInsets = self.sv.contentInset;}
}

这些如果在nib中,如果我们设置view controller的automaticallyAdjustsScrollViewInsets为YES,那么iOS 7 运行时会帮我们自动适应这个scrollview 的contentInset 和 scrollIndicatorInsets,而不需要上面的代码。这个仅仅在nib上有用,一旦我们手动创建scrollview,即使设置这个automaticallyAdjustsScrollViewInsets为YES也没有用,还是需要上面的代码。

Tiling 平铺

假如我们有一个非常大的内容需要显示在scroll view上,这么大的图片,用户可以滚动来查看所有的内容细节。在内存中存放整张图片是不现实的,也是不可能的。

Tiling是一个种解决方法。我们把内容分成一个个小的矩形区域,当用户滚动时,我们查找并让目标的矩形区块显示,同时我们可以释放不在显示范围内的矩形区域。

有一个内建的CALayer子类 CATiledLayer 帮我们实现了这个分块。它的tileSize属性设置区块的大小。它的drawLayer:inContext: 会在需要一个空的区块时被调用;在图形上下文中调用CGContextGetClipBoundingBox 来截取所需区块的位置。下面我们借用苹果自己提供的PhotoScroller例子来说明:

我们给scroll view添加一个子视图,一个TiledView,这个视图用来存放我们的CATiledLayer图层。TILESIZE是256:

-(void)viewDidLoad {CGRect f = CGRectMake(0,0,3*TILESIZE,3*TILESIZE);TiledView* content = [[TiledView alloc] initWithFrame:f];float tsz = TILESIZE * content.layer.contentsScale;[(CATiledLayer*)content.layer setTileSize: CGSizeMake(tsz, tsz)];[self.sv addSubview:content];[self.sv setContentSize: f.size];
}

下面是TiledView的代码,CATiledLayer是它的根图层,所以TiledView是CATiledLayer的委托。意味着当CATiledLayer的 drawLayer:inContext: 被调用时,TiledView的drawRect:方法也会被调用,而且我们必须使用imageWithContentsOfFile:方法获取图片,而不是imageNamed:,防止系统缓存这张图片:

+ (Class) layerClass {return [CATiledLayer class];
}
-(void)drawRect:(CGRect)r {CGRect tile = r;int x = tile.origin.x/TILESIZE;int y = tile.origin.y/TILESIZE;NSString *tileName =[NSString stringWithFormat:@"CuriousFrog_500_%d_%d", x+3, y];NSString *path =[[NSBundle mainBundle] pathForResource:tileName ofType:@"png"];UIImage *image = [UIImage imageWithContentsOfFile:path];[image drawAtPoint:CGPointMake(x*TILESIZE,y*TILESIZE)];
}

上面的代码中并没有明确释放离屏区块,你可以在TiledView中调用setNeedsDisplay 或者setNeedsDisplayInRect: 方法,但是这样并不会清除离屏的区块,我们相信这个CATiledLayer会帮我们处理好。

转载于:https://www.cnblogs.com/YungMing/p/4346768.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/575521.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

NiFi导出自己模板和导入别人模板

NiFi导出模板 进入nifi系统后&#xff0c;选择自己想要导出的流程或者组&#xff08;支持group作为模板导出&#xff09; 点击空白处&#xff0c;右键选择create template 模板文件下载成功 NiFi导入模板 点击空白处&#xff0c;右键选择upload template 查看导入结…

Node.js开发入门(一)——安装Node.js及编辑器配置

Node.js是一个轻松构建快速&#xff0c;可扩展的网络应用平台建立在Chrome的JavaScript运行。Node.js使用事件驱动&#xff0c;非阻塞I/O模型&#xff0c;使得它重量轻&#xff0c;高效&#xff0c;完美的数据密集型实时应用程序运行在分布式设备。 Node.js是让JavaScript脱离浏…

架构风格与基于网络的软件架构设计

原文链接 https://blog.csdn.net/on_1y/article/details/60358117 架构风格与基于网络的软件架构设计 如今许多服务都采用了 RESTful API, 而 REST 这一架构风格&#xff0c;最早即来源于 Roy Thomas Fielding 的博士论文 Architectural Styles and the Design of Network-bas…

win7安装nodejs 高版本不支持 换低版本

win7安装nodejs失败 显示This application is only supported on Windows 8.1,Windows Server 2012 R2,or higer. win7安装nodejs失败 在win7重装nodejs&#xff0c;版本是node-v14.16.1-x64&#xff0c;安装时报了win7系统不支持的问题 2021年4月8日 —— 目前除了最新的v14大…

研究生导师一般希望招什么样的研究生?

虽然还不是导师&#xff0c;但也在学校蹲了不少年头了&#xff0c;看着一级级研究生毕业奔赴各方&#xff0c;其实还是蛮有感触的。 各位同学很多在硕士期间回忆并不算好&#xff0c;其实就是从一开始没有搞明白一件事&#xff1a;导师招生为了什么&#xff0c;我们上研又为了什…

Android Fragment应用实战

现在Fragment的应用真的是越来越广泛了&#xff0c;之前Android在3.0版本加入Fragment的时候&#xff0c;主要是为了解决Android Pad屏幕比较大&#xff0c;空间不能充分利用的问题&#xff0c;但现在即使只是在手机上&#xff0c;也有很多的场景可以运用到Fragment了&#xff…

初识树莓派

初识树莓派这一微型计算机&#xff0c;基本了解概念&#xff0c; 1、树莓派简介 Raspberry Pi(中文名为“树莓派”,简写为RPi&#xff0c;或者 RasPi/RPi)是为学生计算机编程教育而设计&#xff0c;只有信用卡大小的卡片式电脑&#xff0c;其系统基于Linux开发而来的。树莓派由…

ffmpeg抓取rtsp流并保存_详细解析RTSP框架和数据包分析(1)

0.引言 本文主要讲解RTSP框架和抓取RTSP数据包&#xff0c;进行详细分析。可以阅读以下几篇文章&#xff0c;能够帮助你更详细理解。 手把手搭建RTSP流媒体服务器 HLS实战之Wireshark抓包分析 HTTP实战之Wireshark抓包分析 1.RTSP协议简述 RTSP&#xff1a;Real Time Stream…

xshell 软件的窗口一直是置顶 调整为不置顶

突然感觉xshell 软件的窗口一直是置顶状态&#xff0c;点别的软件窗口点不过去&#xff0c; 搜索一下&#xff1a;微信的 alta是截图&#xff0c;和xshell这个冲突了。如果在xshell下按了这个&#xff0c;不知不觉就被置顶了。。。。。MMP 即这个意思&#xff1a; 如果要修改…

Linux下Vim工具常用命令

原文地址&#xff1a; http://www.cnblogs.com/lizhenghn/p/3675011.html 在linux下做开发&#xff0c;甚至是只做管理维护工作&#xff0c;也少不了Vim的使用。作为一个新手&#xff0c;我也是刚刚接触&#xff0c;本节将我日常使用或收集的Vim常用命令记录下来。 当然&#x…

测试telnet安装成功 以及如何进入和退出telnet命令模式

telnet装好之后&#xff0c;测试。 输入&#xff1a;telnet www.baidu.com 80 出现如图提示是意思是&#xff1a; 正常连通。 然后 按 ctrl ] 进入telnet客户端命令模式&#xff0c;然后用quit退出。进入linux命令模式。

数据按时间拆开分批处理示例

我现在的问题是有一个大的事实表,已经有数十亿条数据,过来的临时表需要merge进去. 临时表的大小也不确定,可能上十亿也可能只有几百几千万而已. 如果直接让这两个表merge起来,则需要很大的内存来进行处理.所以我就想着把数据进行按时间拆分的处理,然后merge进去, 拆分的条件是如…

telnet远程登陆、mstsc远程控制、SSH之间的比较和区别

Telnet与远程桌面mstsc 一 演示 打开mstsc演示&#xff1a; 01 win r 输入mstsc 02 输入要远程的机子的用户名和密码 03 登录 04 连接开启了&#xff0c;如图即是通过win窗口操作远程机子。 以上是使用mstsc打开一个远程连接来通过windows窗口的方式来控制远程机器。 …

Linux查看已经开放的端口,开放端口遇到防火墙的问题。

Linux查看已经开放的端口&#xff0c;开放端口&#xff0c;FirewallD is not running 查看已经开放的端口 firewall-cmd --list-all1 如果出现这种情况说明你的防火墙没有开: 则先把防火墙打开&#xff1a;systemctl start firewalld 扩展内容&#xff1a;. 查看firewalld状…

用yum装程序 报[Errno 12] Timeout on Trying other mirror.

使用yum&#xff0c;报错如图下&#xff1a; 记着之前网卡和dns文件都配置好着的&#xff0c;114.114.114.114都加上的。 检查了下&#xff0c;发现dns文件的配置不见了。。。。 故把 /etc/resolv.conf 中加了如下&#xff1a;然后重点不要重启 service network restart &#…

JavaScript中的原型和继承

请在此暂时忘记之前学到的面向对象的一切知识。这里只需要考虑赛车的情况。是的&#xff0c;就是赛车。 最近我正在观看 24 Hours of Le Mans &#xff0c;这是法国流行的一项赛事。最快的车被称为 Le Mans 原型车。这些车虽然是由“奥迪”或“标致”这些厂商制造的&#xff0c…

Centos7.x 网卡启动报错(Failed to start LSB: Bring up/down networking)

环境&#xff1a;华为云服务器、 CentOS 7.x 操作是 调整网卡配置文件和resolv.conf &#xff0c;systemctl restart network 重启网络服务总是失败&#xff1a;如下&#xff1a; 按提示看细节&#xff1a; 搜索到还算靠谱的帖子,都说 大多都是网卡配置文件配置错误&#x…

计算机网络复习-OSI TCP/IP 物理层

我膨胀了&#xff0c;挂我啊~ 作者简介&#xff1a; 每年都吐槽吉师网安奇怪的课程安排、全校正经学网络安全不超20人情景以及割韭菜企业合作的FW&#xff0c;今年是第一年。。 TCP/IP模型 先做两道题&#xff1a; TCP/IP协议模型由高层到低层分为哪几层&#xff1a; 这题…

J2EE的十三个规范

J2EE想必大家都不陌生吧&#xff0c;貌似现在更流行将其称作JavaEE&#xff0c;不管名字怎么变&#xff0c;核心和思想是没有变的。学习J2EE首先要了解它的规范&#xff0c;下面我们一起看看它的十三个规范。 1&#xff0c;JDBC&#xff08;Java Database Connectivity&#xf…

Xshell登录Linux服务器 提示WARNING! The remote SSH server rejected X11 forwarding request 及 提示符显示-bash-4.2#

用root登录 Xshell 登录Linux服务器 提示 WARNING! The remote SSH server rejected X11 forwarding request 以及 用户表示符是&#xff1a;-bash-4.2# 而不是user主机名 路径的显示方式。 首先先查 The remote SSH server rejected X11 forwarding request。 解决过程 …