简介
承上两篇我的iOS入门第一课,我的iOS入门第二课,本次记录的是一年前写的MyLocations,当时是用swift,前几天用OC重写了一遍,感触良多。主要功能是记录个人走过的地方,加以描述和照片在手机里留下美好的记忆。如下:
下面主要讲我学到了什么:
一、基本控件
UITableView,MKMapView
二、框架
CoreLocation、CoreData、MapKit、CoreGraphics、CoreAnimation、AudioToolbox
重点知识:
- 本app尝试使用Autoresizing做布局
- UITabBarController管理的体系
- TabBar第一个界面主要做一件事,获取用户GPS坐标并转换成地址
使用CoreLocation获取地址- CLLocationManager 获取GPS坐标(经纬度)通过代理通知
- startUpdatingLocation 调用此方法得到、更新坐标
- desiredAccuracy 坐标精确度 (精确到十米:kCLLocationAccuracyNearestTenMeters)
CLLocationManagerDelegate 代理方法 - locationManager:didFailWithError (error -> CLError)
- kCLErrorLocationUnknown 没找着,但CL仍继续在找
- kCLErrorDenied 没开定位服务
- kCLErrorNetwork 网络错误
- locationManager:didUpdateLocations 此方法会调用多次,每次得到的location坐标都会更精确。
关于CLLocation:- timestamp location创建的时间
- horizontalAccuracy 精确度,小于0则代表invalid,不准的
- 在获取坐标前必须获得用户权限,使用定位
CLAuthorizationStatus authStatus = [CLLocationManager authorizationStatus];
if (authStatus == kCLAuthorizationStatusNotDetermined) {
[_locationManager requestWhenInUseAuthorization];
return;
}
if (authStatus == kCLAuthorizationStatusDenied || authStatus == kCLAuthorizationStatusRestricted) {
[self showLocationServicesDeniedAlert];
return;
}
并在info.plist中加上
key:NSLocationWhenInUseUsageDescription
type:String
value:This app lets you keep track of interesting places. It needs access to the GPS coordinates for your location.
-
CLLocation得到的是坐标,最终还得用CLGeocoder
转换成地址 -> CLPlacemark- geocoder reverseGeocodeLocation:completionHandler 得到CLPlacemark (按以下顺序写地址)
- subThoroughfare
- thoroughfare
- locality
- administrativeArea
- postalCode
- geocoder reverseGeocodeLocation:completionHandler 得到CLPlacemark (按以下顺序写地址)
-
第一个界面得到的地址希望被描述并存储起来,则进入到Location detail界面。关于Core Data:
- 新建DataModel,生成一个.xcdatamodeld的文件
- 新建实体,增加属性,类型是其它类型则选Transformable,根据环境选择是否是optional(右边third tab)
- 从Core Data中取出的对象都是NSManagedObject
我们通常继承NSManagedObject实现自己的子类。Editor → Create NSManagedObject Subclass - 选择要生成的实体。 生成两种文件,当数据实体发生改变时只需要再次Create NSManagedObject Subclass,只会改变CoreDataProperties这个分类文件。在另一钟文件中写模型方法。
- 在CoreDataProperties分类文件中把已知的属性类型修正过来(id -> CLPlaceMark)
- 若一开始没有用Core Data,则新建一个使用Core Data的项目,把在AppDelegate中两个文件多出来的Core Data代码全部复制到本项目中。
我们只需要知道这些代码做了两件事- 加载数据模型
- 连接SQLite数据库
具体来说就是:
- 找到bundle中先前创建的“DataModel.momd”路径,转化成URL
- 根据URL创建NSManagedObjectModel,这个对象代表所有数据模型,我们可以向它要实体,但一般我们不需要直接使用这个对象
- 创建指向SQLite数据库的URL. SQLite文件在Documents文件夹中
- 创建NSPersistentStoreCoordinator,这个对象管理SQLite数据库
- 把SQLite数据库加到coordinator
- 最后创建NSManagedObjectContext
-
NSManagedObjectContext 这个对象用来与Core Data沟通,我们通常在context上做改变,然后调用它的save方法就可以存储到数据库中。
向数据库中插入一条数据- 对象 = [NSEntityDescription insert…]
- 设置对象属性
- 调用context save()
删除
- 向fetchedResultsController 拿到要删除的对象
- 调用context deleteObject:方法
- 调用context save方法
查询 - 创建NSFetchRequest
- 创建要查询的实体 [NSEntityDescription entityForName…], 并设置request.entity = entity
- 排序,根据字段key排序 [NSSortDescriptor key: ascending:], 并设置request.sortDescriptors = @[]
- 调用context executeFetchRequest:执行查询得到结果
-
在tableView中,通常不用以上这种查询方式,CoreData提供了NSFetchedResultsController来与tableView更加密切合作。(TabBar第二个界面)
使用懒加载NSFetchedResultsController,前3步都与以上相同。
4.选择性设置fetchRequest.fetchBatchSize = 20
5.创建NSFetchedResultsController,sectionNameKeyPath为j结果分组的key,cacheName为缓存的名字(每次performFetch都要把cache删除)
6.设置代理NSFetchedResultsControllerDelegate,4个代理方法用来更新TableView,具体看github代码。- 调用fetchedResultsController performFetch: 从数据库中取数据
调用fetchedResultsController objectAtIndexPath:得到对应的某个数据
- 调用fetchedResultsController performFetch: 从数据库中取数据
调用相册和照相机
UIImagePickerController *imagePicker = [[MyImagePickerController alloc] init];
imagePicker.sourceType = UIImagePickerControllerSourceTypeCamera; // PhotoLibrary为相册
imagePicker.delegate = self;
imagePicker.allowsEditing = YES;
imagePicker.view.tintColor = self.view.tintColor;
[self presentViewController:imagePicker animated:YES completion:nil];
- 必须要遵循UIImagePickerControllerDelegate,UINavigationControllerDelegate,只需要实现ImagePicker的两个代理方法
-(void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info {
_image = info[UIImagePickerControllerEditedImage];//取得照片
if (_image) {
[self showImage:_image];
}
[self.tableView reloadData];
[self dismissViewControllerAnimated:YES completion:nil];
}
-(void)imagePickerControllerDidCancel:(UIImagePickerController *)picker {
[self dismissViewControllerAnimated:YES completion:nil];
}
- MKMapView (TabBar第三个界面)
- 显示自己的位置
MKCoordinateRegion region = MKCoordinateRegionMakeWithDistance(self.mapView.userLocation.coordinate, 1000, 1000); // 1000米的度量
[self.mapView setRegion:[self.mapView regionThatFits:region] animated:YES];
- 显示其它坐标位置,其它坐标必须遵守MKAnnotation协议,并实现三个属性。为了把所有的坐标都显示在地图上,首先遍历坐标得到最小最大经度和最小最大纬度,从而计算出所有坐标的中心center,再用最大经度-最小经度,最大纬度-最小纬度得到要显示的范围span,为了不让坐标显示在屏幕边缘,把范围再乘以1.1,最后调用MKCoordinateRegionMake(center, span)得到最终显示的region. 再调用mapView setRegion:即可。具体看github代码
- 自定义MKPinAnnotationView(就是那个插在地图上的竹竿)
类似自定义tableView上的Cell,可复用。遵循MKMapViewDelegate代理协议,实现mapView:viewForAnnotation:方法。
总结
此app是对tableView使用的进一步加强,还有对CoreData的认识和使用,并且还有对CoreLocation和MapView的理解,初涉CoreAnimation之CABasicAnimation,等等。细节的处理、熟悉api,精密的逻辑,让我对iOS开发又上升了一个层次。不说了,继续加油!
源代码传送门
卧薪藏胆,三千越甲可吞吴。