效果图

概述

关于 省市区 三级联动的 pickerView,我想大多数的 iOS 开发者应该都遇到过这样的需求。在遇到这样的需求的时候,大多数人都会觉的这个很复杂,一时无从下手。其实真的没那么复杂。在这里我们来一起看看,怎么去实现这样的 pickerView,并做一个简单的封装,使其使用的更加简单,从而也减少了 ViewController 中的代码。

实现思路

如何封装

  • 我们使用一个 View(IDAddressPickerView) 来封装 PickerView,来处理 PickerView 的 dataSource 和 delegate,将原本需要在 ViewController处理的 逻辑封装的 View 中。
  • ViewController 只需要为 IDAddressPickerView 提供 dataSource,并获取选中的 Address。而不去关心其他逻辑,不如说:联动逻辑,数据格式化。
  • IDAddressPickerView 使用委托模式来获取 ViewController 提供的数据源。

数据如何组织

  • IDAddressPickerView 的数据源是一个数组,且需要满足一定的格式,这在一定程度上降低了其使用灵活性。

  • 目前 IDAddressPickerView 数据源的需要满足的格式如图:

获取选中的地址

  • 选中地址的格式,目前是通过固定的 key 包装在一个 Dictionary,灵活性不高。
  • 在此没有使用委托等模式,而是通过一个属性保存当前选中的地址,让用户(IDAddressPickerView 的使用者)主动去获取选中的地址。

期望结果

  • 目前实现的 IDAddressPickerView 的数据源缺乏灵活性,尽管我们可以与后台沟通约定数据格式,但是对于一个封装的 IDAddressPickerView 来说,显然是不妥当的。
  • 我期望实现的结果是,在为 IDAddressPickerView 提供数据源的时候,指定一个 dataFormatter,IDAddressPickerView 根据 dataFormatter 去解析数据源的数据。而不是现在的根据固定的格式解析数据。
  • 由于这些问题的存在,我将项目代码上传到 github 上,还希望有兴趣的小伙伴们多提宝贵意见。

具体实现

IDAddressPickerView

  • 自定义 UIView 的 子类 IDAddressPickerView

    @interface IDAddressPickerView : UIView
    @end
  • 添加 UIPickerView 子控件

    - (UIPickerView *)pickerView {
    if (_pickerView == nil) {
    _pickerView = [[UIPickerView alloc] init];
    _pickerView.dataSource = self;
    _pickerView.delegate = self;
    }
    return _pickerView;
    }
    • UIPickerView 的数据源

      - (NSInteger)numberOfComponentsInPickerView:(UIPickerView *)pickerView {
      return 3;
      }
      - (NSInteger)pickerView:(UIPickerView *)pickerView numberOfRowsInComponent:(NSInteger)component {
      NSInteger numberOfRowsInComponent = 0;
      switch (component) {
      case 0:
      numberOfRowsInComponent = self.addressArray.count;
      break;
      case 1:
      {
      NSDictionary *province = self.addressArray[self.provinceIndex];
      numberOfRowsInComponent = [province[@"cities"] count];
      }
      break;
      case 2:
      {
      NSDictionary *province = self.addressArray[self.provinceIndex];
      NSDictionary *cities = province[@"cities"][self.cityIndex];
      numberOfRowsInComponent = [cities[@"areas"] count];
      }
      break;
      }
      return numberOfRowsInComponent;
      }
      - (NSString *)pickerView:(UIPickerView *)pickerView titleForRow:(NSInteger)row forComponent:(NSInteger)component {
      NSString *titleForRow = @"";
      switch (component) {
      case 0:
      titleForRow = self.addressArray[row][@"state"];
      break;
      case 1:
      {
      NSDictionary *province = self.addressArray[self.provinceIndex];
      titleForRow = province[@"cities"][row][@"city"];
      }
      break;
      case 2:
      {
      NSDictionary *province = self.addressArray[self.provinceIndex];
      NSDictionary *city = province[@"cities"][self.cityIndex];
      titleForRow = city[@"areas"][row];
      }
      break;
      }
      return titleForRow;
      }
    • UIPickerView 的代理

      - (void)pickerView:(UIPickerView *)pickerView didSelectRow:(NSInteger)row inComponent:(NSInteger)component {
      switch (component) {
      case 0:
      {
      self.provinceIndex = row;
      self.cityIndex = 0;
      self.areaIndex = 0;
      [pickerView reloadComponent:1];
      [pickerView reloadComponent:2];
      [pickerView selectRow:0 inComponent:1 animated:NO];
      [pickerView selectRow:0 inComponent:2 animated:NO];
      /**
      * 更新选中的 addresss,包括:市,区
      */
      NSDictionary *province = self.addressArray[self.provinceIndex];
      NSDictionary *city = province[@"cities"][self.cityIndex];
      self.selectedAddress[ProvinceKey] = self.addressArray[row][@"state"];
      if ([province[@"cities"] count] > 0) {
      self.selectedAddress[CityKey] = province[@"cities"][0][@"city"];
      } else {
      self.selectedAddress[CityKey] = @"";
      }
      if ([city[@"areas"] count] > 0) {
      self.selectedAddress[AreaKey] = city[@"areas"][0];
      } else {
      self.selectedAddress[AreaKey] = @"";
      }
      }
      break;
      case 1:
      {
      self.cityIndex = row;
      self.areaIndex = 0;
      [pickerView reloadComponent:2];
      [pickerView selectRow:0 inComponent:2 animated:NO];
      /**
      * 更新选中的 addresss,包括:区
      */
      NSDictionary *province = self.addressArray[self.provinceIndex];
      NSDictionary *city = province[@"cities"][self.cityIndex];
      self.selectedAddress[CityKey] = province[@"cities"][row][@"city"];
      if ([city[@"areas"] count] > 0) {
      self.selectedAddress[AreaKey] = city[@"areas"][0];
      } else {
      self.selectedAddress[AreaKey] = @"";
      }
      }
      break;
      case 2:
      {
      self.areaIndex = row;
      /**
      * 更新选中的 addresss
      */
      NSDictionary *province = self.addressArray[self.provinceIndex];
      NSDictionary *city = province[@"cities"][self.cityIndex];
      self.selectedAddress[AreaKey] = city[@"areas"][row];
      }
      break;
      }
      }
    • 关于 UIPickerView 的数据源

      • UIPickerView 的数据源 通过 IDAddressPickerViewDataSource 协议获得

        - (NSArray *)addressArray {
        if (_addressArray == nil) {
        if ([self.dataSource respondsToSelector:@selector(addressArray)]) {
        _addressArray = [self.dataSource addressArray];
        } else {
        _addressArray = [NSArray array];
        }
        }
        return _addressArray;
        }

联动效果的实现

  • 原理

    • 基本的原理是通过更新数据源的方式,来实现选中一列中的某一行时,更新后继(更深层次)的列。
  • 具体实现

    • 在此使用三个属性分别记录省市区三个层次的对应的列中选中的行,UIPickerView 通过这三个属性来获取对应的数据源。

    • 选中一列中的某一行时,需要更新当前列及其后继列所对应的选中行信息。

      /** 选中的省份 */
      @property (nonatomic, assign) NSInteger provinceIndex;
      /** 选中的城市 */
      @property (nonatomic, assign) NSInteger cityIndex;
      /** 选中的省份 */
      @property (nonatomic, assign) NSInteger areaIndex;
  • 更新后继的列

    • 来实现选中一列中的某一行时,更新 所有 后继列,默认选中第一行。即,选中第一列时,更新第二列和第三列;选中第二列时,更新第三列;选中第三列时。
    - (void)pickerView:(UIPickerView *)pickerView didSelectRow:(NSInteger)row inComponent:(NSInteger)component {
    switch (component) {
    case 0:
    {
    self.provinceIndex = row;
    self.cityIndex = 0;
    self.areaIndex = 0;
    [pickerView reloadComponent:1];
    [pickerView reloadComponent:2];
    [pickerView selectRow:0 inComponent:1 animated:NO];
    [pickerView selectRow:0 inComponent:2 animated:NO];
    /**
    * 更新选中的 addresss,包括:市,区
    */
    NSDictionary *province = self.addressArray[self.provinceIndex];
    NSDictionary *city = province[@"cities"][self.cityIndex];
    self.selectedAddress[ProvinceKey] = self.addressArray[row][@"state"];
    if ([province[@"cities"] count] > 0) {
    self.selectedAddress[CityKey] = province[@"cities"][0][@"city"];
    } else {
    self.selectedAddress[CityKey] = @"";
    }
    if ([city[@"areas"] count] > 0) {
    self.selectedAddress[AreaKey] = city[@"areas"][0];
    } else {
    self.selectedAddress[AreaKey] = @"";
    }
    }
    break;
    case 1:
    {
    self.cityIndex = row;
    self.areaIndex = 0;
    [pickerView reloadComponent:2];
    [pickerView selectRow:0 inComponent:2 animated:NO];
    /**
    * 更新选中的 addresss,包括:区
    */
    NSDictionary *province = self.addressArray[self.provinceIndex];
    NSDictionary *city = province[@"cities"][self.cityIndex];
    self.selectedAddress[CityKey] = province[@"cities"][row][@"city"];
    if ([city[@"areas"] count] > 0) {
    self.selectedAddress[AreaKey] = city[@"areas"][0];
    } else {
    self.selectedAddress[AreaKey] = @"";
    }
    }
    break;
    case 2:
    {
    self.areaIndex = row;
    /**
    * 更新选中的 addresss
    */
    NSDictionary *province = self.addressArray[self.provinceIndex];
    NSDictionary *city = province[@"cities"][self.cityIndex];
    self.selectedAddress[AreaKey] = city[@"areas"][row];
    }
    break;
    }
    }

IDAddressPickerViewDataSource 协议

@protocol IDAddressPickerViewDataSource <NSObject>
/**
* 地址信息,指定格式的数组
*/
- (NSArray *)addressArray;
@end

使用示例

  • 设置 textField 的 inputView 为 IDAddressPickerView

    _textField.inputView = self.addressPickerView;
    // getter
    - (IDAddressPickerView *)addressPickerView {
    if (_addressPickerView == nil) {
    _addressPickerView = [[IDAddressPickerView alloc] init];
    _addressPickerView.dataSource = self;
    }
    return _addressPickerView;
    }
  • IDAddressPickerViewDataSource 提供数据

    #pragma mark - IDAddressPickerViewDataSource
    - (NSArray *)addressArray {
    NSString *path = [[NSBundle mainBundle] pathForResource:@"address" ofType:@"plist"];
    NSArray *addressInfo = [NSArray arrayWithContentsOfFile:path];
    return addressInfo;
    }
  • 获取选中的地址

    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSLog(@"%@", self.addressPickerView.selectedAddress);
    }

声明

项目代码已经上传到 gitHub,若需要请自行获取:IDAddressPickerView

省市区三级联动 pickerView的更多相关文章

  1. jQuery省市区三级联动插件

    体验效果:http://hovertree.com/texiao/bootstrap/4/支持PC和手机移动端. 手机扫描二维码体验效果: 代码如下: <!DOCTYPE html> &l ...

  2. JS省市区三级联动

    不需要访问后台服务器端,不使用Ajax,无刷新,纯JS实现的省市区三级联动. 当省市区数据变动是只需调正js即可. 使用方法: <!DOCTYPE html><html>< ...

  3. ajax省市区三级联动

    jdbc+servlet+ajax开发省市区三级联动 技术点:jdbc操作数据库,ajax提交,字符拦截器,三级联动 特点:局部刷新达到省市区三级联动,举一反三可以做商品分类等 宗旨:从实战中学习 博 ...

  4. QQ JS省市区三级联动

    如下图: 首先写一个静态的页面: <!DOCTYPE html> <html> <head> <title>QQ JS省市区三级联动</title ...

  5. 省市区三级联动(二)JS部分简单版

    通过对上一篇<省市区三级联动>的学习发现JScript部分省市区的填充代码几乎相同,所以可以写成一个函数. 注意:html部分和chuli.php部分不变 1.下拉列表填充可以写成带参数的 ...

  6. 从QQ网站中提取的纯JS省市区三级联动

    在 http://ip.qq.com/ 的网站中有QQ自己的JS省市区三级联动 QQ是使用引用外部JS来实现三级联动的.JS如下:http://ip.qq.com/js/geo.js <!DOC ...

  7. 基于ThinkPHP+AJAX的省市区三级联动

    练习,就当练习. 省市区三级联动,样式如下图所示: 1,导入两个js文件并且导入数据库文件. 两个js文件分别是jquery-2.1.4.min.js和jquery-1.js,数据库文件,见附件. 2 ...

  8. java的JCombobox实现中国省市区三级联动

    源代码下载:点击下载源代码 用xml存储中国各大城市的数据. xml数据太多了就不贴上了,贴个图片: 要解释xml,添加了一个jdom.jar,上面的源代码下载里面有. 解释xml的类: packag ...

  9. jquery省市区三级联动

    jquery省市区三级联动(数据来源国家统计局官网)内附源码下载 很久很久没有写博了. 今天更新了项目的省市区三级联动数据,更新后最新的海南三沙都有,分享给所有需要的小伙伴们... JQUERY + ...

随机推荐

  1. 【记录】GitHub/TortoiseGit 修改邮箱/提交者

    我使用 Git 客户端工具是 TortoiseGit,在提交更新的时候,不知何时起会出现下面这种情况: 正常提交作者信息显示应该是: 本来也没怎么注意,但是在提交历史中,记录就不显示出来了,也就是在首 ...

  2. 深入seajs源码系列二

    模块类和状态类 参照上文的demo,我们结合源码分析在简单的API调用的背后,到底使用了什么技巧来实现各个模块的依赖加载以及模块API的导出. 首先定义了一个Module类,对应与一个模块 funct ...

  3. L2/L3/L4 Switch简介

    第二层交换机,是根据第二层数据链路层的MAC地址和通过站表选择路由来完成端到端的数据交换的.因为站表的建立与维护是由交换机自动完成,而路由器又是属于第三层设备,其寻址过程是根据IP地址寻址和通过路由表 ...

  4. java枚举类型学习

    用的不多,但用的时候仅仅简单的使用,不太明白原理,今天就系统的学一下枚举.参考:java编程思想. Update: 枚举可以当做数据字典来存储,通常只要一个字段即instance本身,toString ...

  5. Nancy之文件上传与下载

    零.前言 由于前段时间一直在找工作,找到工作后又比较忙,又加班又通宵的赶项目,所以博客有段时间没有更新了. 今天稍微空闲一点,碰巧前几天看到有园友问我Nancy中下载文件的问题,然后就趁着休息的时间写 ...

  6. 无法解决 equal to 操作中 "SQL_Latin1_General_CP1_CI_AS" 和 "Chinese_PRC_CI_AS"

    无法解决 equal to 操作中 "SQL_Latin1_General_CP1_CI_AS" 和 "Chinese_PRC_CI_AS" 之间 2011-0 ...

  7. jQuery实现方式不一样的跳转到底部

    jQuery跳转到页面底部效果 在线体验:http://hovertree.com/texiao/jquery/9.htm 以下是完整HTML代码: <!DOCTYPE html> < ...

  8. 家族/亲戚(relation)

    题目描述 若某个家族人员过于庞大,要判断两个是否是亲戚,确实还很不容易,现在给出某个亲戚关系图,求任意给出的两个人是否具有亲戚关系. 规定:x和y是亲戚,y和z是亲戚,那么x和z也是亲戚.如果x,y是 ...

  9. 基于小脚丫的ADC081S101 电压采集595数码管显示

    RTL结构图 采集模块运用SPI 通讯 MISO方式收集数据 module ad_collect(input sddata,input rst_n,output reg cs,output reg s ...

  10. PHP intval()

    定义和用法 获取变量的整数值,允许以使用特定的进制返回.默认10进制 注:如果参数为整数,则不做任何处理. 语法 intval (var, base) 参数 描述 var 必须.可以是任何标量类型. ...