iOS自定义UIBarButtonItem的target和action示例代码

需求描述:

在项目开发过程中,遇到一种情况,需要自定义UIBarButtonItem,来实现分享样式,并在iPad中弹出系统分享框(UIActivityViewController),系统分享框需要指定显示位置(barButtonItem)。而自定义的UIBarButtonItem target指向的是UIButton。这与需求不符,需自定义UIBarButtonItem。

在介绍自定义UIBarButtonItem前,先介绍一下相关控件的子父类关系(也可以说继承关系)。

1、UIBarItem

NS_CLASS_AVAILABLE_IOS(2_0) @interface UIBarItem : NSObject <NSCoding, UIAppearance>

2、UIBarButtonItem

NS_CLASS_AVAILABLE_IOS(2_0) @interface UIBarButtonItem : UIBarItem <NSCoding>

3、UITabBarItem

NS_CLASS_AVAILABLE_IOS(2_0) @interface UITabBarItem : UIBarItem

下面是在界面上的显示效果

UIBarButtonItem和UITabBarItem效果显示

从上图中看到UIBarButtonItem有三种效果显示,分别是

1、导航左侧返回按钮,UINavigationItem中的backBarButtonItem属性

@property(nullable,nonatomic,strong) UIBarButtonItem *backBarButtonItem

2、纯文本的UIBarButtonItem

- (instancetype)initWithTitle:(nullable NSString *)title style:(UIBarButtonItemStyle)style target:(nullable id)target action:(nullable SEL)action;

3、纯图片的UIBarButtonItem,其中包括自定义图片和系统样式

- (instancetype)initWithImage:(nullable UIImage *)image style:(UIBarButtonItemStyle)style target:(nullable id)target action:(nullable SEL)action;
- (instancetype)initWithBarButtonSystemItem:(UIBarButtonSystemItem)systemItem target:(nullable id)target action:(nullable SEL)action;

UIToolBar使用UIBarButtonItem与导航效果一致。

关于UITabBarItem在这里就不多介绍,只是拿其显示效果与UIBarButtonItem对比。

在开发过程中,我们会使用到自定义UIBarButtonItem,来显示我们想要的界面效果。使用的方法常为:

- (instancetype)initWithCustomView:(UIView *)customView;
- (void)viewDidLoad {
 [super viewDidLoad];
 //自定义View
 UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0.0, 0.0, 60.0, 40.0)];
 view.backgroundColor = [UIColor redColor];
 //自定义按钮
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 btn.frame = view.bounds;
 [btn addTarget:self action:@selector(clickRight:) forControlEvents:UIControlEventTouchUpInside];
 [view addSubview:btn];
 //自定义Item
 UIBarButtonItem *barItem = [[UIBarButtonItem alloc] initWithCustomView:view];
 //
 self.navigationItem.leftBarButtonItem = barItem;
}

#pragma mark -

- (void)clickRight:(id)sender {
 NSLog(@"sender:%@",sender);
}

其中打印sender,其类型是UIButton。

2017-10-17 16:08:43.917 TestImage[5482:163865] sender:<UIButton: 0x7fb9bad12e60; frame = (0 0; 60 40); opaque = NO; layer = <CALayer: 0x61000003b940>>

通过上面描述,发现系统方法不能实现项目需求效果。当然也可以通过属性保存UIBarButtonItem方法来实现需求效果。即在点击按钮响应后,直接使用保存的UIBarButtonItem,但是我没有采用这种方法。

下面是我给出的两种解决方案:

方案一

继承UIBarButtonItem,实现子类。

定义子类

#import <UIKit/UIKit.h>

@interface LLBarButtonItem : UIBarButtonItem

@end
#import "LLBarButtonItem.h"

@implementation LLBarButtonItem

- (id)initWithCustomView:(UIView *)customView {
 self = [super initWithCustomView:customView];
 if (self) {
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 btn.frame = customView.bounds;
 btn.backgroundColor = [UIColor clearColor];
 [btn addTarget:self action:@selector(clickButton:) forControlEvents:UIControlEventTouchUpInside];
 [customView addSubview:btn];
 }
 return self;
}

- (void)clickButton:(UIButton *)sender {
 if (self.target && [self.target respondsToSelector:self.action]) {
 //[self.target performSelector:self.action withObject:self];
 IMP imp = [self.target methodForSelector:self.action];

 void (*func)(id, SEL, id) = (void *)imp;

 func(self.target, self.action, self);
 }
}

@end

定义子类对象,调用子类对象

- (void)viewDidLoad {
 [super viewDidLoad];
 //自定义View
 UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0.0, 0.0, 60.0, 40.0)];
 view.backgroundColor = [UIColor clearColor];
 //自定义Item
 LLBarButtonItem *barItem = [[LLBarButtonItem alloc] initWithCustomView:view];
 barItem.target = self;
 barItem.action = @selector(clickRight:);
 //
 self.navigationItem.leftBarButtonItem = barItem;
}

#pragma mark -

- (void)clickRight:(id)sender {
 NSLog(@"sender:%@",sender);
}

打印target对象

2017-10-17 16:24:11.696 TestImage[5557:170144] sender:<LLBarButtonItem: 0x7fb403c16080>

方案二

UIBarButtonItem类别

定义类别

#import <UIKit/UIKit.h>

@interface UIBarButtonItem (Custom)

- (void)addCutomTarget:(id)target action:(SEL)action;

@end
#import "UIBarButtonItem+Custom.h"

@implementation UIBarButtonItem (Custom)

- (void)addCutomTarget:(id)target action:(SEL)action {
 if (self.customView != nil) {
 self.target = target;
 self.action = action;
 //
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 btn.frame = self.customView.bounds;
 btn.backgroundColor = [UIColor clearColor];
 [btn addTarget:self action:@selector(clickButton:) forControlEvents:UIControlEventTouchUpInside];
 [self.customView addSubview:btn];
 }
}

- (void)clickButton:(UIButton *)sender {
 if (self.target && [self.target respondsToSelector:self.action]) {
 //[self.target performSelector:self.action withObject:self];
 IMP imp = [self.target methodForSelector:self.action];

 void (*func)(id, SEL, id) = (void *)imp;

 func(self.target, self.action, self);
 }
}

@end

调用类别方法

- (void)viewDidLoad {
 [super viewDidLoad];
 //自定义View
 UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0.0, 0.0, 60.0, 40.0)];
 view.backgroundColor = [UIColor clearColor];
 //自定义Item
 UIBarButtonItem *barItem = [[UIBarButtonItem alloc] initWithCustomView:view];
 [barItem addCutomTarget:self action:@selector(clickRight:)];
 //
 self.navigationItem.leftBarButtonItem = barItem;
}

#pragma mark -

- (void)clickRight:(id)sender {
 NSLog(@"sender:%@",sender);
}

打印target对象

2017-10-17 16:28:14.407 TestImage[5598:172418] sender:<UIBarButtonItem: 0x7ffeda609e20>

两种方法都使用了IMP做消息传递。

你更喜欢哪一种?!

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • 详解iOS应用中自定义UIBarButtonItem导航按钮的创建方法

    iOS系统导航栏中有leftBarButtonItem和rightBarButtonItem,我们可以根据自己的需求来自定义这两个UIBarButtonItem. 四种创建方法 系统提供了四种创建的方法: 复制代码 代码如下: - (instancetype)initWithBarButtonSystemItem:(UIBarButtonSystemItem)systemItem target:(id)target action:(SEL)action; - (instancetype)init

  • iOS应用开发中导航栏按钮UIBarButtonItem的添加教程

    1.UINavigationController导航控制器如何使用 UINavigationController可以翻译为导航控制器,在iOS里经常用到. 我们看看它的如何使用: 下面的图显示了导航控制器的流程.最左侧是根视图,当用户点击其中的General项时 ,General视图会滑入屏幕:当用户继续点击Auto-Lock项时,Auto-Lock视图将滑入屏幕.相应地,在对象管理上,导航控制器使用了导航堆栈.根视图控制器在堆栈最底层,接下来入栈的是General视图控制器和Auto-Lock

  • iOS如何改变UIBarButtonItem的大小详解

    前言 基本上每个iOS APP里面都有导航,比如微信.QQ.支付宝.导航可以很方便地帮助我们管理视图控制器(UIViewController).导航的重要性不言而喻,基本上是每一位iOS初学者都要接触到的问题. iOS系统导航栏中有leftBarButtonItem和rightBarButtonItem,我们可以根据自己的需求来自定义这两个UIBarButtonItem. 本文主要介绍的是关于iOS改变UIBarButtonItem大小的相关内容,下面话不多说了,来一起看看详细的介绍吧 实现方法

  • iOS自定义UIBarButtonItem的target和action示例代码

    需求描述: 在项目开发过程中,遇到一种情况,需要自定义UIBarButtonItem,来实现分享样式,并在iPad中弹出系统分享框(UIActivityViewController),系统分享框需要指定显示位置(barButtonItem).而自定义的UIBarButtonItem target指向的是UIButton.这与需求不符,需自定义UIBarButtonItem. 在介绍自定义UIBarButtonItem前,先介绍一下相关控件的子父类关系(也可以说继承关系). 1.UIBarItem

  • SpringBoot通过自定义注解实现日志打印的示例代码

    前言 在我们日常的开发过程中通过打印详细的日志信息能够帮助我们很好地去发现开发过程中可能出现的Bug,特别是在开发Controller层的接口时,我们一般会打印出Request请求参数和Response响应结果,但是如果这些打印日志的代码相对而言还是比较重复的,那么我们可以通过什么样的方式来简化日志打印的代码呢? SpringBoot 通过自定义注解实现权限检查可参考我的博客:SpringBoot 通过自定义注解实现权限检查 正文 Spring AOP Spring AOP 即面向切面,是对OO

  • IOS获取系统相册中照片的示例代码

    先来看看效果图 下面话不多少,我们直接上代码: #import "ViewController.h" @interface ViewController ()<UINavigationControllerDelegate,UIImagePickerControllerDelegate> @property (weak, nonatomic) IBOutlet UIImageView *IconView; @end @implementation ViewController

  • jquery自定义插件——window的实现【示例代码】

    本例子实现弹窗的效果: 1.jquery.show.js /* * 实现功能:点击在鼠标位置显示div * 版本序号:1.0 */ (function($){ $.fn.showDIV = function(options){ var defaults = {}; var options = $.extend(defaults, options); var showdiv=$(this); var close, title, content; close=$(" "); title=$

  • iOS整个APP实现灰色主题的示例代码

    灰色主题 背景 在一些哀悼日,清明节的时候app会实现一些灰色主题功能,部分app需求是tab首页实现灰色模式就可以,但一些需求是直接整个app都变为灰色模. 普通UI界面 web页面 xib界面 attributeText加载的htmlString页面 attachment挂件页面 实现方式 基本的实现方式1,普通页面用hook颜色方式2.web页面用注入灰色js实现方式 图片变灰 重新绘制图片变为灰色代码实现 //image类别 - (UIImage *)getGrayImage { con

  • C#实现自定义打印文字和图片的示例代码

    目录 1.调用打印机设置 2.关联文档 3.绘制内容 C#中打印其实就是自己绘图+调用系统打印函数,于是便有了以下操作 1.调用打印机设置 如果你想在打印前设置打印机属性(或者切换打印机),请务必添加这段代码,否则电脑会直接按照预设的设置进行打印(打印机都没法选) //打印机设置 PrintDialog printDialog = new PrintDialog(); printDialog.Document = ptDoc; printDialog.ShowDialog(); ptDoc就是打

  • iOS实现实时检测网络状态的示例代码

    前言 在网络应用中,需要对用户设备的网络状态进行实时监控,有两个目的: (1)让用户了解自己的网络状态,防止一些误会(比如怪应用无能) (2)根据用户的网络状态进行智能处理,节省用户流量,提高用户体验 WIFI\3G网络:自动下载高清图片 低速网络:只下载缩略图 没有网络:只显示离线的缓存数据 最近在工作中遇到一个功能就是根据用户当前的网络状,用户未联网需要提示一下,如果是Wifi可以推荐一些图片新闻,如果是3G模式设置为无图的模式,获取网络状态比较简单,毕竟中国现在的流量还是一个比较贵的状态,

  • iOS屏幕旋转与锁屏的示例代码

    在做视频开发时遇到屏幕旋转问题,其中涉及到 StatusBar. UINavigationController.UITabBarController .UIViewcontroller . 在设备锁屏下的整体效果图 iOS-旋转.gif 主要涉及以下4点: 横竖屏的旋转 屏幕旋转相应改变视图位置 旋转时状态栏的隐藏与显示 锁屏 1.横竖屏旋转 第1步: -(UIInterfaceOrientationMask)application:(UIApplication *)application su

  • iOS如何封装带复制功能的UILabel示例代码

    前言 UILabel继承自UIView是iOS中使用非常频繁的一个视图控件一般用于显示文字. 一:基本使用 1.创建 UILabel *label = [[UILabel alloc]initWithFrame:CGRectMake(20, 64, 100, 30)]; [self.view addSubview:label]; 2.属性设置 在iOS中你想要使用一个属性一般就直接"."属性英文名称,或者"set"属性英文名称一般就可以出现 label.backgr

  • ios原生和react-native各种交互的示例代码

    需求:让一个表格视图中的cell能左滑删除,效果图如下: 目前RN中的ListView主要问题是复用,以及其他一些细节如索引视图.左滑删除.编辑等,要想在RN上自定义实现原生的这种效果尚有一定的问题,在必要时可以考虑使用原生的UITableView,数据从RN端传递 1.原生端编写表格控制器NativeTableViewController,暴露的属性如下 datas为表格数据源,另外一个为需要暴露给RN调用用方法. 2.框架只提供了暴露UIView给RN端的接口,所以需要制作一个中转UIVie

随机推荐