改代码是参考一个Demo直接改的,代码中有一些漏洞,如果发现其他的问题,可以下方直接留言

.h文件

#import <UIKit/UIKit.h>
typedef void(^PopoverBlock)(NSInteger index);
@interface CustomPopView : UIView
//@property(nonatomic,copy)void(^block)(int index);
//-(void)setDataArr:(NSArray *)titleArr withView:(id *)view withBlock:(void(^)(NSString *a))block;
@property (nonatomic, copy) NSArray *menuTitles;
@property(nonatomic,copy)void(^PopoverHiddenBlock)(BOOL isHidden );
- (void)showFromView:(id)aView selected:(PopoverBlock)selected;
@end @interface PopoverArrow : UIView @end

.m文件

#import "CustomPopView.h"
// 字体大小
#define kPopoverFontSize 14.f // 十六进制颜色
#define UIColorFromRGB(rgbValue) [UIColor colorWithRed:((float)((rgbValue & 0xFF0000) >> 16))/255.0 green:((float)((rgbValue & 0xFF00) >> 8))/255.0 blue:((float)(rgbValue & 0xFF))/255.0 alpha:1.0] #define SCREEN_W [UIScreen mainScreen].bounds.size.width
#define SCREEN_H [UIScreen mainScreen].bounds.size.height // 箭头高度
#define kArrowH 8
// 箭头宽度
#define kArrowW 15
//每行的高度
#define CELL_HEIGHT 38
//
#define Identifier @"cell" // 边框颜色
#define kBorderColor UIColorFromRGB(0xE1E2E3)
@interface CustomPopView () <UITableViewDelegate, UITableViewDataSource>
{
PopoverBlock _selectedBlock;
UIView *_backgroundView;
PopoverArrow *_arrowView;
} @property (nonatomic, retain) UITableView *tableView; @end @implementation CustomPopView
- (instancetype)initWithFrame:(CGRect)frame
{
if (!(self = [super initWithFrame:frame])) return nil; self.backgroundColor = [UIColor clearColor]; // 箭头
_arrowView = [PopoverArrow new];
[self addSubview:_arrowView]; // tableView放在箭头底下, 用于箭头挡住tableView边框
[self insertSubview:self.tableView belowSubview:_arrowView];
return self;
} - (void)layoutSubviews
{
[super layoutSubviews]; // 设置tableView默认的分割线起终点位置
if ([self.tableView respondsToSelector:@selector(setSeparatorInset:)]) {
[self.tableView setSeparatorInset:UIEdgeInsetsZero];
}
if ([self.tableView respondsToSelector:@selector(setLayoutMargins:)]) {
[self.tableView setLayoutMargins:UIEdgeInsetsZero];
} self.tableView.layer.cornerRadius = 5.f;
self.tableView.layer.borderColor = kBorderColor.CGColor;
self.tableView.layer.borderWidth = 1.f;
} #pragma mark -- getter - (UITableView *)tableView
{
if (_tableView) return _tableView; _tableView = [UITableView new]; _tableView.delegate = self;
_tableView.dataSource = self;
_tableView.rowHeight = CELL_HEIGHT;
_tableView.backgroundColor = [UIColor whiteColor];
_tableView.showsVerticalScrollIndicator = NO;
[_tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:Identifier];
_tableView.tableFooterView = UIView.new; return _tableView;
} #pragma mark -- delegate - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return self.menuTitles.count;
} - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:Identifier];
cell.textLabel.font = [UIFont systemFontOfSize:kPopoverFontSize];
cell.textLabel.text = [self.menuTitles objectAtIndex:indexPath.row];
return cell;
} - (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath
{
if ([cell respondsToSelector:@selector(setSeparatorInset:)]) {
[cell setSeparatorInset:UIEdgeInsetsZero];
}
if ([cell respondsToSelector:@selector(setLayoutMargins:)]) {
[cell setLayoutMargins:UIEdgeInsetsZero];
}
} - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
[UIView animateWithDuration:0.25 animations:^{
self.alpha = 0;
} completion:^(BOOL finished) {
[_backgroundView removeFromSuperview];
_backgroundView = nil; if (_selectedBlock) {
_selectedBlock(indexPath.row);
} [self removeFromSuperview];
}];
} #pragma mark -- private
// 点击透明层隐藏
- (void)hide
{
[UIView animateWithDuration:0.25 animations:^{
self.alpha = 0;
} completion:^(BOOL finished) {
[_backgroundView removeFromSuperview];
_backgroundView = nil;
if(self.PopoverHiddenBlock){
self.PopoverHiddenBlock(YES);
}
[self removeFromSuperview];
}];
} #pragma mark -- public /*!
* @author lifution
*
* @brief 显示弹窗
*
* @param aView 箭头指向的控件
* @param selected 选择完成回调
*/
- (void)showFromView:(id)aView selected:(PopoverBlock)selected
{
if (selected) _selectedBlock = selected;
//aView只能传两种参数,一种是UIView 另一种UIBarButtonItem
if(!([aView isKindOfClass:[UIView class]] || [aView isKindOfClass:[UIBarButtonItem class]])){
return;
}
UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
// 背景遮挡
_backgroundView = UIView.new;
_backgroundView.frame = keyWindow.bounds;
_backgroundView.backgroundColor = [UIColor blackColor];
_backgroundView.alpha = 0.2;
_backgroundView.userInteractionEnabled = YES;
[_backgroundView addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(hide)]];
[keyWindow addSubview:_backgroundView];
// 刷新数据更新contentSize
[self.tableView reloadData]; // 获取触发弹窗的按钮在window中的坐标
CGRect triggerRect ;
if([aView isKindOfClass:[UIView class]]){
UIView *view = (UIView *)aView;
triggerRect = [view convertRect:view.bounds toView:keyWindow];
}else{
UIView *bgView = [aView valueForKey:@"_view"];
triggerRect = [bgView convertRect:bgView.bounds toView: keyWindow];
}
// 箭头指向的中心点 CGFloat arrowCenterX = CGRectGetMaxX(triggerRect)-CGRectGetWidth(triggerRect)/2; // 取得标题中的最大宽度
CGFloat maxWidth = 0;
for (id obj in self.menuTitles) {
if ([obj isKindOfClass:[NSString class]]) {
CGSize titleSize = [obj sizeWithAttributes:@{NSFontAttributeName:[UIFont systemFontOfSize:kPopoverFontSize]}];
if (titleSize.width > maxWidth) {
maxWidth = titleSize.width;
}
}
} CGFloat curWidth = ((maxWidth+80)>SCREEN_W-30)?SCREEN_W-30:(maxWidth+80);
CGFloat curHeight = CELL_HEIGHT*self.menuTitles.count+kArrowH;
CGFloat curX = arrowCenterX-curWidth/2;
CGFloat curY = CGRectGetMaxY(triggerRect)+10; // 如果箭头指向点距离屏幕右边减去5px不足curWidth的一半的话就重新设置curX
if ((SCREEN_W-arrowCenterX-5)<curWidth/2) {
curX = curX-(curWidth/2-(SCREEN_W-arrowCenterX-5));
}
// 如果箭头指向点距离屏幕左边加上5px不足curWidth的一半的话就重新设置curX
if (arrowCenterX+5<curWidth/2) {
curX = curX+(curWidth/2-arrowCenterX)+5;
} //如果高度大于10行,则最高按10计算
if(curHeight>CELL_HEIGHT*10+kArrowH){
curHeight = CELL_HEIGHT*10+kArrowH;
} self.frame = CGRectMake(curX, curY - 18, curWidth, curHeight);
_arrowView.frame = CGRectMake(arrowCenterX-curX-kArrowW/2, 0, kArrowW, kArrowH+1);
// 箭头高度 +1 遮挡住tableView的边框
self.tableView.frame = CGRectMake(0, kArrowH, curWidth,curHeight - kArrowH );
[keyWindow addSubview:self]; self.alpha = 0;
[UIView animateWithDuration:0.3 animations:^{
self.alpha = 1;
}];
} @end // 箭头
@implementation PopoverArrow - (instancetype)initWithFrame:(CGRect)frame
{
if (!(self = [super initWithFrame:frame])) return nil; self.backgroundColor = [UIColor clearColor]; return self;
} // Only override drawRect: if you perform custom drawing.
// An empty implementation adversely affects performance during animation.
- (void)drawRect:(CGRect)rect {
[super drawRect:rect];
// Drawing code
CGSize curSize = rect.size;
CGContextRef context = UIGraphicsGetCurrentContext();
CGContextSetLineWidth(context, 1);
CGContextSetStrokeColorWithColor(context, kBorderColor.CGColor);
CGContextSetFillColorWithColor(context, UIColor.whiteColor.CGColor);
CGContextBeginPath(context);
CGContextMoveToPoint(context, 0, curSize.height);
CGContextAddLineToPoint(context, curSize.width/2, 0);
CGContextAddLineToPoint(context, curSize.width, curSize.height);
CGContextDrawPath(context, kCGPathFillStroke);
} @end

使用:

view = [CustomPopView new];
view.menuTitles = @[@"1",@"2",@"3"];
UIBarButtonItem *item = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemAdd target:self action:@selector(addBtnClick:)];
self.navigationItem.rightBarButtonItem = item; -(void)addBtnClick:(UIBarButtonItem *)item{
[view showFromView:item selected:^(NSInteger index) { }];
}

自定义PopView的更多相关文章

  1. iOS实现自定义的弹出视图(popView)

    前段时间,在项目中有个需求是支付完成后,弹出红包,实现这么一个发红包的功能.做了最后,实现的效果大致如下: 一.使用方法 整个ViewController的代码大致如下 // //  SecondVi ...

  2. Android定位&地图&导航——自定义公交路线代码

    一.问题描述 基于百度地图实现检索指定城市指定公交的交通路线图,效果如图所示 二.通用组件Application类,主要创建并初始化BMapManager public class App exten ...

  3. Android自定义遮罩层设计

    在做网页设计时,前端设计人员会经常用到基于JS开发的遮罩层,并且背景半透明.这样的效果怎么样在Android上实现呢?这个实现并不困难,先来上效果图: <ignore_js_op> 201 ...

  4. 关于Unity3D自定义编辑器的学习

    被人物编辑器折腾了一个月,最终还是交了点成品上去(还要很多优化都还么做).  刚接手这项工作时觉得没概念,没想法,不知道.后来就去看<<Unity5.X从入门到精通>>中有关于 ...

  5. 一起学微软Power BI系列-使用技巧(5)自定义PowerBI时间日期表

    1.日期函数表作用 经常使用Excel或者PowerBI,Power Pivot做报表,时间日期是一个重要的纬度,加上做一些钻取,时间日期函数表不可避免.所以今天就给大家分享一个自定义的做日期表的方法 ...

  6. JavaScript自定义浏览器滚动条兼容IE、 火狐和chrome

    今天为大家分享一下我自己制作的浏览器滚动条,我们知道用css来自定义滚动条也是挺好的方式,css虽然能够改变chrome浏览器的滚动条样式可以自定义,css也能够改变IE浏览器滚动条的颜色.但是css ...

  7. ASP.NET Aries 入门开发教程8:树型列表及自定义右键菜单

    前言: 前面几篇重点都在讲普通列表的相关操作. 本篇主要讲树型列表的操作. 框架在设计时,已经把树型列表和普通列表全面统一了操作,用法几乎是一致的. 下面介绍一些差距化的内容: 1:树型列表绑定: v ...

  8. ASP.NET Aries 入门开发教程5:自定义列表页工具栏区

    前言: 抓紧时间,继续写教程,因为发现用户期待的内容,都在业务处理那一块. 不得不继续勤劳了. 这节主要介绍工具栏区的玩法. 工具栏的默认介绍: 工具栏默认包括5个按钮,根据不同的权限决定显示: 添加 ...

  9. UWP中实现自定义标题栏

    UWP中实现自定义标题栏 0x00 起因 在UWP开发中,有时候我们希望实现自定义标题栏,例如在标题栏中加入搜索框.按钮之类的控件.搜了下资料居然在一个日文网站找到了一篇介绍这个主题的文章: http ...

随机推荐

  1. docker中,如何将镜像保存为tar文件或者将镜像保存为文件,将tar文件导入到docker中

    需求说明: 在实际使用docker的过程中,比如某些机器不能上网,恰巧需要某些docker镜像,一般可以采用在能上网的机器上,pull下来基础镜像,然后将docker 镜像保存为文件,在不能上网的机器 ...

  2. ActiveMQ JMS 项目 基于 Maven 搭建 部署

    JAVA版本: IntellJ IDEA 版本: IntelliJ IDEA 2017.2Build #IU-172.3317.76, built on July 15, 2017Licensed t ...

  3. python-docx 设置标题heading的中文字体类型+设置正文的中文字体类型

    依赖包: from docx import Document from docx.shared import Pt from docx.shared import Inches from docx.o ...

  4. 使用Python3.7.0搭建简易服务器

    一.下载并安装Python 官网地址 二.新建start_server.bat 在需要搭建服务器的根目录位置,新建一个start_server.bat文件 内容 python -m http.serv ...

  5. MVC的ViewData自动给Razor写的input赋值

    问题: 写编辑的时候,突然发现,没有值的model,突然出现了值,而且值是ViewData中值. 后台: this.ViewData["test"] = "测试" ...

  6. 正则表达式中,[\s\S]* 什么意思

    https://blog.csdn.net/haoyuedangkong_fei/article/details/53781936 例如:[a-z]表示从a到z之间的任意一个. 不是这样的吗?谁能给我 ...

  7. 使用 systemctl 创建 ss 开机

    有自启动脚本.可以设置开机自启. 下载python 安装 ss就不说了.使用 systemctl 创建ss开机自启服务. 创建配置文件 vim /usr/lib/systemd/system/shad ...

  8. 一、在windows环境下修改pip镜像源的方法(以python3为例)

    在windows环境下修改pip镜像源的方法(以python3为例) 1.在windows文件管理器中,输入 %APPDATA% 2.会定位到一个新的目录下,在该目录下新建pip文件夹,然后到pip文 ...

  9. day7 七、字符编码,字符字节与文件操作

    一.字符编码 1.定义:人类能识别的是字符等高级标识符,电脑只能识别0,1组成的标识符,要完成人与机器之间的信息交流,一定需要一个媒介,进行两种标识符的转化(两种标识符的对应关系) 对应关系形成的结构 ...

  10. 160道Java技术面试题

    1.面向对象的特征有哪些方面?2.访问修饰符public,private,protected,以及不写(默认)时的区别?3.String 是最基本的数据类型吗?4.float f=3.4;是否正确?5 ...