进击的KFC:UI(三)自定义视图、视图控制器

2023-12-22 08:38

本文主要是介绍进击的KFC:UI(三)自定义视图、视图控制器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

⼀、⾃定义label-textField视图
1.1⾃定义视图:系统标准UI之外,⾃⼰组合⽽出的新的视图。
iOS提供了很多UI组件,借助它们,我们可以做各种程序。
尽管如此,实际开发中,我们还需⾃定义视图。积累⾃⼰的代码库。
⽅便开发。⾃⼰封装的视图,能像系统UI控件⼀样,⽤于别的项⺫
中,能⼤⼤降低开发成本,提⾼开发效率。

1.2⾃定义视图步骤
根据需求的不同,⾃定义视图继承的类也有所不同。⼀般⾃定义的
视图会继承于UIView。以下是⾃定义视图的要点:
1、创建⼀个UIView⼦类
2、在类的初始化⽅法中添加⼦视图
3、类的.h⽂件提供⼀些接⼝(⽅法),便于外界操作⼦视图

1.3label-textField视图
假设我们使⽤LTView类代表label-textfield视图。
我们创建⼀个LTView类继承于UIView。
我们将LTView作为⼀个容器,在LTView的初始化⽅法中创建并添加
label和textField。
此时的LTView就变成了⼀个具有label和textField的视图了
这里写图片描述
主要是在LTView初始化的时候,把UILabel,和 UITextField也初始化并添加到LTView上去

- (instancetype)initWithFrame:(CGRect)frame
{self = [super initWithFrame:frame];if (self) {     // 用属性写:   // 获取动态宽度CGFloat width = frame.size.width;// 获取动态高度CGFloat height = frame.size.height;     // 需要根据上面的宽度,来添加labelself.label = [[[UILabel alloc]initWithFrame:CGRectMake(0, 0, width / 3, height)]autorelease];// self.label.backgroundColor = [UIColor greenColor];// 添加到自己身上[self addSubview:self.label];self.textField = [[[UITextField alloc] initWithFrame:CGRectMake(width / 3 + 20, 0, width * 2 / 3 - 20, height)]autorelease];//  self.textField.backgroundColor = [UIColor magentaColor];self.textField.borderStyle = UITextBorderStyleRoundedRect;[self addSubview:self.textField];}return  self;
}

二,视图控制器
UIViewController
2.1视图控制器概述
UIViewController:视图控制器。
控制视图显⽰,响应事件。
分担AppDelegate的⼯作。
实现模块独⽴,提⾼复⽤性。

2.2视图控制器功能
控制视图⼤⼩变换、布局视图、响应事件。
检测以及处理内存警告。
检测以及处理屏幕旋转。
检测视图的切换。
2.3在AppDelegate.m中:设置根视图控制器

// 创建控制器RootViewController *rootVC = [[RootViewController alloc]init];// 设置根视图self.window.rootViewController = rootVC;// 释放[rootVC release];

2.4MVC概述
MVC是一种设计的模式,好处:让视图View可以复用
应用 1.首先 视图层(View)只写视图的布局
2.数据模型层(Model) 只写数据的结构 (直接写NS开头的)
3.控制器(Controller) 负责逻辑部分, ( 例如从Model里取数据, 然后 去更改视图显示 )
MVC把工程中的代码 模块化(降低耦合度) 尽量让 视图部分 的代码可以复用
一般一个视图控制器,控制一个页面

三、视图控制器指定⾃定义view
3.1UIViewController⾃带⼀个空的view,与需求不符合。
视图控制器只负责控制视图显⽰,响应事件
3.2如何设置?
⾃定义视图类继承UIView。在初始化⽅法中添加⼦视图控件。
重写controller的loadView⽅法。创建⾃定义视图对象,并指定为controller
的view。
将⼦视图控件对象设置为⾃定义视图类的属性,在viewDidLoad⽅法中进
⾏设置:添加action、设置delegate等等。
在controller中添加按钮点击事件实现和代理⽅法的实现。

需要记住的两个方法:1.添加的时候:self.secondVC = [[SecondViewController alloc]init];// 把secondVC添加成子控制器 让rootVC去管理[self addChildViewController:self.secondVC];// 把secondVC自带的.view添加到rootVC自带的.view上 (把自控制器的视图显示出来)[self.view addSubview:self.secondVC.view];2.移除的时候// 先移除视图[self.secondVC.view removeFromSuperview];// 再移除子视图控制器[self.secondVC removeFromParentViewController];

四、检测屏幕旋转

/*屏幕旋转横屏时,横屏布局竖屏时,竖屏布局步骤:1.允许屏幕旋转2.指定屏幕旋转的方向3.找到旋转触发的方法4.改变布局5.测试*/
1.RootViewController.m中
// 屏幕旋转
// view重写layoutSubviews⽅法,根据设备⽅向,重新布局
// [UIApplication shareApplication].statusBarOrientation提供设备当前⽅向。// 1.允许屏幕旋转
- (BOOL)shouldAutorotate
{return YES;
}// 2.指定屏幕旋转的方向
- (NSUInteger)supportedInterfaceOrientations
{return UIInterfaceOrientationMaskAll;   
}// 看一下:宽和高的互换
-(void)viewWillTransitionToSize:(CGSize)size withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator
{NSLog(@"%@",NSStringFromCGSize(size)); // 打印旋转时,宽和高的互换 
}2.LoginView.m中
// 重新布局子视图
// frame发生变化的时候,会触发layoutSubviews方法 (这是重点)
-(void)layoutSubviews
{// 因为不知道父类这个方法 做了啥// 所以需要先调用写一下的 先调用父类的方法// 然后 再写咱们自己的[super layoutSubviews];// frame发生变化,相当于横屏了,这时咱们需要重新布局// 判断竖屏还是横屏// 1.把应用程序取出来,// 2.判断当前应用程序屏幕的朝向// 取出应用程序 单例方法的命名规范share...什么UIApplication *app = [UIApplication sharedApplication];// 判断方向// 若为上下if (app.statusBarOrientation == UIInterfaceOrientationPortrait || app.statusBarOrientation ==UIInterfaceOrientationPortraitUpsideDown) {NSLog(@"竖着");// 不变,上面粘过来self.userNameLTView.frame = CGRectMake((kScreenWidth - 300)/ 2 , 100, 300, 50);self.passwordLTView.frame = CGRectMake(self.userNameLTView.frame.origin.x, self.userNameLTView.frame.origin.y + self.userNameLTView.frame.size.height +  kRowHeight, self.userNameLTView.frame.size.width, self.userNameLTView.frame.size.height);}else{NSLog(@"横着");// 重新设置frame,无需创建新的LTViewself.passwordLTView.frame = CGRectMake(self.userNameLTView.frame.origin.x + self.userNameLTView.frame.size.width + 10, self.userNameLTView.frame.origin.y, self.userNameLTView.frame.size.width, self.userNameLTView.frame.size.height);}
}

五、处理内存警告
5.1处理内存警告
控制器能监测内存警告,以便我们避免内存不够引起的crash。
在定义的controller⼦类中重写didReceiveMemoryWarning⽅法。
释放暂时不使⽤的资源。(数据对象、图像)

// 接到内存警告
- (void)didReceiveMemoryWarning {[super didReceiveMemoryWarning];// Dispose of any resources that can be recreated.    NSLog(@"内存警告了");   // 释放已经被显示过的,并且不是当前显示的视图// (如果这个视图正在被显示,那么,这个视图的window属性不是空的)if ([self isViewLoaded] == YES && self.view.window == nil) { // 只有被加载过,并且不在被展示,才能呗释放// 把当前视图 释放self.view = nil;}
}

总结:
iOS提供了很多标准组件,标准组件完成我们的需要,但是适当的
封装,组合⼀个新的控件⽆疑能给我们带来效率上的提⾼。学会⾃
定义控件是你成为⾼⼿的关键⼀步。
视图控制器是MVC中的C,⽤于处理视图触发的事件。
处理旋转和内存警告

这篇关于进击的KFC:UI(三)自定义视图、视图控制器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/523306

相关文章

聊聊springboot中如何自定义消息转换器

《聊聊springboot中如何自定义消息转换器》SpringBoot通过HttpMessageConverter处理HTTP数据转换,支持多种媒体类型,接下来通过本文给大家介绍springboot中... 目录核心接口springboot默认提供的转换器如何自定义消息转换器Spring Boot 中的消息

Python自定义异常的全面指南(入门到实践)

《Python自定义异常的全面指南(入门到实践)》想象你正在开发一个银行系统,用户转账时余额不足,如果直接抛出ValueError,调用方很难区分是金额格式错误还是余额不足,这正是Python自定义异... 目录引言:为什么需要自定义异常一、异常基础:先搞懂python的异常体系1.1 异常是什么?1.2

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

C语言自定义类型之联合和枚举解读

《C语言自定义类型之联合和枚举解读》联合体共享内存,大小由最大成员决定,遵循对齐规则;枚举类型列举可能值,提升可读性和类型安全性,两者在C语言中用于优化内存和程序效率... 目录一、联合体1.1 联合体类型的声明1.2 联合体的特点1.2.1 特点11.2.2 特点21.2.3 特点31.3 联合体的大小1

Django中的函数视图和类视图以及路由的定义方式

《Django中的函数视图和类视图以及路由的定义方式》Django视图分函数视图和类视图,前者用函数处理请求,后者继承View类定义方法,路由使用path()、re_path()或url(),通过in... 目录函数视图类视图路由总路由函数视图的路由类视图定义路由总结Django允许接收的请求方法http

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束

WinForm跨线程访问UI及UI卡死的解决方案

《WinForm跨线程访问UI及UI卡死的解决方案》在WinForm开发过程中,跨线程访问UI控件和界面卡死是常见的技术难题,由于Windows窗体应用程序的UI控件默认只能在主线程(UI线程)上操作... 目录前言正文案例1:直接线程操作(无UI访问)案例2:BeginInvoke访问UI(错误用法)案例

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注