很多App里都有一种点击显示的悬浮气泡菜单,例如下图:
在iPad上可以使用UIPopoverController实现这个功能,popoverController继承自NSObject而不是UIView,这是因为它本身并不能显示,popoverController的显示内容取决于成员属性contentViewController,并且该属性必须在初始化时被传入,否则会崩溃。
popover的尺寸应该由contentView决定,而不应该在外部设置,下面的代码实现了一个contentView,用于popover显示一个类似上图的菜单。
使用self.preferredContentSize属性来决定popover的尺寸。
// // TableViewController.m // iPad初步 // // Created by 11 on 8/4/15. // Copyright (c) 2015 soulghost. All rights reserved. // #import "TableViewController.h" @interface TableViewController () @property (nonatomic, strong) NSMutableArray *menuList; @end @implementation TableViewController - (void)viewDidLoad { [super viewDidLoad]; self.title = @"菜单"; CGFloat width = self.menuList.count * 44; // 过期的方法self.contentSizeForViewInPopover需要计算导航控制器的标题宽度44,在iOS8不可用。 self.preferredContentSize = CGSizeMake(self.view.bounds.size.width * 0.5, width); } - (NSMutableArray *)menuList{ if (_menuList == nil) { _menuList = [NSMutableArray arrayWithObjects:@"文件",@"编辑",@"设置",@"返回",nil]; } return _menuList; } #pragma mark - Table view data source - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { // Return the number of rows in the section. return self.menuList.count; } - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{ static NSString *ID = @"cell"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; if (cell == nil) { cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID]; } // 在这里设置cell数据 cell.textLabel.text = self.menuList[indexPath.row]; return cell; } @end创建popoverView后,在iOS7如果不主动对其进行强引用会使得popoverView在显示时被释放,因此需要指定一个成员属性防止其释放;在iOS8和以后,即使不强引用也可以,但是为了适配,应该指定这个成员属性。
下面的代码演示了一个popoverView的创建:
- (IBAction)menuClick:(id)sender { // 创建Popover的显示内容(View)。 TableViewController *tvc = [[TableViewController alloc] init]; UINavigationController *nvc = [[UINavigationController alloc] initWithRootViewController:tvc]; // 创建Popover,确定内容,确定尺寸,指定显示位置。 UIPopoverController *popover = [[UIPopoverController alloc] initWithContentViewController:nvc]; _popoverVc = popover; _popoverVc.delegate = self; // popover的尺寸由其内的视图控制器决定。 // 设置穿透蒙板的Views _popoverVc.passthroughViews = @[_segmentBtn]; // 要指向谁,rect传入谁的bounds // inView是Rect的参考系 [popover presentPopoverFromRect:_menuBtn.bounds inView:_menuBtn permittedArrowDirections:UIPopoverArrowDirectionAny animated:YES]; }注意其中的passthroughViews数组是指的能够穿过popoverView蒙板的views,因为popoverView显示时会有一个遮盖阻止用户与其他控件交互,在这个数组中的views可以穿过蒙板进行交互。
present方法传入的inView是popoverView位置的参考系,rect是显示的位置,一般有两种传递方法:①rect传入bounds而参考系为当前控件,②rect传入frame而参考系为父控件。
popoverView还有代理方法,用于阻止其dismiss或者监听销毁。
- (BOOL)popoverControllerShouldDismissPopover:(UIPopoverController *)popoverController{ return YES;// 返回NO不能销毁 } // 只有系统自己销毁才会调用,主动dismiss不会调用。 - (void)popoverControllerDidDismissPopover:(UIPopoverController *)popoverController{ NSLog(@"已经销毁"); }第一个代理方法返回NO时不能销毁,利用这个方法可以做一些判断来决定能否销毁。
第二个代理方法在系统销毁popoverView时调用,如果自己dismiss是不会调用的。