iOS开发之功能模块--用runtime给UIView类别拓展PressMenu工具
这是个很有用的列别工具类,在聊天对话框添加和QQ一样的"复制、粘贴、取消"等选项,而且使用起来很方便,只要找到聊天泡泡内部的某个View,比如Label或者背景冒泡的UIImageView,直接add...即可实现下面的效果:
直接上源码:
UIView+PressMenu.h
#import <UIKit/UIKit.h> @interface UIView (PressMenu)
@property (strong, nonatomic) NSArray *menuTitles;
@property (strong, nonatomic) UILongPressGestureRecognizer *pressGR;
@property (copy, nonatomic) void(^menuClickedBlock)(NSInteger index, NSString *title);
@property (strong, nonatomic) UIMenuController *menuVC; - (void)addPressMenuTitles:(NSArray *)menuTitles menuClickedBlock:(void(^)(NSInteger index, NSString *title))block;
- (void)showMenuTitles:(NSArray *)menuTitles menuClickedBlock:(void(^)(NSInteger index, NSString *title))block; - (BOOL)isMenuVCVisible;
- (void)removePressMenu;
@end
UIView+PressMenu.m
#import "UIView+PressMenu.h"
#import <objc/runtime.h> @implementation UIView (PressMenu) static const NSString *kPressMenuSelectorPrefix = @"easePressMenuClicked_";
static char PressMenuTitlesKey, PressMenuBlockKey, PressMenuGestureKey, MenuVCKey; #pragma mark M
- (void)addPressMenuTitles:(NSArray *)menuTitles menuClickedBlock:(void(^)(NSInteger index, NSString *title))block{
self.userInteractionEnabled = YES;
self.menuClickedBlock = block;
self.menuTitles = menuTitles;
if (self.pressGR == nil) {
self.pressGR = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handlePress:)];
}
[self addGestureRecognizer:self.pressGR];
} - (void)showMenuTitles:(NSArray *)menuTitles menuClickedBlock:(void(^)(NSInteger index, NSString *title))block{
self.menuClickedBlock = block;
self.menuTitles = menuTitles;
[self p_showMenu];
} - (BOOL)isMenuVCVisible{
if (self.menuVC) {
return [self.menuVC isMenuVisible];
}
return NO;
} - (void)removePressMenu{
if (self.menuVC) {
[self.menuVC setMenuVisible:NO animated:YES];
self.menuVC = nil;
}
if ([self.pressGR isKindOfClass:[UILongPressGestureRecognizer class]]) {
[self removeGestureRecognizer:self.pressGR];
self.pressGR = nil;
}
if (self.menuClickedBlock) {
self.menuClickedBlock = nil;
}
if (self.menuTitles) {
self.menuTitles = nil;
}
} #pragma mark SET_GET
- (void)setMenuTitles:(NSArray *)menuTitles{
objc_setAssociatedObject(self, &PressMenuTitlesKey, menuTitles, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSArray *)menuTitles{
return objc_getAssociatedObject(self, &PressMenuTitlesKey);
} - (void)setMenuClickedBlock:(void (^)(NSInteger, NSString *))menuClickedBlock{
objc_setAssociatedObject(self, &PressMenuBlockKey, menuClickedBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (void (^)(NSInteger, NSString *))menuClickedBlock{
return objc_getAssociatedObject(self, &PressMenuBlockKey);
} - (void)setPressGR:(UILongPressGestureRecognizer *)pressGR{
objc_setAssociatedObject(self, &PressMenuGestureKey, pressGR, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (UILongPressGestureRecognizer *)pressGR{
return objc_getAssociatedObject(self, &PressMenuGestureKey);
} - (void)setMenuVC:(UIMenuController *)menuVC{
objc_setAssociatedObject(self, &MenuVCKey, menuVC, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (UIMenuController *)menuVC{
return objc_getAssociatedObject(self, &MenuVCKey);
} #pragma mark canPerformAction
- (BOOL)canBecomeFirstResponder{
if (self.menuClickedBlock) {
return YES;
}else{
return [super canBecomeFirstResponder];
}
} -(BOOL)canPerformAction:(SEL)action withSender:(id)sender{
if (self.menuClickedBlock) {
for (int i=; i<self.menuTitles.count; i++) {
if (action == NSSelectorFromString([NSString stringWithFormat:@"%@%d:", kPressMenuSelectorPrefix, i])) {
return YES;
}
}
return NO;
}else{
return [super canPerformAction:action withSender:sender];
}
} -(void)handlePress:(UIGestureRecognizer*)recognizer{
if (recognizer.state == UIGestureRecognizerStateBegan) {
[self p_showMenu];
}
} - (void)p_showMenu{
[self becomeFirstResponder];
NSMutableArray *menuItems = [[NSMutableArray alloc] initWithCapacity:self.menuTitles.count];
Class cls = [self class];
SEL imp = @selector(pressMenuClicked:);
for (int i=; i<self.menuTitles.count; i++) {
NSString *title = [self.menuTitles objectAtIndex:i];
//注册名添加方法sel,sel的具体实现在imp(pressMenuClicked:)
SEL sel = sel_registerName([[NSString stringWithFormat:@"%@%d:", kPressMenuSelectorPrefix, i] UTF8String]);
class_addMethod(cls, sel, [cls instanceMethodForSelector:imp], "v@");
UIMenuItem *menuItem = [[UIMenuItem alloc] initWithTitle:title action:sel];
[menuItems addObject:menuItem];
}
UIMenuController *menu = [UIMenuController sharedMenuController];
[menu setMenuItems:menuItems];
[menu setTargetRect:self.frame inView:self.superview];
[menu setMenuVisible:YES animated:YES];
self.menuVC = menu;
} - (void)pressMenuClicked:(id)sender {
NSString *selStr = NSStringFromSelector(_cmd);
NSString *indexStr = [selStr substringFromIndex:kPressMenuSelectorPrefix.length];
NSInteger index = indexStr.integerValue;
if (index >= && index<self.menuTitles.count) {
NSString *title = [self.menuTitles objectAtIndex:index];
if (self.menuClickedBlock) {
self.menuClickedBlock(index, title);
}
}
} @end
iOS开发之功能模块--用runtime给UIView类别拓展PressMenu工具的更多相关文章
- IOS开发之功能模块--给任意的UIView添加点击事件
前言:好久没写博客,今天来一波.我在实际项目开发中,会遇到这样功能需求:我已经搭好了iOS的UI界面,但是很多界面的子View用了UIView,然后这些UIView中用了UILabel和UIImage ...
- iOS开发之功能模块--高仿Boss直聘的常用语的开发
首先上Boss直聘的功能界面截图,至于交互请读者现在Boss直聘去交互体验: 本人的公司项目要高仿Boss直聘的IM常用语的交互功能,居然花费了我前后17个小时完成,这回自己测试了很多遍,代码 ...
- IOS开发之功能模块--自定义导航控制器类常用自定义的代码
前言:本文篇幅不多,但是涉及到的内容却是开发中常用的. 涉及的内容: 1.统一设置导航控制器子控制器的返回按钮. 2.因为修改了系统的返回按钮,所以还需要设置手势事件. 3.隐藏底部的工具条. 这里直 ...
- iOS开发之功能模块--Apns推送中的的json格式介绍
在开发向苹果Apns推送消息服务功能,我们需要根据Apns接受的数据格式进行推送.下面接受我在进行apns推送时候总结的一点apns服务接受的Json数据格式 示例 1: 以下负载包含哦一个简单的 a ...
- iOS开发之功能模块--根据需求开发横向的子弹盒View
这个需求是本人工作开发中后期需求要添加的新功能,本人模仿UITableView的代理和数据源方法进行了第一阶段的开发.第二阶段是添加丰富的动画. 这个功能需求描述:能上传添加五个待选头像,五个头像分别 ...
- iOS开发之功能模块--推送之坑问题解决
不管想不想看我后面再开发中总结的经验,但是很值得推荐一位大神写的关于苹果推送,很多内容哦:http://www.cnblogs.com/qiqibo/category/408304.html 苹果开发 ...
- iOS开发之功能模块--高仿Boss直聘的IM界面交互功能
本人公司项目属于社交类,高仿Boss直聘早期的版本,现在Boss直聘界面风格,交互风格都不如Boss直聘以前版本的好看. 本人通过iPhone模拟器和本人真机对聊,将完成的交互功能通过Mac截屏模拟器 ...
- iOS开发之功能模块--本地序列化
下面只展示项目开发中,本地序列化的示例代码: AuthenticationManager.h #import <Foundation/Foundation.h> #import " ...
- iOS开发之功能模块--计算高度Demo探究手稿
本篇记录关于计算文本高度和Label高度的代码,以备后期再探究: 首先是YouXianMing老师的工具类别: NSString+LabelWidthAndHeight.h // // NSStrin ...
随机推荐
- C# 如何将List拆分成多个子集合
网上的例子: 问:List<string> list = new List<string>(); for (int i = 1; i < 95; i++) ...
- 基于keepalived搭建MySQL的高可用集群
MySQL的高可用方案一般有如下几种: keepalived+双主,MHA,MMM,Heartbeat+DRBD,PXC,Galera Cluster 比较常用的是keepalived+双主,MHA和 ...
- MySQL中有关TIMESTAMP和DATETIME的总结
一.MySQL中如何表示当前时间? 其实,表达方式还是蛮多的,汇总如下: CURRENT_TIMESTAMP CURRENT_TIMESTAMP() NOW() LOCALTIME LOCALTIME ...
- Google Chrome 应用商店上传扩展程序
1. 进入 Google 开发者中心 网址: https://chrome.google.com/webstore/developer/dashboard?hl=zh-cn 2. 添加新项 上传后,出 ...
- spring控制并发数的工具类ConcurrencyThrottleSupport和ConcurrencyThrottleInterceptor
官方文档: /** * Support class for throttling concurrent access to a specific resource. * * <p>Desi ...
- 软件工程-构建之法 理解C#一小段程序
一.前言 老师给出的要求: 阅读下面程序,请回答如下问题: 问题1:这个程序要找的是符合什么条件的数? 问题2:这样的数存在么?符合这一条件的最小的数是什么? 问题3:在电脑上运行这一程序,你估计多长 ...
- Django admin 权威指南(一)
版本: Django 1.10 此部分由官方文档<6.5.1 The Django admin site>翻译而来. 6.5.1.1 概览 默认情况下,使用startproject的时候, ...
- 10.Configure One-to-Many(配置一对多关系)【Code-First系列】
现在,我们将学习怎么配置一对多的关系. Visit Entity Relationship section to understand how EF manages one-to-one, one-t ...
- 【原创】kafka server源代码分析(二)
十四.AbstractFetcherManager.scala 该scala定义了两个case类和一个抽象类.两个case类很简单: 1. BrokerAndFectherId:封装了一个broker ...
- 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间
[源码下载] 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间 作者:webabcd 介绍速战速决 之 PHP 动态地创 ...