iOS 三级下拉菜单功能实现

前言

App 常用控件 -- 多级下拉菜单, 如团购类, 房屋类, 对数据进行筛选. 有一级, 二级, 三级, 再多就不会以这种样式,呈现给用户了. 作者就简单聊一下 多级下拉菜单


一 目标

  1. 默认显示一个 TableView, 点击数据后, 添加第二个TableView, 并实现大小变化
  2. 第二次打开下拉菜单. 保存上次选中数据

二 菜单控件DropMenuView

.h文件

#import <UIKit/UIKit.h>

@class DropMenuView;

@protocol DropMenuViewDelegate <NSObject>

-(void)dropMenuView:(DropMenuView *)view didSelectName:(NSString *)str;

@end

@interface DropMenuView : UIView

@property (nonatomic, weak) id<DropMenuViewDelegate> delegate;

/** 箭头变化 */

@property (nonatomic, span) UIView *arrowView;

/**

控件设置

@param view 提供控件 位置信息

@param tableNum 显示TableView数量

@param arr 使用数据

*/

-(void)creatDropView:(UIView *)view withShowTableNum:(NSInteger)tableNum withData:(NSArray *)arr;

/** 视图消失 */

- (void)dismiss;

@end

.m文件

#import "DropMenuView.h"

#define kWidth [UIScreen mainScreen].bounds.size.width

#define kHeight [UIScreen mainScreen].bounds.size.height

@interface DropMenuView ()<UITableViewDelegate, UITableViewDataSource>

{

@private

/** 保存 选择的数据(行数) */

NSInteger selects[3];

}

@property (nonatomic, assign) BOOL show; // 按钮点击后 视图显示/隐藏

@property (nonatomic, assign) CGFloat rowHeightNum; // 设置 rom 高度

/* 底层取消按钮 */

@property (nonatomic, span) UIButton *cancelButton;

/** 表视图数组 */

@property (nonatomic, span) NSArray *tableViewArr;

/** 表视图的 底部视图 */

@property (nonatomic, span) UIView *tableViewUnderView;

/** 显示 TableView 数量 */

@property (nonatomic, assign) NSInteger tableCount;

/** 数据 */

@property (nonatomic, span) NSArray *dataArr;

@end

@implementation DropMenuView

- (instancetype)init

{

self = [super init];

if (self) {

/** 数据初始化 */

self.dataArr = [NSArray array];

/** 保存 初始值为-1 */

for (int i = 0; i < 3; i++) {

selects[i] = -1;

}

/* 底层取消按钮 */

self.cancelButton = [UIButton buttonWithType:UIButtonTypeCustom];

self.cancelButton.backgroundColor = [UIColor colorWithWhite:0.0 alpha:0.3];

[self.cancelButton addTarget:self action:@selector(clickCancelButton:) forControlEvents:UIControlEventTouchUpInside];

[self addSubview:self.cancelButton];

/** 表视图的 底部视图初始化 */

self.tableViewUnderView = [[UIView alloc] init];

self.tableViewUnderView.backgroundColor = [UIColor colorWithRed:0.74 green:0.73 blue:0.76 alpha:1.000];

[self.cancelButton addSubview:self.tableViewUnderView];

/** 默认设置为no, row高度为40 */

self.show = NO;

self.rowHeightNum = 40.0f;

}

return self;

}

-(void)creatDropView:(UIView *)view withShowTableNum:(NSInteger)tableNum withData:(NSArray *)arr{

if (!self.show) {

self.show = !self.show;

// 显示 TableView数量

self.tableCount = tableNum;

// 数据

self.dataArr = arr;

for (UITableView *tableView in self.tableViewArr) {

[tableView reloadData];

}

// 初始位置 设置

CGFloat x = 0.f;

CGFloat y = view.frame.origin.y + view.frame.size.height;

CGFloat w = kWidth;

CGFloat h = kHeight - y;

self.frame = CGRectMake(x, y, w, h);

self.cancelButton.frame = CGRectMake(0, 0, self.frame.size.width, self.frame.size.height);

self.tableViewUnderView.frame = CGRectMake(0, 0, self.frame.size.width, self.rowHeightNum * 7);

if (!self.superview) {

[[[UIApplication sharedApplication] keyWindow] addSubview:self];

self.alpha = 0.0f;

[UIView animateWithDuration:0.2f animations:^{

self.alpha = 1.0f;

}];

[self loadSelects];

[self adjustTableViews];

}

}else{

/** 什么也不选择时候, 再次点击按钮 消失视图 */

[self dismiss];

}

}

#pragma mark - 加载选中的TableView

-(void)loadSelects{

[self.tableViewArr enumerateObjectsUsingBlock:^(UITableView *tableView, NSUInteger idx, BOOL * _Nonnull stop) {

// 刷新TableView数据

[tableView reloadData];

// 选中TableView某一行

[tableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:selects[idx] inSection:0] animated:NO scrollPosition:UITableViewScrollPositionNone];

// 加 !idx 是因为 循环第一次 idx == 0 方法不执行, 所以需要循环一次 加载一个TableView.

if((selects[idx] != -1 && !tableView.superview) || !idx) {

[self.tableViewUnderView addSubview:tableView];

[UIView animateWithDuration:0.2 animations:^{

if (self.arrowView) {

self.arrowView.transform = CGAffineTransformMakeRotation(M_PI);

}

}];

}

}];

}

#pragma mark - 重置TableView的 位置

-(void)adjustTableViews{

// 显示的 TableView 数量

int addTableCount = 0;

for (UITableView *tableView in self.tableViewArr) {

if (tableView.superview) {

addTableCount++;

}

}

for (int i = 0; i < addTableCount; i++) {

UITableView *tableView = self.tableViewArr[i];

CGRect adjustFrame = tableView.frame;

adjustFrame.size.width = kWidth / addTableCount ;

adjustFrame.origin.x = adjustFrame.size.width * i + 0.5 * i;

adjustFrame.size.height = self.tableViewUnderView.frame.size.height ;

tableView.frame = adjustFrame;

}

}

#pragma mark - TableView协议

/** 行数 */

-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{

NSInteger __block count;

[self.tableViewArr enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {

if (obj == tableView) {

NSInteger firstSelectRow = ((UITableView *)self.tableViewArr[0]).indexPathForSelectedRow.row ;

NSInteger secondSelectRow = ((UITableView *)self.tableViewArr[1]).indexPathForSelectedRow.row ;

count = [self countForChooseTable:idx firstTableSelectRow:firstSelectRow withSecondTableSelectRow:secondSelectRow];

}

}];

return count;

}

// 可以将 方法提出来, 如果有需要 可以设置为协议实现封装, 作者仅提取一个, 其他均在 TableView自身协议中写

-(NSInteger)countForChooseTable:(NSInteger)idx firstTableSelectRow:(NSInteger)firstSelectRow withSecondTableSelectRow:(NSInteger)secondSelectRow{

if (idx == 0) {

return self.dataArr.count;

}else if (idx == 1){

if (firstSelectRow == -1) {

return 0;

}else{

if (self.tableCount == 2) {

return [self.dataArr[firstSelectRow][@"subcategories"] count];

}else{

return [self.dataArr[firstSelectRow][@"sub"] count];

}

}

}else{

if (secondSelectRow == -1) {

return 0;

}else{

return [self.dataArr[firstSelectRow][@"sub"][secondSelectRow][@"sub"] count];

}

}

}

/** 自定义cell */

-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{

UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"DropCell"];

cell.textLabel.font = [UIFont systemFontOfSize:14];

if (self.tableCount == 1) {

cell.textLabel.text = self.dataArr[indexPath.row][@"label"];

}else if (self.tableCount == 2){

NSInteger firstSelectRow = ((UITableView *)self.tableViewArr[0]).indexPathForSelectedRow.row;

if (tableView == self.tableViewArr[0]) {

cell.textLabel.text = self.dataArr[indexPath.row][@"name"];

}else if (tableView == self.tableViewArr[1]){

cell.textLabel.text = self.dataArr[firstSelectRow][@"subcategories"][indexPath.row];

}

}else if (self.tableCount == 3){

NSInteger firstSelectRow = ((UITableView *)self.tableViewArr[0]).indexPathForSelectedRow.row;

NSInteger secondSelectRow = ((UITableView *)self.tableViewArr[1]).indexPathForSelectedRow.row;

if (tableView == self.tableViewArr[0]) {

cell.textLabel.text = self.dataArr[indexPath.row][@"name"];

}else if (tableView == self.tableViewArr[1]){

cell.textLabel.text = self.dataArr[firstSelectRow][@"sub"][indexPath.row][@"name"];

}else if (tableView == self.tableViewArr[2]){

cell.textLabel.text = self.dataArr[firstSelectRow][@"sub"][secondSelectRow][@"sub"][indexPath.row];

}

}

return cell;

}

/** 点击 */

-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{

UITableView *secondTableView = self.tableViewArr[1];

UITableView *thirdTableView = self.tableViewArr[2];

if (self.tableCount == 1) {

[self saveSelects];

[self dismiss];

[_delegate dropMenuView:self didSelectName:self.dataArr[indexPath.row][@"label"]];

}else if (self.tableCount == 2){

if (tableView == self.tableViewArr[0]) {

if (!secondTableView.superview) {

[self.tableViewUnderView addSubview:secondTableView];

}

[secondTableView reloadData];

[self adjustTableViews];

}else if (tableView == self.tableViewArr[1]){

[self saveSelects];

[self dismiss];

NSInteger firstSelectRow = ((UITableView *)self.tableViewArr[0]).indexPathForSelectedRow.row;

[_delegate dropMenuView:self didSelectName:self.dataArr[firstSelectRow][@"subcategories"][indexPath.row]];

}

}else if (self.tableCount == 3){

NSInteger firstSelectRow = ((UITableView *)self.tableViewArr[0]).indexPathForSelectedRow.row;

NSInteger secondSelectRow = ((UITableView *)self.tableViewArr[1]).indexPathForSelectedRow.row;

if (tableView == self.tableViewArr[0]) {

if (!secondTableView.superview) {

[self.tableViewUnderView addSubview:secondTableView];

}

[self adjustTableViews];

[secondTableView reloadData];

}else if (tableView == self.tableViewArr[1]){

if (!thirdTableView.superview) {

[self.tableViewUnderView addSubview:thirdTableView];

}

[self adjustTableViews];

[thirdTableView reloadData];

}else if (tableView == self.tableViewArr[2]){

[self saveSelects];

[self dismiss];

[_delegate dropMenuView:self didSelectName:self.dataArr[firstSelectRow][@"sub"][secondSelectRow][@"sub"][indexPath.row]];

}

}

}

#pragma mark - 记录 选择状态

-(void)saveSelects{

[self.tableViewArr enumerateObjectsUsingBlock:^(UITableView *tableView, NSUInteger idx, BOOL * _Nonnull stop) {

selects[idx] = tableView.superview ? tableView.indexPathForSelectedRow.row : -1;

}];

}

#pragma mark - 视图消失

- (void)dismiss{

if(self.superview) {

self.show = !self.show;

[self endEditing:YES];

[UIView animateWithDuration:.25f animations:^{

self.alpha = .0f;

} completion:^(BOOL finished) {

[self.tableViewUnderView.subviews enumerateObjectsUsingBlock:^(UIView *obj, NSUInteger idx, BOOL *stop) {

[obj removeFromSuperview];

}];

[self removeFromSuperview];

[UIView animateWithDuration:0.2 animations:^{

if (self.arrowView) {

self.arrowView.transform = CGAffineTransformMakeRotation(0);

}

}];

}];

}

}

/** 底部按钮, 视图消失 */

-(void)clickCancelButton:(UIButton *)button{

[self dismiss];

}

/** 懒加载 */

-(NSArray *)tableViewArr{

if (_tableViewArr == nil) {

_tableViewArr = @[[[UITableView alloc] init], [[UITableView alloc] init], [[UITableView alloc] init]];

for (UITableView *tableView in _tableViewArr) {

[tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"DropCell"];

tableView.delegate = self;

tableView.dataSource = self;

tableView.frame = CGRectMake(0, 0, 0, 0);

tableView.backgroundColor = [UIColor whiteColor];

tableView.tableFooterView = [[UIView alloc] init];

tableView.showsVerticalScrollIndicator = NO;

tableView.rowHeight = self.rowHeightNum;

}

}

return _tableViewArr;

}

@end

三 调用控件MenuScreeningView

.h文件

#import <UIKit/UIKit.h>

@interface MenuScreeningView : UIView

#pragma mark - 筛选菜单消失

-(void)menuScreeningViewDismiss;

@end

.m文件

#import "MenuScreeningView.h"

#import "DropMenuView.h"

#define kWidth [UIScreen mainScreen].bounds.size.width

#define kHeight [UIScreen mainScreen].bounds.size.height

@interface MenuScreeningView ()<DropMenuViewDelegate>

@property (nonatomic, span) UIButton *oneLinkageButton;

@property (nonatomic, span) UIButton *twoLinkageButton;

@property (nonatomic, span) UIButton *threeLinkageButton;

@property (nonatomic, span) DropMenuView *oneLinkageDropMenu;

@property (nonatomic, span) DropMenuView *twoLinkageDropMenu;

@property (nonatomic, span) DropMenuView *threeLinkageDropMenu;

@property (nonatomic, span) NSArray *addressArr;

@property (nonatomic, span) NSArray *categoriesArr;

@property (nonatomic, span) NSArray *sortsArr;

@end

@implementation MenuScreeningView

- (instancetype)initWithFrame:(CGRect)frame

{

self = [super initWithFrame:frame];

if (self) {

self.oneLinkageButton = [UIButton buttonWithType:UIButtonTypeCustom];

self.oneLinkageButton.frame = CGRectMake(0, 0, kWidth/3, 36);

[self setUpButton:self.oneLinkageButton withText:@"一级"];

self.oneLinkageDropMenu = [[DropMenuView alloc] init];

self.oneLinkageDropMenu.arrowView = self.oneLinkageButton.imageView;

self.oneLinkageDropMenu.delegate = self;

self.twoLinkageButton = [UIButton buttonWithType:UIButtonTypeCustom];

self.twoLinkageButton.frame = CGRectMake(kWidth/3, 0, kWidth/3, 36);

[self setUpButton:self.twoLinkageButton withText:@"二级"];

self.twoLinkageDropMenu = [[DropMenuView alloc] init];

self.twoLinkageDropMenu.arrowView = self.twoLinkageButton.imageView;

self.twoLinkageDropMenu.delegate = self;

self.threeLinkageButton = [UIButton buttonWithType:UIButtonTypeCustom];

self.threeLinkageButton.frame = CGRectMake(2 * kWidth/3, 0, kWidth/3, 36);

[self setUpButton:self.threeLinkageButton withText:@"三级"];

self.threeLinkageDropMenu = [[DropMenuView alloc] init];

self.threeLinkageDropMenu.arrowView = self.threeLinkageButton.imageView;

self.threeLinkageDropMenu.delegate = self;

/** 最下面横线 */

UIView *horizontalLine = [[UIView alloc] initWithFrame:CGRectMake(0, self.frame.size.height - 0.6, kWidth, 0.6)];

horizontalLine.backgroundColor = [UIColor colorWithWhite:0.8 alpha:1.000];

[self addSubview:horizontalLine];

}

return self;

}

#pragma mark - 按钮点击推出菜单 (并且其他的菜单收起)

-(void)clickButton:(UIButton *)button{

if (button == self.oneLinkageButton) {

[self.twoLinkageDropMenu dismiss];

[self.threeLinkageDropMenu dismiss];

[self.oneLinkageDropMenu creatDropView:self withShowTableNum:1 withData:self.sortsArr];

}else if (button == self.twoLinkageButton){

[self.oneLinkageDropMenu dismiss];

[self.threeLinkageDropMenu dismiss];

[self.twoLinkageDropMenu creatDropView:self withShowTableNum:2 withData:self.categoriesArr];

}else if (button == self.threeLinkageButton){

[self.oneLinkageDropMenu dismiss];

[self.twoLinkageDropMenu dismiss];

[self.threeLinkageDropMenu creatDropView:self withShowTableNum:3 withData:self.addressArr];

}

}

#pragma mark - 筛选菜单消失

-(void)menuScreeningViewDismiss{

[self.oneLinkageDropMenu dismiss];

[self.twoLinkageDropMenu dismiss];

[self.threeLinkageDropMenu dismiss];

}

#pragma mark - 协议实现

-(void)dropMenuView:(DropMenuView *)view didSelectName:(NSString *)str{

if (view == self.oneLinkageDropMenu) {

[self.oneLinkageButton setTitle:str forState:UIControlStateNormal];

[self buttonEdgeInsets:self.oneLinkageButton];

}else if (view == self.twoLinkageDropMenu){

[self.twoLinkageButton setTitle:str forState:UIControlStateNormal];

[self buttonEdgeInsets:self.twoLinkageButton];

}else if (view == self.threeLinkageDropMenu){

[self.threeLinkageButton setTitle:str forState:UIControlStateNormal];

[self buttonEdgeInsets:self.threeLinkageButton];

}

}

#pragma mark - 设置Button

-(void)setUpButton:(UIButton *)button withText:(NSString *)str{

[button addTarget:self action:@selector(clickButton:) forControlEvents:UIControlEventTouchUpInside];

[self addSubview:button];

[button setTitle:str forState:UIControlStateNormal];

button.titleLabel.font = [UIFont systemFontOfSize:11];

button.titleLabel.lineBreakMode = NSLineBreakByTruncatingTail;

[button setTitleColor:[UIColor colorWithWhite:0.3 alpha:1.000] forState:UIControlStateNormal];

[button setImage:[UIImage imageNamed:@"downarr"] forState:UIControlStateNormal];

[self buttonEdgeInsets:button];

UIView *verticalLine = [[UIView alloc]init];

verticalLine.backgroundColor = [UIColor colorWithWhite:0.9 alpha:1.0];

[button addSubview:verticalLine];

verticalLine.frame = CGRectMake(button.frame.size.width - 0.5, 3, 0.5, 30);

}

-(void)buttonEdgeInsets:(UIButton *)button{

[button setTitleEdgeInsets:UIEdgeInsetsMake(0, -button.imageView.bounds.size.width + 2, 0, button.imageView.bounds.size.width + 10)];

[button setImageEdgeInsets:UIEdgeInsetsMake(0, button.titleLabel.bounds.size.width + 10, 0, -button.titleLabel.bounds.size.width + 2)];

}

#pragma mark - 懒加载

-(NSArray *)addressArr{

if (_addressArr == nil) {

NSDictionary *dic = [NSDictionary dictionaryWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"address.plist" ofType:nil]];

_addressArr = dic[@"address"];

}

return _addressArr;

}

-(NSArray *)categoriesArr{

if (_categoriesArr == nil) {

_categoriesArr = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"categories.plist" ofType:nil]];

}

return _categoriesArr;

}

-(NSArray *)sortsArr{

if (_sortsArr == nil) {

_sortsArr = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"sorts.plist" ofType:nil]];

}

return _sortsArr;

}

@end

四 调用

MenuScreeningView *menuScreening = [[MenuScreeningView alloc] initWithFrame:CGRectMake(0, 64, kWidth, 36)];

[self.view addSubview:menuScreening];

menuScreening.backgroundColor = [UIColor whiteColor];

五 效果图


六 demo下载

因为数据源 无法上次上传[简书], 所以上传个demo, 细节方面, 可能有未注意地方,仅供参考.

传送门 : LinkageMenu_jb51.rar

以上是 iOS 三级下拉菜单功能实现 的全部内容, 来源链接: utcz.com/z/337128.html

回到顶部