iOS 使用UI控件的外观协议UIAppearance进行设置默认UI控件样式

在iOS开发中,经常会对UINavigationBar的样式进行全局样式。采用的设置方式有两种:

第一种,采用方式如下:

[UINavigationBar appearance]

这种是对一类对象的默认全局外观样式设置,它对设置时机有要求。

通常需要在UIWindow的viewlayout之前。错过了时机后,设置是没有效果的。

可以选择在下面方法内设置:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.

    return YES;
}

第二种,采用方式如下:

[self.navigationController.navigationBar setBarTintColor:[UIColor redColor]];

这种是即时性的,设置后就会有效果。

第一种方式通过appearance使得UINavigationBar产生了全局默认样式,那么appearance是什么东西呢?

Appearance(外观协议)讲解

UIAppearance文件包含于UIKit框架中,该文件很短,包括注释加起来总共56行

点击去看看。

里面有一个宏定义和两个协议声明,分别如下:

1.宏定义:

#define UI_APPEARANCE_SELECTOR __attribute__((annotate("ui_appearance_selector")))

由注释可知:凡是被这个宏标记的属性方法,都可以当用作全局样式的调用方法。

如:UIView的backgroundColor属性

@property(nullable, nonatomic,copy)            UIColor          *backgroundColor UI_APPEARANCE_SELECTOR; //

2.外观容器协议:

@protocol UIAppearanceContainer <NSObject> @end

这是个空协议,里面啥都没有

3.外观协议:

@protocol UIAppearance <NSObject>
//返回实现了此外观协议的UI控件实例,用这个返回的对象设置的属性是全局性。
+ (instancetype)appearance;
//返回实现了此外观协议的UI控件实例,用这个返回的对象设置的属性只对ContainerClass内部包裹的对象有效。
+ (instancetype)appearanceWhenContainedIn:(nullable Class <UIAppearanceContainer>)ContainerClass, ... NS_REQUIRES_NIL_TERMINATION NS_DEPRECATED_IOS(5_0, 9_0, "Use +appearanceWhenContainedInInstancesOfClasses: instead") __TVOS_PROHIBITED;
//是iOS9之后的替代方法,作用和上面的一样。
+ (instancetype)appearanceWhenContainedInInstancesOfClasses:(NSArray<Class <UIAppearanceContainer>> *)containerTypes NS_AVAILABLE_IOS(9_0);
//是iOS8新增的方法,是对不同的布局方案(紧凑型,普通型)采用不同的外观样式
+ (instancetype)appearanceForTraitCollection:(UITraitCollection *)trait NS_AVAILABLE_IOS(8_0);
//与上面的方法相似,多了一个参数条件,只对ContainerClass内部包裹的对象有效。
+ (instancetype)appearanceForTraitCollection:(UITraitCollection *)trait whenContainedIn:(nullable Class <UIAppearanceContainer>)ContainerClass, ... NS_REQUIRES_NIL_TERMINATION NS_DEPRECATED_IOS(8_0, 9_0, "Use +appearanceForTraitCollection:whenContainedInInstancesOfClasses: instead") __TVOS_PROHIBITED;
//iOS9后的新方法,是对上面方法的替换。
+ (instancetype)appearanceForTraitCollection:(UITraitCollection *)trait whenContainedInInstancesOfClasses:(NSArray<Class <UIAppearanceContainer>> *)containerTypes  NS_AVAILABLE_IOS(9_0);
@end

在上面代码中对里面定义的5个方法进行了说明。

遵守了这两个协议的类,可以进行同一默认外观设置,那么哪些类遵守了这两个协议了呢?

看一下UI控件的父类UIView

NS_CLASS_AVAILABLE_IOS(2_0) @interface UIView : UIResponder <NSCoding, UIAppearance, UIAppearanceContainer, UIDynamicItem, UITraitEnvironment, UICoordinateSpace, UIFocusItem, UIFocusItemContainer, CALayerDelegate>

通过上面UIView的定义可以发现,UIView遵守了外观协议和外观容器协议。

也就是说所有的UI控件,只有其属性方法被UI_APPEARANCE_SELECTOR宏标示了,那么就可以进行全局外观

默认设置。

这里列一下UINavigationBar通常用的的全局设置。

//??: 1与2互斥,且1的优先级高

//1.设置导航背景

UIImage *bg = [UIImage pureImageWithColor:[UIColor brownColor]];
[[UINavigationBar appearance] setBackgroundImage:bg forBarMetrics:UIBarMetricsDefault];

//2.设置导航颜色

[[UINavigationBar appearance] setBarTintColor:[UIColor redColor]];

//3.设置导航文字颜色

[[UINavigationBar appearance] setTintColor:[UIColor blueColor]];

//4.设置导航字体

NSShadow *shadow = [[NSShadow alloc] init];
shadow.shadowColor = [UIColor colorWithRed:0.0 green:0.0 blue:0.0 alpha:0.8];
shadow.shadowOffset = CGSizeMake(0, 1);
[[UINavigationBar appearance] setTitleTextAttributes: [NSDictionary dictionaryWithObjectsAndKeys: [UIColor colorWithRed:245.0/255.0 green:245.0/255.0 blue:245.0/255.0 alpha:1.0], NSForegroundColorAttributeName, shadow, NSShadowAttributeName, [UIFont fontWithName:@"HelveticaNeue-CondensedBlack" size:21.0], NSFontAttributeName, nil]];

//5.特定环境或某种场合下外观样式设置(水平紧凑型,UINavigationBar默认外观为粉色。)

[[UINavigationBar appearanceForTraitCollection:[UITraitCollection traitCollectionWithHorizontalSizeClass:UIUserInterfaceSizeClassCompact]] setBarTintColor:[UIColor purpleColor]];

另外对UIButton的全局默认设置如下:

//让一类控件同时表现出某种属性

[[UIButton appearance] setBackgroundColor:[UIColor yellowColor]];
[[UIButton appearance] setTitle:@"同一设置" forState:UIControlStateNormal];

//让一类控件在某种环境下表现出某种外观样式

[[UIButton appearanceWhenContainedInInstancesOfClasses:@[[UINavigationBar class]]] setTitle:@"比较特别" forState:UIControlStateNormal];
[[UIButton appearanceWhenContainedInInstancesOfClasses:@[[UINavigationBar class]]] setBackgroundColor:[UIColor redColor]];

UI控件的默认全局设置方式符合预期。

那么iOS开发中,所有的UI控件都可以通过上面两种方法进行全局样式默认设置和自定义设置了。

原文地址:https://www.cnblogs.com/zhou--fei/p/10381268.html

时间: 2024-12-13 23:31:33

iOS 使用UI控件的外观协议UIAppearance进行设置默认UI控件样式的相关文章

html中&lt;radio&gt;单选按钮控件标签用法解析及如何设置默认选中

<input type="radio" name="radio" value="1">单选1 <input type="radio" name="radio" value="2">单选2 <input type="radio" name="radio" value="3" checked>单选

【iOS开发-57】案例改进:block动画、控件的removeFromSuperview、利用layer设置圆角矩形以及代理模式运用

接上述案例,案例改进:[iOS开发-50]利用创建新的类实现代码封装,从而不知不觉实践一个简单的MVC实验,附带个动画 在上述案例中,我们最后实现了一个动画,点击"下载"按钮变成"已下载"不可点击,然后中间出现提示框. (1)其中有一个小BUG,就是这个提示的透明度变成0之后,这个提示框并没有显示还留在内存中.需要: [tipsLabel removeFromSuperview]; (2)其次,我们可以用另一个代码实现动画,就是用block,这一次是2个block嵌

android - 自定义(组合)控件 + 自定义控件外观

转载:http://www.cnblogs.com/bill-joy/archive/2012/04/26/2471831.html android - 自定义(组合)控件 + 自定义控件外观 Android自定义View实现很简单 继承View,重写构造函数.onDraw,(onMeasure)等函数. 如果自定义的View需要有自定义的属性,需要在values下建立attrs.xml.在其中定义你的属性. 在使用到自定义View的xml布局文件中需要加入xmlns:前缀="http://sc

iOS学习笔记1--在xcode6以上的版本中不使用storyboard以及部分控件使用

首先建立一个iOS新工程,删除工程自动建立的main.storyboard以及xib文件,并且在info.plist上删除这两个选项 然后在项目配置中将maninterface设置为空,将launch screen file设置为空,新建一个view,命名为ViewController,因为此时已经有自动生成的类ViewController,所以不用新建类 建立新建xib文件和类的连接,将file own的class修改为ViewControll,修改完成之后才能进行下一步 连接新建的xib文件

使用样式更改多个控件的外观

若要对同一类型的多个控件应用相同的属性设置,则可以创建一个样式并将该样式应用于该控件. 样式基本上就是针对特定类型的属性设置的集合. 样式通常在 XAML 中定义. 样式的 targetType 决定样式适用的控件类型. 或者,可以通过使用 x:Key 属性为样式分配一个键,当声明类型时可以按名称引用该键. 有关样式的更多信息,请参见 Style. 您可以在代码中或使用设计器来将样式分配到控件. 声明样式并将其应用于控件 在 XAML 文件中创建资源部分,如 Grid.Resources 或 S

IOS 开发指南 第5章 委托协议 数据源协议 高级视图学习

1 委托对象负责控制控件的外观和对控件的事件和状态作出反应 数据源对象是控件与应用数据(model)的桥梁,一般是必须实现的. 2 选择器 UIPickerView 为用户提供选择 1)日期选择器 UIDatePicker 设置属性检查器中的各个属性-代码 -setDateFormat:设置日期格式 -stringFromDate:获取时间 - (IBAction)onclick:(id)sender { NSDate * theDate = self.datePicker.date;获取选中日

七:理解控件的运行机制(例:基于CompositeControl命名空间的控件)

组合控件与WebControl控件的事件和属性相差不大组合控件,顾名思义就是把一些控件组合起来形成一个控件这个控件将包含这些控件称为他的子控件 CompositeControl类实现了INameContainer接口这样使得复合控件下的子控件都根据各自的层级关系生成唯一的客户端标识不至于产生重复的ID 组合控件比较重要的方法是:1.EnsureChildControls此方法判断属性ChildControlsCreated是否为true如果不是将执行下面的事件2.CreateChildContr

重新想象 Windows 8.1 Store Apps (77) - 控件增强: 文本类控件的增强, 部分控件增加了 Header 属性和 HeaderTemplate 属性, 部分控件增加了 PlaceholderText 属性

[源码下载] 重新想象 Windows 8.1 Store Apps (77) - 控件增强: 文本类控件的增强, 部分控件增加了 Header 属性和 HeaderTemplate 属性, 部分控件增加了 PlaceholderText 属性 作者:webabcd介绍重新想象 Windows 8.1 Store Apps 之控件增强 文本类控件的增强 为一些控件增加了 Header 属性和 HeaderTemplate 属性 为一些控件增加了 PlaceholderText 属性 示例1.演示

控件包含代码块(即 &lt;% ... %&gt;),因此无法修改控件集合

错误: “/”应用程序中的服务器错误. 控件包含代码块(即 <% ... %>),因此无法修改控件集合. 说明: 执行当前 Web 请求期间,出现未经处理的异常.请检查堆栈跟踪信息,以了解有关该错误以及代码中导致错误的出处的详细信息. 异常详细信息: System.Web.HttpException: 控件包含代码块(即 <% ... %>),因此无法修改控件集合. 源错误: 执行当前 Web 请求期间生成了未经处理的异常.可以使用下面的异常堆栈跟踪信息确定有关异常原因和发生位置的