(1)效果
(2)源代码与第三方类库下载
http://download.csdn.net/detail/wsb200514/8155979
(3)总结
——导航控制器,可以直接用代码的push和pop来控制控制器之间的跳转。也可以使用storyboard的segue来:这里又涉及2种,一种是直接用按钮拖拽到另一个控制器形成segue,这种segue不可拦截,如果点击直接跳转。另一种是从一个控制器拖拽到另一个控制器形成的segue,这种segue没有明确的点击谁来跳转,所以有一个performSegueWithIdentifier方法,一执行这个方法就跳转,所以一般用在需要判断的跳转上面,比如“登录”的按钮,添加addTarget之后,在按钮的点击方法中执行perform跳转。
——最重要的是数据的传递,正向传递的方法,主要是利用segue的属性destinationViewController获得后面一个控制器,然后给后面这个控制器赋值,可以赋值某一个地方,也可以通过一个数据模型对象把整个数据模型赋值给后面的控制器,前提是这前后两个控制器都要有这个数据模型属性。而且这个赋值操作,一般是在prepareForSegue中进行,因为这是在segue跳转之前调用的一个方法。
——还有反向传递,即点击“返回”时,值需要从当前页面传递到“返回”后的那个页面。这里需要用到代理,即在当前页面设置一个协议,然后设置一个代理属性以及代理方法(主要用来传递数据),这个代理方法的参数可以是某一个值也可以是一个数据模型对象。然后在“返回”后的那个页面上,遵守协议并且实现代理方法,接受到数据,再把数据处理一下。
——我们这里有用到tableView,所以,涉及到添加和删除cell。我们还涉及到数据永久花存储,本案例中我们利用NSKeyedArchive方法把数据存储为data格式。所以我们每次删除和添加数据后,不仅需要“刷新”tableView,还要把数据归档一遍。
——我们懒加载数组的时候,也要多一个判断:如果数组是空,我们就从数据文件中读取数据用NSKeyedUnarchive,然后再判断一下,如果读取了数据发现还是空就初始化创建这个数据。这里面的NSKeyedUnchive和上面的NSKeyedArchive都需要数据模型对象遵守NSCoding协议并实现init和encode方法,就是归档和解归档数据的方式,这里如果有子类的话,只需要继承父类,然后实现子类自己的即可。
——第三方类库的使用,可以直接把.m和.h文件拖拽到工程里面即可,当然还有其他安装方法。我们这里用到的是MBProgressHUD,下载和使用方法见:https://github.com/jdg/MBProgressHUD
——这里面监听键盘的值改变,我们用的是通知,记住:用到通知的时候,必须在dealloc函数里面移除通知,养成好习惯:
- (void)viewDidLoad { [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(changeText) name:UITextFieldTextDidChangeNotification object:self.userField]; [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(changeText) name:UITextFieldTextDidChangeNotification object:self.pwdField]; [super viewDidLoad]; // Do any additional setup after loading the view. } -(void)dealloc{ [[NSNotificationCenter defaultCenter]removeObserver:self]; }
——在storyboard里面的导航栏的右边不能添加2个item,所以只能用代码实现,即先把右边那个取出来,然后创建一个,最后把这2个装成数组赋值给items。
//在storyboard里面无法添加两个右边按钮,只能代码 UIBarButtonItem *addItem=self.navigationItem.rightBarButtonItem; UIBarButtonItem *deleteItem=[[UIBarButtonItem alloc]initWithTitle:@"操作" style:UIBarButtonItemStylePlain target:self action:@selector(deleteClick)]; [email protected][addItem,deleteItem];
——tableView有一个编辑操作,即,一进入编辑操作,就类似于“短信删除界面”的效果,在所有cell左边都出现一个“减号”删除按钮一样。默认是删除,但可以修改成添加样式。以下代码就是实现tableView点击一下可编辑再点击不可编辑的效果。
-(void)deleteClick{ self.tableView.editing=!self.tableView.isEditing; }
——如下方法就是设置每一个cell左边出来的是”减号“的删除,还是”加号“的添加。我们这一行是添加一行是删除。
-(UITableViewCellEditingStyle)tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath{ return indexPath.row%2 ? UITableViewCellEditingStyleDelete : UITableViewCellEditingStyleInsert; }
——点击注销按钮后出现的是UIActionSheet,而没有用UIAlert,并且为了监听点击的时sheet上得哪个按钮,这个控制器还遵守了协议UIActionSheetDelegate,并实现了判断:如果是点击”确定“,那么就退出,返回登录界面,也就是需要在导航控制器上实现pop操作。
- (IBAction)logoutClick:(id)sender { UIActionSheet *sheet=[[UIActionSheet alloc]initWithTitle:@"确定要注销嘛?" delegate:self cancelButtonTitle:@"取消" destructiveButtonTitle:@"确定" otherButtonTitles:nil, nil]; [sheet showInView:self.view]; }
- (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex{ if (buttonIndex==0) { [self.navigationController popViewControllerAnimated:YES]; } }
——在tableView的那个控制器上,因为有2种跳转,所以需要isKindOfClass判断一下是跳转到哪个页面。这里面else if里面就是实现了一个数据的正向传递,直接赋值即可。
-(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender{ id vc=segue.destinationViewController; if ([vc isKindOfClass:[WPAddViewController class]]) { WPAddViewController *addVc=vc; addVc.delegate=self; }else if ([vc isKindOfClass:[WPEditViewController class]]){ WPEditViewController *editVc=segue.destinationViewController; NSIndexPath *path=[self.tableView indexPathForSelectedRow]; editVc.contact=self.contacts[path.row]; editVc.delegate=self; } }
——每当cell左边出现”减号“或”加号“的时候,我们一点击,就会调用下面这个方法。所以在这里我们可以判断到底点击的是什么按钮,然后执行操作,如果是减号删除,我们就删除数据、更新表格、归档,如果是添加数据,我们也是添加数据、更新表格、归档。只不过,我们这里的更新表格,并不是用reload更新表格所有的cell,而是用tableView的delete和insert方法实现部分更新。
//删除数据或添加数据 -(void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath{ if (editingStyle==UITableViewCellEditingStyleDelete) { //1、先更新模型 [self.contacts removeObjectAtIndex:indexPath.row]; //2、再刷新表格(建议使用第一种只刷新删除行以下的那些cell,第二种刷新全部的表格,耗性能) [self.tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationTop]; //[self.tableView reloadData]; [NSKeyedArchiver archiveRootObject:self.contacts toFile:filePath]; }else if (editingStyle==UITableViewCellEditingStyleInsert){ WPContact *p2=[[WPContact alloc]init]; [email protected]"jack"; [email protected]"10086"; [self.contacts insertObject:p2 atIndex:indexPath.row+1]; NSIndexPath *indexPathP2=[NSIndexPath indexPathForRow:indexPath.row+1 inSection:0]; [self.tableView insertRowsAtIndexPaths:@[indexPathP2] withRowAnimation:UITableViewRowAnimationBottom]; [NSKeyedArchiver archiveRootObject:self.contacts toFile:filePath]; } }
——因为要实现数据归档,所以在每一次更改数据(即这里存放数据的数组)后,都需要重新归档一次。