android仿iphone的地区选择
最近项目要做一个,类似淘宝手机客户端的,选择收货地址的三级联动滚动选择组件,下面是它的大致界面截图:
在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三种解析方式,这里就不讲了,可以看我的前面的几篇学习的文章:
此次项目中使用的是SAX解析方式,因为它占用内存少,并且速度快,数据解析代码写在了 com.mrwujay.cascade.service/XmlParserHandler.java中,代码如下:
- package com.mrwujay.cascade.service;
- import java.util.ArrayList;
- import java.util.List;
- import org.xml.sax.Attributes;
- import org.xml.sax.SAXException;
- import org.xml.sax.helpers.DefaultHandler;
- import com.mrwujay.cascade.model.CityModel;
- import com.mrwujay.cascade.model.DistrictModel;
- import com.mrwujay.cascade.model.ProvinceModel;
- public class XmlParserHandler extends DefaultHandler {
- /**
- * 存储所有的解析对象
- */
- private List<ProvinceModel> provinceList = new ArrayList<ProvinceModel>();
- public XmlParserHandler() {
- }
- public List<ProvinceModel> getDataList() {
- return provinceList;
- }
- @Override
- public void startDocument() throws SAXException {
- // 当读到第一个开始标签的时候,会触发这个方法
- }
- ProvinceModel provinceModel = new ProvinceModel();
- CityModel cityModel = new CityModel();
- DistrictModel districtModel = new DistrictModel();
- @Override
- public void startElement(String uri, String localName, String qName,
- Attributes attributes) throws SAXException {
- // 当遇到开始标记的时候,调用这个方法
- if (qName.equals("province")) {
- provinceModel = new ProvinceModel();
- provinceModel.setName(attributes.getValue(0));
- provinceModel.setCityList(new ArrayList<CityModel>());
- } else if (qName.equals("city")) {
- cityModel = new CityModel();
- cityModel.setName(attributes.getValue(0));
- cityModel.setDistrictList(new ArrayList<DistrictModel>());
- } else if (qName.equals("district")) {
- districtModel = new DistrictModel();
- districtModel.setName(attributes.getValue(0));
- districtModel.setZipcode(attributes.getValue(1));
- }
- }
- @Override
- public void endElement(String uri, String localName, String qName)
- throws SAXException {
- // 遇到结束标记的时候,会调用这个方法
- if (qName.equals("district")) {
- cityModel.getDistrictList().add(districtModel);
- } else if (qName.equals("city")) {
- provinceModel.getCityList().add(cityModel);
- } else if (qName.equals("province")) {
- provinceList.add(provinceModel);
- }
- }
- @Override
- public void characters(char[] ch, int start, int length)
- throws SAXException {
- }
- }
通过XmlParserHandler.java提供的getDataList()方法获取得到,之后再进行拆分放到省、市、区不同的HashMap里面方便做数据适配。
这里是它的具体实现代码:
- protected void initProvinceDatas()
- {
- List<ProvinceModel> provinceList = null;
- AssetManager asset = getAssets();
- try {
- InputStream input = asset.open("province_data.xml");
- // 创建一个解析xml的工厂对象
- SAXParserFactory spf = SAXParserFactory.newInstance();
- // 解析xml
- SAXParser parser = spf.newSAXParser();
- XmlParserHandler handler = new XmlParserHandler();
- parser.parse(input, handler);
- input.close();
- // 获取解析出来的数据
- provinceList = handler.getDataList();
- //*/ 初始化默认选中的省、市、区
- if (provinceList!= null && !provinceList.isEmpty()) {
- mCurrentProviceName = provinceList.get(0).getName();
- List<CityModel> cityList = provinceList.get(0).getCityList();
- if (cityList!= null && !cityList.isEmpty()) {
- mCurrentCityName = cityList.get(0).getName();
- List<DistrictModel> districtList = cityList.get(0).getDistrictList();
- mCurrentDistrictName = districtList.get(0).getName();
- mCurrentZipCode = districtList.get(0).getZipcode();
- }
- }
- //*/
- mProvinceDatas = new String[provinceList.size()];
- for (int i=0; i< provinceList.size(); i++) {
- mProvinceDatas[i] = provinceList.get(i).getName();
- List<CityModel> cityList = provinceList.get(i).getCityList();
- String[] cityNames = new String[cityList.size()];
- for (int j=0; j< cityList.size(); j++) {
- cityNames[j] = cityList.get(j).getName();
- List<DistrictModel> districtList = cityList.get(j).getDistrictList();
- String[] distrinctNameArray = new String[districtList.size()];
- DistrictModel[] distrinctArray = new DistrictModel[districtList.size()];
- for (int k=0; k<districtList.size(); k++) {
- DistrictModel districtModel = new DistrictModel(districtList.get(k).getName(), districtList.get(k).getZipcode());
- mZipcodeDatasMap.put(districtList.get(k).getName(), districtList.get(k).getZipcode());
- distrinctArray[k] = districtModel;
- distrinctNameArray[k] = districtModel.getName();
- }
- mDistrictDatasMap.put(cityNames[j], distrinctNameArray);
- }
- mCitisDatasMap.put(provinceList.get(i).getName(), cityNames);
- }
- } catch (Throwable e) {
- e.printStackTrace();
- } finally {
- }
- }
在使用wheel组件时,数据适配起来也很方便,只需要做些数据、显示数量的配置即可,我这边设置了一行显示7条数据
- initProvinceDatas();
- mViewProvince.setViewAdapter(new ArrayWheelAdapter<String>(MainActivity.this, mProvinceDatas));
- // 设置可见条目数量
- mViewProvince.setVisibleItems(7);
- mViewCity.setVisibleItems(7);
- mViewDistrict.setVisibleItems(7);
- updateCities();
- updateAreas();
要监听wheel组件的滑动、点击、选中改变事件,可以通过实现它的三个事件监听接口来实现,分别是:
1、OnWheelScrollListener 滑动事件:
- /**
- * Wheel scrolled listener interface.
- */
- public interface OnWheelScrollListener {
- /**
- * Callback method to be invoked when scrolling started.
- * @param wheel the wheel view whose state has changed.
- */
- void onScrollingStarted(WheelView wheel);
- /**
- * Callback method to be invoked when scrolling ended.
- * @param wheel the wheel view whose state has changed.
- */
- void onScrollingFinished(WheelView wheel);
- }
2、OnWheelClickedListener 条目点击事件:
- /**
- * Wheel clicked listener interface.
- * <p>The onItemClicked() method is called whenever a wheel item is clicked
- * <li> New Wheel position is set
- * <li> Wheel view is scrolled
- */
- public interface OnWheelClickedListener {
- /**
- * Callback method to be invoked when current item clicked
- * @param wheel the wheel view
- * @param itemIndex the index of clicked item
- */
- void onItemClicked(WheelView wheel, int itemIndex);
- }
3、OnWheelChangedListener 被选中项的positon变化事件:
- /**
- * Wheel changed listener interface.
- * <p>The onChanged() method is called whenever current wheel positions is changed:
- * <li> New Wheel position is set
- * <li> Wheel view is scrolled
- */
- public interface OnWheelChangedListener {
- /**
- * Callback method to be invoked when current item changed
- * @param wheel the wheel view whose state has changed
- * @param oldValue the old value of current item
- * @param newValue the new value of current item
- */
- void onChanged(WheelView wheel, int oldValue, int newValue);
- }
这里只要知道哪个省、市、区被选中了,实现第三个接口就行,在方法回调时去作同步和更新数据,比如省级条目滑动的时候,市级和县级数据都要做对应的适配、市级滑动时需要去改变县级(区)的数据,这样才能实现级联的效果,至于如何改变,需要三个HashMap来分别保存他们的对应关系:
- /**
- * key - 省 value - 市
- */
- protected Map<String, String[]> mCitisDatasMap = new HashMap<String, String[]>();
- /**
- * key - 市 values - 区
- */
- protected Map<String, String[]> mDistrictDatasMap = new HashMap<String, String[]>();
- /**
- * key - 区 values - 邮编
- */
- protected Map<String, String> mZipcodeDatasMap = new HashMap<String, String>();
在onChanged()回调方法中,对于省、市、区/县的滑动,分别做数据的适配,代码如下:
- @Override
- public void onChanged(WheelView wheel, int oldValue, int newValue) {
- // TODO Auto-generated method stub
- if (wheel == mViewProvince) {
- updateCities();
- } else if (wheel == mViewCity) {
- updateAreas();
- } else if (wheel == mViewDistrict) {
- mCurrentDistrictName = mDistrictDatasMap.get(mCurrentCityName)[newValue];
- mCurrentZipCode = mZipcodeDatasMap.get(mCurrentDistrictName);
- }
- }
- /**
- * 根据当前的市,更新区WheelView的信息
- */
- private void updateAreas() {
- int pCurrent = mViewCity.getCurrentItem();
- mCurrentCityName = mCitisDatasMap.get(mCurrentProviceName)[pCurrent];
- String[] areas = mDistrictDatasMap.get(mCurrentCityName);
- if (areas == null) {
- areas = new String[] { "" };
- }
- mViewDistrict.setViewAdapter(new ArrayWheelAdapter<String>(this, areas));
- mViewDistrict.setCurrentItem(0);
- }
- /**
- * 根据当前的省,更新市WheelView的信息
- */
- private void updateCities() {
- int pCurrent = mViewProvince.getCurrentItem();
- mCurrentProviceName = mProvinceDatas[pCurrent];
- String[] cities = mCitisDatasMap.get(mCurrentProviceName);
- if (cities == null) {
- cities = new String[] { "" };
- }
- mViewCity.setViewAdapter(new ArrayWheelAdapter<String>(this, cities));
- mViewCity.setCurrentItem(0);
- updateAreas();
- }
综上代码,最终实现的界面截图:
源码下载地址:http://download.csdn.net/detail/wulianghuan/8205211
android仿iphone的地区选择的更多相关文章
- Android仿iPhone 滚轮控件 实现
Android_开发 实用滚轮效果选择数字http://blog.csdn.net/zhangtengyuan23/article/details/8653771 Android仿iPhone滚轮控件 ...
- Android 仿土巴兔选择效果
1,前两天在群里看到有人在讨论土巴兔的选择装修风格的效果,自己也想实现,果断百度一下,有些好的文章,就花了些时间来分析了下,先看看别人土巴兔原装的功能 2,可以看到,基本上可以使用一个vviewpag ...
- Android仿iPhone晃动撤销输入功能(微信摇一摇功能)
重力传感器微信摇一摇SensorMannager自定义alertdialogSensorEventListener 很多程序中我们可能会输入长文本内容,比如短信,写便笺等,如果想一次性撤销所有的键入内 ...
- Android 仿iPhone的日期时间选择器
可选只选择日期,也可以同时选择时间 只选择日期的情况 同时选择日期和时间的情况 关键代码: findViewById(R.id.selectDateButton).setOnClickListener ...
- Android 轻松实现仿淘宝地区选择
介绍 最近用淘宝客户端的时候,编辑地址的时候有个地区选择的功能.看上面的效果觉得挺酷,滚动的时候,是最后一个从下面飞上来挨着前一个.就自己鼓捣一个出来玩玩. 说了效果可能不太直观,下面上两张图看看效果 ...
- Android 仿PhotoShop调色板应用(四) 不同区域颜色选择的颜色生成响应
版权声明:本文为博主原创文章,未经博主允许不得转载. Android 仿PhotoShop调色板应用(四) 不同区域颜色选择的颜色生成响应 上一篇讲过了主体界面的绘制,这里讲解调色板应用中的另外一 ...
- 仿iphone日历插件(beta)
前言 小伙伴们好,很久不见了.最近工作进入正常期了,所以慢慢的悠闲的时间久没有了,所以不能每天水一篇了. 最近也在听师傅(http://home.cnblogs.com/u/aaronjs/)的教导开 ...
- 转-Android仿微信气泡聊天界面设计
微信的气泡聊天是仿iPhone自带短信而设计出来的,不过感觉还不错可以尝试一下仿着微信的气泡聊天做一个Demo,给大家分享一下!效果图如下: 气泡聊天最终要的是素材,要用到9.png文件的素材,这样气 ...
- Android 仿PhotoShop调色板应用(三) 主体界面绘制
版权声明:本文为博主原创文章,未经博主允许不得转载. Android 仿PhotoShop调色板应用(三) 主体界面绘制 关于PhotoShop调色板应用的实现我总结了两个最核心的部分: 1 ...
随机推荐
- Notepad++连接Centos
Notepad++设置 插件 -- > plugin Manager --> show plugin manager --> NppFtp 安装重启notepad++ 插件 --& ...
- WPF 实现换肤功能
将所有控件的基本样式汇集到一个资源字典中,构成界面的基本样式文件,然后进行不同颜色皮肤的定制. 即在新的皮肤资源字典文件中引入基本样式文件,然后使用资源继承,并且只设置控件的颜色属性等,形成一个皮肤文 ...
- 自然语言处理工具:中文 word2vec 开源项目,教程,数据集
word2vec word2vec/glove/swivel binary file on chinese corpus word2vec: https://code.google.com/p/wor ...
- thymeleaf:局部变量 th:with
当th:with被处理,firstPer变量创建一个局部变量和变量添加到map自上下文,以便它是用于评估和其他上下文中声明的变量从开始,但只有包含< div >标记的范围内. <di ...
- @RequestBody和@RequestParam区别
@RequestParam 用来处理Content-Type: 为 application/x-www-form-urlencoded编码的内容.(Http协议中,默认传递的参数就是applicati ...
- 【集合框架】JDK1.8源码分析之HashMap(一) 转载
[集合框架]JDK1.8源码分析之HashMap(一) 一.前言 在分析jdk1.8后的HashMap源码时,发现网上好多分析都是基于之前的jdk,而Java8的HashMap对之前做了较大的优化 ...
- 在移动端画出真正的1px边框
一.问题 写H5的样式时候,设置元素的边框为1px,不幸的事情在IOS设备上发生了,设计师会说,咦,边框怎么那么大,这是2px了吧?改成1px.我明明设置成1px了啊. 二.为什么边框变粗了? ...
- Response ServletContext 中文乱码 Request 编码 请求行 共享数据 转发重定向
Day35 Response 1.1.1 ServletContext概念 u 项目的管理者(上下文对象),服务器启动时,会为每一个项目创建一个对应的ServletContext对象. 1.1.2 ...
- 03 持续集成和部署/基础设施 - DevOps之路
02 持续集成和部署/基础设施 - DevOps之路 文章Github地址,欢迎start:https://github.com/li-keli/DevOps-WiKi 服务的持续集成和部署这里有两套 ...
- Spring Boot Cache Redis缓存
1.集成MyBatis 1.1.引入maven依赖 1.2.生成Mapper 具体可以看MyBatis Generator官网 http://www.mybatis.org/generator/run ...