最近项目要做一个,类似淘宝手机客户端的,选择收货地址的三级联动滚动选择组件,下面是它的大致界面截图:

在IOS中有个叫UIPickerView的选择器,并且在dataSource中定义了UIPickerView的数据源和定制内容,所以用只要熟悉它的基本用法,要实现这么个三级联动滑动选择是挺简单的。

言归正传,今天讨论的是在Android里面如何来实现这么个效果,那么如何实现呢??? 相信部分童鞋首先想到的是android.widget.DatePicker和android.widget.TimePicker,因为它们的样子长得很像,事实就是它们仅仅是长得相而已,Google在设计这个两个widget的时候,并没有提供对外的数据源适配接口,带来的问题就是,我们只能通过它们来选择日期和时间,至于为什么这样设计,如果有童鞋知道,请给我留言,Thanks~

DatePicker.class包含的方法截图:

 全都是关于时间获取用的方法.

好了,既然在Android中没办法偷懒的用一个系统widget搞定,那么只能自己来自定义view来实现了,这篇就围绕这个来展开分享一下,我在项目中实现这个的全过程。首先是做了下开源代码调研,在github上面有一个叫做 android-wheel 的开源控件,
代码地址https://github.com/maarek/android-wheel

是一个非常好用的组件,对于数据适配接口的抽取和事件的回调都做了抽取,代码的耦合度低,唯一不足就是在界面的定制这块,如果你需要做更改,需要去动源代码的。我这里在界面的代码做了改动,放在我的项目src目录下了:

在此次项目中,省市区及邮编的数据是放在了assets/province_data.xml里面,是产品经理花了好几天时间整理的,绝对是最齐全和完善了,辛苦辛苦!!!

关于XML的解析,一共有SAX、PULL、DOM三种解析方式,这里就不讲了,可以看我的前面的几篇学习的文章:

Android解析XML方式(一)使用SAX解析

Android解析XML方式(二)使用PULL解析XML

Android解析XML方式(三)使用DOM解析XML

此次项目中使用的是SAX解析方式,因为它占用内存少,并且速度快,数据解析代码写在了 com.mrwujay.cascade.service/XmlParserHandler.java中,代码如下:

  1. package com.mrwujay.cascade.service;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. import org.xml.sax.Attributes;
  5. import org.xml.sax.SAXException;
  6. import org.xml.sax.helpers.DefaultHandler;
  7. import com.mrwujay.cascade.model.CityModel;
  8. import com.mrwujay.cascade.model.DistrictModel;
  9. import com.mrwujay.cascade.model.ProvinceModel;
  10. public class XmlParserHandler extends DefaultHandler {
  11. /**
  12. * 存储所有的解析对象
  13. */
  14. private List<ProvinceModel> provinceList = new ArrayList<ProvinceModel>();
  15. public XmlParserHandler() {
  16. }
  17. public List<ProvinceModel> getDataList() {
  18. return provinceList;
  19. }
  20. @Override
  21. public void startDocument() throws SAXException {
  22. // 当读到第一个开始标签的时候,会触发这个方法
  23. }
  24. ProvinceModel provinceModel = new ProvinceModel();
  25. CityModel cityModel = new CityModel();
  26. DistrictModel districtModel = new DistrictModel();
  27. @Override
  28. public void startElement(String uri, String localName, String qName,
  29. Attributes attributes) throws SAXException {
  30. // 当遇到开始标记的时候,调用这个方法
  31. if (qName.equals("province")) {
  32. provinceModel = new ProvinceModel();
  33. provinceModel.setName(attributes.getValue(0));
  34. provinceModel.setCityList(new ArrayList<CityModel>());
  35. } else if (qName.equals("city")) {
  36. cityModel = new CityModel();
  37. cityModel.setName(attributes.getValue(0));
  38. cityModel.setDistrictList(new ArrayList<DistrictModel>());
  39. } else if (qName.equals("district")) {
  40. districtModel = new DistrictModel();
  41. districtModel.setName(attributes.getValue(0));
  42. districtModel.setZipcode(attributes.getValue(1));
  43. }
  44. }
  45. @Override
  46. public void endElement(String uri, String localName, String qName)
  47. throws SAXException {
  48. // 遇到结束标记的时候,会调用这个方法
  49. if (qName.equals("district")) {
  50. cityModel.getDistrictList().add(districtModel);
  51. } else if (qName.equals("city")) {
  52. provinceModel.getCityList().add(cityModel);
  53. } else if (qName.equals("province")) {
  54. provinceList.add(provinceModel);
  55. }
  56. }
  57. @Override
  58. public void characters(char[] ch, int start, int length)
  59. throws SAXException {
  60. }
  61. }

通过XmlParserHandler.java提供的getDataList()方法获取得到,之后再进行拆分放到省、市、区不同的HashMap里面方便做数据适配。

这里是它的具体实现代码:

  1. protected void initProvinceDatas()
  2. {
  3. List<ProvinceModel> provinceList = null;
  4. AssetManager asset = getAssets();
  5. try {
  6. InputStream input = asset.open("province_data.xml");
  7. // 创建一个解析xml的工厂对象
  8. SAXParserFactory spf = SAXParserFactory.newInstance();
  9. // 解析xml
  10. SAXParser parser = spf.newSAXParser();
  11. XmlParserHandler handler = new XmlParserHandler();
  12. parser.parse(input, handler);
  13. input.close();
  14. // 获取解析出来的数据
  15. provinceList = handler.getDataList();
  16. //*/ 初始化默认选中的省、市、区
  17. if (provinceList!= null && !provinceList.isEmpty()) {
  18. mCurrentProviceName = provinceList.get(0).getName();
  19. List<CityModel> cityList = provinceList.get(0).getCityList();
  20. if (cityList!= null && !cityList.isEmpty()) {
  21. mCurrentCityName = cityList.get(0).getName();
  22. List<DistrictModel> districtList = cityList.get(0).getDistrictList();
  23. mCurrentDistrictName = districtList.get(0).getName();
  24. mCurrentZipCode = districtList.get(0).getZipcode();
  25. }
  26. }
  27. //*/
  28. mProvinceDatas = new String[provinceList.size()];
  29. for (int i=0; i< provinceList.size(); i++) {
  30. mProvinceDatas[i] = provinceList.get(i).getName();
  31. List<CityModel> cityList = provinceList.get(i).getCityList();
  32. String[] cityNames = new String[cityList.size()];
  33. for (int j=0; j< cityList.size(); j++) {
  34. cityNames[j] = cityList.get(j).getName();
  35. List<DistrictModel> districtList = cityList.get(j).getDistrictList();
  36. String[] distrinctNameArray = new String[districtList.size()];
  37. DistrictModel[] distrinctArray = new DistrictModel[districtList.size()];
  38. for (int k=0; k<districtList.size(); k++) {
  39. DistrictModel districtModel = new DistrictModel(districtList.get(k).getName(), districtList.get(k).getZipcode());
  40. mZipcodeDatasMap.put(districtList.get(k).getName(), districtList.get(k).getZipcode());
  41. distrinctArray[k] = districtModel;
  42. distrinctNameArray[k] = districtModel.getName();
  43. }
  44. mDistrictDatasMap.put(cityNames[j], distrinctNameArray);
  45. }
  46. mCitisDatasMap.put(provinceList.get(i).getName(), cityNames);
  47. }
  48. } catch (Throwable e) {
  49. e.printStackTrace();
  50. } finally {
  51. }
  52. }

在使用wheel组件时,数据适配起来也很方便,只需要做些数据、显示数量的配置即可,我这边设置了一行显示7条数据

  1. initProvinceDatas();
  2. mViewProvince.setViewAdapter(new ArrayWheelAdapter<String>(MainActivity.this, mProvinceDatas));
  3. // 设置可见条目数量
  4. mViewProvince.setVisibleItems(7);
  5. mViewCity.setVisibleItems(7);
  6. mViewDistrict.setVisibleItems(7);
  7. updateCities();
  8. updateAreas();

要监听wheel组件的滑动、点击、选中改变事件,可以通过实现它的三个事件监听接口来实现,分别是:

1、OnWheelScrollListener 滑动事件:

  1. /**
  2. * Wheel scrolled listener interface.
  3. */
  4. public interface OnWheelScrollListener {
  5. /**
  6. * Callback method to be invoked when scrolling started.
  7. * @param wheel the wheel view whose state has changed.
  8. */
  9. void onScrollingStarted(WheelView wheel);
  10. /**
  11. * Callback method to be invoked when scrolling ended.
  12. * @param wheel the wheel view whose state has changed.
  13. */
  14. void onScrollingFinished(WheelView wheel);
  15. }

2、OnWheelClickedListener 条目点击事件:

  1. /**
  2. * Wheel clicked listener interface.
  3. * <p>The onItemClicked() method is called whenever a wheel item is clicked
  4. * <li> New Wheel position is set
  5. * <li> Wheel view is scrolled
  6. */
  7. public interface OnWheelClickedListener {
  8. /**
  9. * Callback method to be invoked when current item clicked
  10. * @param wheel the wheel view
  11. * @param itemIndex the index of clicked item
  12. */
  13. void onItemClicked(WheelView wheel, int itemIndex);
  14. }

3、OnWheelChangedListener 被选中项的positon变化事件:

  1. /**
  2. * Wheel changed listener interface.
  3. * <p>The onChanged() method is called whenever current wheel positions is changed:
  4. * <li> New Wheel position is set
  5. * <li> Wheel view is scrolled
  6. */
  7. public interface OnWheelChangedListener {
  8. /**
  9. * Callback method to be invoked when current item changed
  10. * @param wheel the wheel view whose state has changed
  11. * @param oldValue the old value of current item
  12. * @param newValue the new value of current item
  13. */
  14. void onChanged(WheelView wheel, int oldValue, int newValue);
  15. }

这里只要知道哪个省、市、区被选中了,实现第三个接口就行,在方法回调时去作同步和更新数据,比如省级条目滑动的时候,市级和县级数据都要做对应的适配、市级滑动时需要去改变县级(区)的数据,这样才能实现级联的效果,至于如何改变,需要三个HashMap来分别保存他们的对应关系:

  1. /**
  2. * key - 省 value - 市
  3. */
  4. protected Map<String, String[]> mCitisDatasMap = new HashMap<String, String[]>();
  5. /**
  6. * key - 市 values - 区
  7. */
  8. protected Map<String, String[]> mDistrictDatasMap = new HashMap<String, String[]>();
  9. /**
  10. * key - 区 values - 邮编
  11. */
  12. protected Map<String, String> mZipcodeDatasMap = new HashMap<String, String>();

在onChanged()回调方法中,对于省、市、区/县的滑动,分别做数据的适配,代码如下:

  1. @Override
  2. public void onChanged(WheelView wheel, int oldValue, int newValue) {
  3. // TODO Auto-generated method stub
  4. if (wheel == mViewProvince) {
  5. updateCities();
  6. } else if (wheel == mViewCity) {
  7. updateAreas();
  8. } else if (wheel == mViewDistrict) {
  9. mCurrentDistrictName = mDistrictDatasMap.get(mCurrentCityName)[newValue];
  10. mCurrentZipCode = mZipcodeDatasMap.get(mCurrentDistrictName);
  11. }
  12. }
  13. /**
  14. * 根据当前的市,更新区WheelView的信息
  15. */
  16. private void updateAreas() {
  17. int pCurrent = mViewCity.getCurrentItem();
  18. mCurrentCityName = mCitisDatasMap.get(mCurrentProviceName)[pCurrent];
  19. String[] areas = mDistrictDatasMap.get(mCurrentCityName);
  20. if (areas == null) {
  21. areas = new String[] { "" };
  22. }
  23. mViewDistrict.setViewAdapter(new ArrayWheelAdapter<String>(this, areas));
  24. mViewDistrict.setCurrentItem(0);
  25. }
  26. /**
  27. * 根据当前的省,更新市WheelView的信息
  28. */
  29. private void updateCities() {
  30. int pCurrent = mViewProvince.getCurrentItem();
  31. mCurrentProviceName = mProvinceDatas[pCurrent];
  32. String[] cities = mCitisDatasMap.get(mCurrentProviceName);
  33. if (cities == null) {
  34. cities = new String[] { "" };
  35. }
  36. mViewCity.setViewAdapter(new ArrayWheelAdapter<String>(this, cities));
  37. mViewCity.setCurrentItem(0);
  38. updateAreas();
  39. }

综上代码,最终实现的界面截图:

源码下载地址:http://download.csdn.net/detail/wulianghuan/8205211

android仿iphone的地区选择的更多相关文章

  1. Android仿iPhone 滚轮控件 实现

    Android_开发 实用滚轮效果选择数字http://blog.csdn.net/zhangtengyuan23/article/details/8653771 Android仿iPhone滚轮控件 ...

  2. Android 仿土巴兔选择效果

    1,前两天在群里看到有人在讨论土巴兔的选择装修风格的效果,自己也想实现,果断百度一下,有些好的文章,就花了些时间来分析了下,先看看别人土巴兔原装的功能 2,可以看到,基本上可以使用一个vviewpag ...

  3. Android仿iPhone晃动撤销输入功能(微信摇一摇功能)

    重力传感器微信摇一摇SensorMannager自定义alertdialogSensorEventListener 很多程序中我们可能会输入长文本内容,比如短信,写便笺等,如果想一次性撤销所有的键入内 ...

  4. Android 仿iPhone的日期时间选择器

    可选只选择日期,也可以同时选择时间 只选择日期的情况 同时选择日期和时间的情况 关键代码: findViewById(R.id.selectDateButton).setOnClickListener ...

  5. Android 轻松实现仿淘宝地区选择

    介绍 最近用淘宝客户端的时候,编辑地址的时候有个地区选择的功能.看上面的效果觉得挺酷,滚动的时候,是最后一个从下面飞上来挨着前一个.就自己鼓捣一个出来玩玩. 说了效果可能不太直观,下面上两张图看看效果 ...

  6. Android 仿PhotoShop调色板应用(四) 不同区域颜色选择的颜色生成响应

    版权声明:本文为博主原创文章,未经博主允许不得转载.  Android 仿PhotoShop调色板应用(四) 不同区域颜色选择的颜色生成响应  上一篇讲过了主体界面的绘制,这里讲解调色板应用中的另外一 ...

  7. 仿iphone日历插件(beta)

    前言 小伙伴们好,很久不见了.最近工作进入正常期了,所以慢慢的悠闲的时间久没有了,所以不能每天水一篇了. 最近也在听师傅(http://home.cnblogs.com/u/aaronjs/)的教导开 ...

  8. 转-Android仿微信气泡聊天界面设计

    微信的气泡聊天是仿iPhone自带短信而设计出来的,不过感觉还不错可以尝试一下仿着微信的气泡聊天做一个Demo,给大家分享一下!效果图如下: 气泡聊天最终要的是素材,要用到9.png文件的素材,这样气 ...

  9. Android 仿PhotoShop调色板应用(三) 主体界面绘制

    版权声明:本文为博主原创文章,未经博主允许不得转载. Android 仿PhotoShop调色板应用(三) 主体界面绘制    关于PhotoShop调色板应用的实现我总结了两个最核心的部分:   1 ...

随机推荐

  1. Python SMTP邮件发送

    SMTP是发送邮件的协议,Python内置对SMTP的支持,可以发送纯文本邮件.HTML邮件以及带附件的邮件. Python对SMTP支持有smtplib和email两个模块: email负责构造邮件 ...

  2. java线程池的使用

    一.简介 线程的使用在java中占有极其重要的地位,在jdk1.4极其之前的jdk版本中,关于线程池的使用是极其简陋的.在jdk1.5之后这一情况有了很大的改观.Jdk1.5之后加入了java.uti ...

  3. 从JVM角度看i++ 与++i

    1.i++和++i的问题 反编译结果为 Code:  0:   iconst_1  1:   istore_1  2:   iinc    1, 1 //这个个指令,把局部变量1,也就是i,增加1,这 ...

  4. Mac Webview OC与JS交互实现

    1.首先,需要定义一个JS可识别的变量(如external)用于OC与JS交互 - (void)webView:(WebView *)sender didClearWindowObject:(WebS ...

  5. Mysql--存储引擎(MyISam & InnoDB)

    Mysql 系列文章主页 =============== 查看 Mysql 支持的存储引擎: show engines; 查看当前数据库使用的存储引擎: show variables like '%s ...

  6. angular2+ionic2架构介绍

    不要用angular的语法去写angular2,有人说二者就像Java和JavaScript的区别.   1. 项目所用:angular2+ionic2+typescript 2. 项目结构 3. S ...

  7. 微信小程序 发现之旅(一)—— 项目搭建与页面跳转

    开发微信小程序需要注册一个小程序账号,具体流程可以参照官方教程: https://mp.weixin.qq.com/debug/wxadoc/dev/index.html 开通账户之后,在 “开发设置 ...

  8. python环境搭建(python2和python3共存)

    安装两个版本的意义 验证自己代码对版本的兼容性 网上下载的某些源码只能在python2或者python3中运行 安装过程记录 1.去python官网下载python的安装包, 下载完成后如下图所示 2 ...

  9. 硬盘存储计量单位KB、MB、GB大小换算

    一. 预备知识 1. bit与byte 1. bit(简记为 b) 1 bit = 0 or 1 = one binary 2. byte(简记为 B) 1 byte = 8 bits 1字节,8个二 ...

  10. 《Non-Negative Matrix Factorization for Polyphonic Music Transcription》译文

    NMF(非负矩阵分解),由于其分解出的矩阵是非负的,在一些实际问题中具有非常好的解释,因此用途很广.在此,我给大家介绍一下NMF在多声部音乐中的应用.要翻译的论文是利用NMF转录多声部音乐的开山之作, ...