iOS开发实战:从入门到上架App Store(第2版) (移动开发丛书)
上QQ阅读APP看书,第一时间看更新

2.13 选择器控件——UIPickerView

UIPickerView是一个简易的列表控件,用于提供有限个数的选项供用户选择,UIPickerView的UI设计十分漂亮,是iOS系统特有的UI模式。在实际应用中,UIPickerView的应用也十分广泛,如省市县选择列表、时间选择、日期选择等都可以通过UIPickerView设计。

2.13.1 创建一个UIPickerView控件

UIPickerView与之前章节中的UI控件有着很大不同,UIPickerView更加复杂一些,它是通过代理和数据源的方法对其进行设置和数据源填充的,这种控件的设计模式也是代理模式的应用之一。使用Xcode开发工具创建一个名为UIPickerViewTest的工程,在ViewController类的声明部分添加遵守相应的协议,这里需要遵守两个协议,分别是UIPickerViewDelegate和UIPickerViewDataSource。

Swift语言版本:

        class ViewController: UIViewController, UIPickerViewDataSource, UIPickerViewDelegate

Objective-C语言版本:

        @interface ViewController ()<UIPickerViewDataSource, UIPickerViewDelegate>
        @end

在viewDidLoad方法中添加如下代码。

Swift语言版本:

            override func viewDidLoad() {
                super.viewDidLoad()
                let picker = UIPickerView(frame: CGRect(x: 20, y: 100, width: 280, height: 150))
                picker.delegate = self
                picker.dataSource = self
                self.view.addSubview(picker)
            }

Objective-C语言版本:

        - (void)viewDidLoad {
            [super viewDidLoad];
            UIPickerView * picker = [[UIPickerView alloc]initWithFrame:CGRectMake(20, 100, 280, 150)];
            picker.delegate=self;
            picker.dataSource=self;
            [self.view addSubview:picker];
        }

上面的代码中除了对UIPickerView进行创建和初始化工作外,还设置了当前类对象为其数据源和代理。在ViewController类中实现如下代理方法。

Swift语言版本:

            func numberOfComponents(in pickerView: UIPickerView) -> Int {
                return 2
            }
            func pickerView(_ pickerView: UIPickerView, numberOfRowsInComponent component: Int) -> Int {
                return 10
            }
            func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) ->String? {
                return "\(component)分区\(row)行数据"
            }
            func pickerView(_ pickerView: UIPickerView, rowHeightForComponent component: Int) -> CGFloat {
                return 44
            }
            func pickerView(_ pickerView: UIPickerView, widthForComponent component: Int) -> CGFloat {
                return 144
            }

Objective-C语言版本:

        -(NSInteger)numberOfComponentsInPickerView:(UIPickerView *)pickerView{
            return 2;
        }
        -(NSInteger)pickerView:(UIPickerView *)pickerView numberOfRowsInComponent:(NSInteger)component{
            return 10;
        }
        -(NSString *)pickerView:(UIPickerView *)pickerView titleForRow:(NSInteger)row forComponent:
    (NSInteger)component{
            return [NSString stringWithFormat:@"%lu分区%lu数据", component, row];
        }
        -(CGFloat)pickerView:(UIPickerView *)pickerView rowHeightForComponent:(NSInteger)component{
            return 44;
        }
        -(CGFloat)pickerView:(UIPickerView *)pickerView widthForComponent:(NSInteger)component{
            return 140;
        }

· numberOfComponentsInPickerView方法返回一个整型数据,用于设置UIPickerView视图的分区数,也可以理解为选择列表的列数。

· pickerView:numberOfRowsInComponent方法的返回值设置UIPickerView每个分区的行数,参数component用于判断具体的分区。

· titleForRow方法的返回值设置列表中每一行的数据,这个方法中的两个参数row和component分别用于区分行与列。

· rowHeightForComponent方法的返回值设置具体行的行高。

· widthForComponent方法的返回值设置分区的宽度,即列的宽度。

运行工程,效果如图2-39所示。

图2-39 UIPickerView控件

2.13.2 UIPickerView选中数据时的回调代理

UIPickerView总是会展现几列数据帮助用户进行快速选择,当用户上下滑动UIPickerView列表时,列表中的数据会进行上下滑动移动;当移动动作停止时,这时悬停在UIPickerView列表中间的数据就是用户选中的数据,并且此时系统会调用UIPickerView的如下代理方法通知开发者用户的选择。

Swift语言风格:

            func pickerView(_ pickerView: UIPickerView, didSelectRow row: Int, inComponent component: Int) {
                print("\(component)分区\(row)行")
            }

Objective-C风格:

        -(void)pickerView:(UIPickerView *)pickerView didSelectRow:(NSInteger)row
    inComponent:(NSInteger)component{
            NSLog(@"%lu, %lu", row, component);
        }

在调用didSelectRow方法时,row参数和component参数会将用户选择的行和分区的信息传递给开发者。