码迷,mamicode.com
首页 > 移动开发 > 详细

ios开发级联菜单(利用父子控制器--两个菜单封装为两个子控制器来实现)

时间:2016-08-03 14:53:28      阅读:343      评论:0      收藏:0      [点我收藏+]

标签:

一:1:级联菜单可以使用两个tableView来实现,也可以利用父子控制器,两个控制器来实现,根视图控制器作为两个控制器的父控制器,来管理两个子控制器。2:将左右菜单分别交给两个控制器去管理,对于一些复杂的业务逻辑,涉及大量回调操作,业务逻辑也要相对复杂,则不建议采取封装成view去处理,最好还是利用一个控制器去管理其内部复杂的业务逻辑,具体做法就是:利用父子控制器,将子控制器交由父控制器去管理,将子控制器的view添加到父控制器的view上。效果如图:

技术分享

 

二:代码

1:根控制器代码:添加两个子控制器到,并将子view添加到根视图控制器的view上

 1 #import "ViewController.h"
 2 #import "RHCategoryViewController.h"
 3 #import "RHSubCategoryViewController.h"
 4 @interface ViewController ()
 5 
 6 @end
 7 
 8 @implementation ViewController
 9 
10 - (void)viewDidLoad {
11     [super viewDidLoad];
12     
13     
14     CGFloat width = self.view.frame.size.width *0.5;
15     CGFloat height = self.view.frame.size.height;
16     //右侧菜单
17     RHSubCategoryViewController *sub = [[RHSubCategoryViewController alloc]init];
18     sub.view.frame = CGRectMake(width, 0, width, height);
19     [self addChildViewController:sub];
20     [self.view addSubview:sub.view];
21     
22     //左侧菜单
23     RHCategoryViewController *category = [[RHCategoryViewController alloc]init];
24     category.view.frame = CGRectMake(0, 0, width, height);
25     category.delegate = sub;
26     [self.view addSubview:category.view];
27     [self addChildViewController:category];
28 
29    
30     
31    
32 }
33 
34 
35 @end

2:左侧菜单控制器

 1 //左侧菜单:
 2 
 3 #import <UIKit/UIKit.h>
 4 @class RHCategoryViewController;
 5 @protocol RHCategoryViewControllerDelegate <NSObject>
 6 
 7 @optional
 8 
 9 - (void)categoryViewController:(RHCategoryViewController*)controller didSelectRowWithData:(NSArray*)dataArr;
10 
11 @end
12 
13 @interface RHCategoryViewController : UITableViewController
14 
15 @property (nonatomic,weak)id <RHCategoryViewControllerDelegate>delegate;
16 
17 @end
  1 #import "RHCategoryViewController.h"
  2 #import "RHCategoryModel.h"
  3 static NSString *const cellID = @"cellID";
  4 @interface RHCategoryViewController ()
  5 /*数据源*/
  6 @property (nonatomic ,strong)NSMutableArray *dataArr;;
  7 
  8 @end
  9 
 10 @implementation RHCategoryViewController
 11 
 12 - (NSMutableArray*)dataArr {
 13     
 14     if (!_dataArr) {
 15         NSString *filePath = [[NSBundle mainBundle]pathForResource:@"categories" ofType:@"plist"];
 16         NSArray *data = [NSArray arrayWithContentsOfFile:filePath];
 17         NSMutableArray *dataArray = [NSMutableArray array];
 18         for (NSDictionary *dic in data) {
 19             RHCategoryModel *model = [RHCategoryModel categoryModelWithDic:dic];
 20             [dataArray addObject:model];
 21         }
 22         /*
 23          1:此时的数组可以自己初始化,或是将其他的数组赋值给没初始化的数组 2:在懒加载还可以调用set方法为数组赋值,也就是,self.dataArr = dataArray;
 24          */
 25         _dataArr = dataArray;
 26     }
 27     
 28     return _dataArr;
 29 }
 30 
 31 - (void)viewDidLoad {
 32     [super viewDidLoad];
 33     //1:注册表格
 34     [self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:cellID];
 35     
 36     /*
 37      注: 1:写在viewDidLoad此处不会默认选中第一行,应该在viewWillAppear:(BOOL)animated,或是viewDidAppear:(BOOL)animated方法中设置默认选中第一行时,才会选中
 38          2:控制器的view是懒加载的,当在根控制器中设置view的frame时,vc.view.frame,vc.view调用的是vc中view的get方法,此时会加载view,执行viewDidload方法,加载view,但是此时view的UI,tableView还没有加载,先调用viewWillAppear:(BOOL)animated,在调用数据源代理方法,在调用viewDidAppear:(BOOL)animated
 39      //2:默认第0行被选中
 40      //     [self.tableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:NO scrollPosition:UITableViewScrollPositionTop];
 41      */
 42    
 43 }
 44 
 45 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
 46     
 47     return self.dataArr.count;
 48 }
 49 
 50 - (UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
 51     /*
 52      1:当cell被选中的时候,cell上的子控件都会显示高亮行为,所以可以设置cell上子控件高亮时的状态,来显示cell被选中时的状态。 cell.textLabel.highlightedTextColor,cell.imageView.highlightedImage
 53      */
 54     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];
 55     RHCategoryModel *model = self.dataArr[indexPath.row];
 56     cell.textLabel.text = model.name;
 57     cell.textLabel.highlightedTextColor = [UIColor orangeColor];
 58     cell.imageView.image = [UIImage imageNamed:model.icon];
 59     cell.imageView.highlightedImage = [UIImage imageNamed:model.highlighted_icon];
 60     cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
 61     return cell;
 62 }
 63 
 64 - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
 65     
 66     if (self.delegate && [self.delegate respondsToSelector:@selector(categoryViewController:didSelectRowWithData:)]) {
 67         RHCategoryModel *model = self.dataArr[indexPath.row];
 68         [self.delegate categoryViewController:self didSelectRowWithData:model.subcategories];
 69       
 70     }
 71 }
 72 
 73 - (void)setDelegate:(id<RHCategoryViewControllerDelegate>)delegate {
 74     
 75     _delegate = delegate;
 76     /*
 77      1:默认表格的某一行被选中,不会调用didSelectRowAtIndexPath方法,只是设置cell的选中的一个状态
 78      2:获得tableView选中的某一行的indexpath:self.tableView.indexPathForSelectedRow
 79      */
 80     [self tableView:self.tableView didSelectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0]];
 81     
 82 }
 83 
 84 /*
 85  1:两方法分别是:视图即将出现的时候调用,视图已经出现的时候调用
 86  2:当某个控制器被压入栈里,或是被moda的控制器遮住的时候,返回到此控制器,则两个方法依然会被调用
 87  */
 88 - (void)viewWillAppear:(BOOL)animated {
 89     
 90     [super viewWillAppear:animated];
 91 //     [self.tableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:NO scrollPosition:UITableViewScrollPositionTop];
 92 }
 93 
 94 - (void)viewDidAppear:(BOOL)animated {
 95     
 96     [super viewDidAppear:animated];
 97    
 98     [self.tableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:NO scrollPosition:UITableViewScrollPositionTop];
 99     
100 }
101 @end

3:右侧菜单控制器

1 #import <UIKit/UIKit.h>
2 #import "RHCategoryViewController.h"
3 @interface RHSubCategoryViewController : UITableViewController<RHCategoryViewControllerDelegate>
4 
5 @end
 1 #import "RHSubCategoryViewController.h"
 2 #import "RHCategoryViewController.h"
 3 static NSString *const cellID = @"cellID";
 4 @interface RHSubCategoryViewController ()
 5 /*数据源*/
 6 @property (nonatomic ,strong)NSArray *dataArr;
 7 @end
 8 
 9 @implementation RHSubCategoryViewController
10 
11 - (void)viewDidLoad {
12     [super viewDidLoad];
13     [self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:cellID];
14 }
15 
16 - (void)categoryViewController:(RHCategoryViewController *)controller didSelectRowWithData:(NSArray *)dataArr {
17     
18     self.dataArr = dataArr;
19     [self.tableView reloadData];
20 }
21 
22 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
23     
24     return self.dataArr.count;
25 }
26 
27 - (UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
28     
29     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];
30     cell.textLabel.text = self.dataArr[indexPath.row];
31     return cell;
32 }
33 
34 
35 @end

5:数据模型model

 1 #import <Foundation/Foundation.h>
 2 
 3 @interface RHCategoryModel : NSObject
 4 /*高亮图片*/
 5 @property (nonatomic,copy)NSString *highlighted_icon;
 6 /*icon*/
 7 @property (nonatomic,copy)NSString *icon;
 8 /*名字*/
 9 @property (nonatomic,copy)NSString *name;
10 /*分类数组*/
11 @property (nonatomic ,strong)NSArray *subcategories;
12 
13 +(instancetype)categoryModelWithDic:(NSDictionary*)dic;
14 @end
 1 #import "RHCategoryModel.h"
 2 
 3 @implementation RHCategoryModel
 4 
 5 +(instancetype)categoryModelWithDic:(NSDictionary*)dic {
 6     
 7     RHCategoryModel *model = [[self alloc]init];
 8     [model setValuesForKeysWithDictionary:dic];
 9     return model;
10 }
11 @end

 

三:知识点总结

1:从plist中读取数据:1:先加载plist的路径: NSString *filePath = [[NSBundle mainBundle]pathForResource:@"categories" ofType:@"plist"]; 2:在查看pilist文件中根节点是数组还是字典,从而将plist数据读出: NSArray *data = [NSArray arrayWithContentsOfFile:filePath];

 2:懒加载:属性修饰必须用strong,若用weak则刚创建完对象就会被销毁。在懒加载中,懒加载数组或是其他变量时,1:若没初始化,可以将一个已经初始化的变量直接赋值 2:若已经初始化,则可以返回 3:只要是属性定义的变量,就会生成下划线的变量,set方法,get方法,所以在懒加载时,除了可以用带下划线的成员变量,也可以用self.data = dataArray;左侧调用的是set方法,右侧调用的是get方法 4:若是系统或是自定的属性,调用完set方法赋值后,若想在其他方法中获得所赋的值,则可直接调用其get方法,例如titleLable.font,从父控制器数组中取出子控制器,直接调用title的get方法,直接获得title,不用再讲title放在数组中再去赋值。5:若外部向获得在.m中声明的成员属性,则该变量可以再.h中提供自身的get方法供外部调用

3:1:[self.tableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:NO scrollPosition:UITableViewScrollPositionTop];tableView的默认选中某一行 1:默认表格的某一行被选中,不会调用didSelectRowAtIndexPath方法,只是设置cell的选中的一个状态  2:获得tableView选中的某一行的indexpath:self.tableView.indexPathForSelectedRow 2:1:写在viewDidLoad此处不会默认选中第一行,应该在viewWillAppear:(BOOL)animated,或是viewDidAppear:(BOOL)animated方法中设置默认选中第一行时,才会选中 2:控制器的view是懒加载的,当在根控制器中设置view的frame时,vc.view.frame,vc.view调用的是vc中view的get方法,此时会加载view,执行viewDidload方法,加载view完毕,但是此时view的UI视图并没有去加载,tableView还没有加载,先调用viewWillAppear:(BOOL)animated,在去绘制视图UI,完毕后在调用viewDidAppear:(BOOL)animated。3:只有当整个表格绘制完毕或是即将绘制时,设置selectRowAtIndexPath才会起作用

4:当cell被选中的时候,cell上的子控件都会显示高亮行为,所以可以设置cell上子控件高亮时的状态,来显示cell被选中时的状态。 cell.textLabel.highlightedTextColor,cell.imageView.highlightedImage

5:重写setDelegate方法:因为在viewDidload里如果调用didselecrow方法,默认选中,则不会执行,是因为此时还没有设置代理,代理为空,self.delegate对象为空值,所以重写setDelegate方法,在设置代理成功后,再调用didselecrow,来设置默认选中

6:- (void)viewWillAppear:(BOOL)animated ,- (void)viewDidAppear:(BOOL)animated。 1:两方法分别是:视图即将出现的时候调用,视图已经出现的时候调用  2:当某个控制器被压入栈里,或是被moda的控制器遮住的时候,返回到此控制器,则两个方法依然会被调用

7: [self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:cellID];self.tableView注册完cell后,就不用在其数据源方法中判断cell是否存在了,直接从缓存池中取出cell,就可以了

8:用类方法快速获得数据模型的对象:+(instancetype)categoryModelWithDic:(NSDictionary*)dic;在实现方法中,创建出对象后,利用对象调用kvc方法快速为属性赋值,并将模型对象返回。self在类方法中指的是当前类,在对象方法中指的是当前的对象,所以在类方法中不能用self调用对象方法

 

ios开发级联菜单(利用父子控制器--两个菜单封装为两个子控制器来实现)

标签:

原文地址:http://www.cnblogs.com/cqb-learner/p/5732561.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!