android实现双日期选择控件(可隐藏日,只显示年月)

在安卓开发中,会碰到选开始日期和结束日期的问题。特别是在使用Pad时,如果弹出一个Dialog,能够同时选择开始日期和结束日期,那将是极好的。我在开发中在DatePickerDialog的基础上做了修改,实现了这种Dialog。效果如下:

具体实现方法为:

先新建一个安卓项目DoubleDatePicker,在res/layout文件夹下新建date_picker_dialog.xml,内容如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:gravity="center_horizontal"
  android:orientation="horizontal"
  android:paddingTop="10dp" >

  <LinearLayout
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:padding="5dip" >

    <TextView
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="开始日期" />

    <DatePicker
      android:id="@+id/datePickerStart"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:calendarViewShown="false" />
  </LinearLayout>

  <ImageView
    android:layout_width="wrap_content"
    android:layout_height="fill_parent"
    android:src="@drawable/fenge" />

  <LinearLayout
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:gravity="center_horizontal"
    android:orientation="vertical"
    android:padding="5dip" >

    <TextView
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="结束日期" />

    <DatePicker
      android:id="@+id/datePickerEnd"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:calendarViewShown="false" />
  </LinearLayout>

</LinearLayout>

然后,在src的 默认包下新建文件DoubleDatePickerDialog.java,内容如下:

/*
 * Copyright (C) 2007 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.example.doubledatepicker;

import java.lang.reflect.Field;

import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnClickListener;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.DatePicker;
import android.widget.DatePicker.OnDateChangedListener;

/**
 * A simple dialog containing an {@link android.widget.DatePicker}.
 *
 * <p>
 * See the <a href="{@docRoot}guide/topics/ui/controls/pickers.html">Pickers</a>
 * guide.
 * </p>
 */
public class DoubleDatePickerDialog extends AlertDialog implements OnClickListener, OnDateChangedListener {

  private static final String START_YEAR = "start_year";
  private static final String END_YEAR = "end_year";
  private static final String START_MONTH = "start_month";
  private static final String END_MONTH = "end_month";
  private static final String START_DAY = "start_day";
  private static final String END_DAY = "end_day";

  private final DatePicker mDatePicker_start;
  private final DatePicker mDatePicker_end;
  private final OnDateSetListener mCallBack;

  /**
   * The callback used to indicate the user is done filling in the date.
   */
  public interface OnDateSetListener {

    /**
     * @param view
     *      The view associated with this listener.
     * @param year
     *      The year that was set.
     * @param monthOfYear
     *      The month that was set (0-11) for compatibility with
     *      {@link java.util.Calendar}.
     * @param dayOfMonth
     *      The day of the month that was set.
     */
    void onDateSet(DatePicker startDatePicker, int startYear, int startMonthOfYear, int startDayOfMonth,
        DatePicker endDatePicker, int endYear, int endMonthOfYear, int endDayOfMonth);
  }

  /**
   * @param context
   *      The context the dialog is to run in.
   * @param callBack
   *      How the parent is notified that the date is set.
   * @param year
   *      The initial year of the dialog.
   * @param monthOfYear
   *      The initial month of the dialog.
   * @param dayOfMonth
   *      The initial day of the dialog.
   */
  public DoubleDatePickerDialog(Context context, OnDateSetListener callBack, int year, int monthOfYear, int dayOfMonth) {
    this(context, 0, callBack, year, monthOfYear, dayOfMonth);
  }

  public DoubleDatePickerDialog(Context context, int theme, OnDateSetListener callBack, int year, int monthOfYear,
      int dayOfMonth) {
    this(context, 0, callBack, year, monthOfYear, dayOfMonth, true);
  }

  /**
   * @param context
   *      The context the dialog is to run in.
   * @param theme
   *      the theme to apply to this dialog
   * @param callBack
   *      How the parent is notified that the date is set.
   * @param year
   *      The initial year of the dialog.
   * @param monthOfYear
   *      The initial month of the dialog.
   * @param dayOfMonth
   *      The initial day of the dialog.
   */
  public DoubleDatePickerDialog(Context context, int theme, OnDateSetListener callBack, int year, int monthOfYear,
      int dayOfMonth, boolean isDayVisible) {
    super(context, theme);

    mCallBack = callBack;

    Context themeContext = getContext();
    setButton(BUTTON_POSITIVE, "确 定", this);
    setButton(BUTTON_NEGATIVE, "取 消", this);
    // setButton(BUTTON_POSITIVE,
    // themeContext.getText(android.R.string.date_time_done), this);
    setIcon(0);

    LayoutInflater inflater = (LayoutInflater) themeContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    View view = inflater.inflate(R.layout.date_picker_dialog, null);
    setView(view);
    mDatePicker_start = (DatePicker) view.findViewById(R.id.datePickerStart);
    mDatePicker_end = (DatePicker) view.findViewById(R.id.datePickerEnd);
    mDatePicker_start.init(year, monthOfYear, dayOfMonth, this);
    mDatePicker_end.init(year, monthOfYear, dayOfMonth, this);
    // updateTitle(year, monthOfYear, dayOfMonth);

    // 如果要隐藏当前日期,则使用下面方法。
    if (!isDayVisible) {
      hidDay(mDatePicker_start);
      hidDay(mDatePicker_end);
    }
  }

  /**
   * 隐藏DatePicker中的日期显示
   *
   * @param mDatePicker
   */
  private void hidDay(DatePicker mDatePicker) {
    Field[] datePickerfFields = mDatePicker.getClass().getDeclaredFields();
    for (Field datePickerField : datePickerfFields) {
      if ("mDaySpinner".equals(datePickerField.getName())) {
        datePickerField.setAccessible(true);
        Object dayPicker = new Object();
        try {
          dayPicker = datePickerField.get(mDatePicker);
        } catch (IllegalAccessException e) {
          e.printStackTrace();
        } catch (IllegalArgumentException e) {
          e.printStackTrace();
        }
        // datePicker.getCalendarView().setVisibility(View.GONE);
        ((View) dayPicker).setVisibility(View.GONE);
      }
    }
  }

  public void onClick(DialogInterface dialog, int which) {
    // Log.d(this.getClass().getSimpleName(), String.format("which:%d",
    // which));
    // 如果是“取 消”按钮,则返回,如果是“确 定”按钮,则往下执行
    if (which == BUTTON_POSITIVE)
      tryNotifyDateSet();
  }

  @Override
  public void onDateChanged(DatePicker view, int year, int month, int day) {
    if (view.getId() == R.id.datePickerStart)
      mDatePicker_start.init(year, month, day, this);
    if (view.getId() == R.id.datePickerEnd)
      mDatePicker_end.init(year, month, day, this);
    // updateTitle(year, month, day);
  }

  /**
   * 获得开始日期的DatePicker
   *
   * @return The calendar view.
   */
  public DatePicker getDatePickerStart() {
    return mDatePicker_start;
  }

  /**
   * 获得结束日期的DatePicker
   *
   * @return The calendar view.
   */
  public DatePicker getDatePickerEnd() {
    return mDatePicker_end;
  }

  /**
   * Sets the start date.
   *
   * @param year
   *      The date year.
   * @param monthOfYear
   *      The date month.
   * @param dayOfMonth
   *      The date day of month.
   */
  public void updateStartDate(int year, int monthOfYear, int dayOfMonth) {
    mDatePicker_start.updateDate(year, monthOfYear, dayOfMonth);
  }

  /**
   * Sets the end date.
   *
   * @param year
   *      The date year.
   * @param monthOfYear
   *      The date month.
   * @param dayOfMonth
   *      The date day of month.
   */
  public void updateEndDate(int year, int monthOfYear, int dayOfMonth) {
    mDatePicker_end.updateDate(year, monthOfYear, dayOfMonth);
  }

  private void tryNotifyDateSet() {
    if (mCallBack != null) {
      mDatePicker_start.clearFocus();
      mDatePicker_end.clearFocus();
      mCallBack.onDateSet(mDatePicker_start, mDatePicker_start.getYear(), mDatePicker_start.getMonth(),
          mDatePicker_start.getDayOfMonth(), mDatePicker_end, mDatePicker_end.getYear(),
          mDatePicker_end.getMonth(), mDatePicker_end.getDayOfMonth());
    }
  }

  @Override
  protected void onStop() {
    // tryNotifyDateSet();
    super.onStop();
  }

  @Override
  public Bundle onSaveInstanceState() {
    Bundle state = super.onSaveInstanceState();
    state.putInt(START_YEAR, mDatePicker_start.getYear());
    state.putInt(START_MONTH, mDatePicker_start.getMonth());
    state.putInt(START_DAY, mDatePicker_start.getDayOfMonth());
    state.putInt(END_YEAR, mDatePicker_end.getYear());
    state.putInt(END_MONTH, mDatePicker_end.getMonth());
    state.putInt(END_DAY, mDatePicker_end.getDayOfMonth());
    return state;
  }

  @Override
  public void onRestoreInstanceState(Bundle savedInstanceState) {
    super.onRestoreInstanceState(savedInstanceState);
    int start_year = savedInstanceState.getInt(START_YEAR);
    int start_month = savedInstanceState.getInt(START_MONTH);
    int start_day = savedInstanceState.getInt(START_DAY);
    mDatePicker_start.init(start_year, start_month, start_day, this);

    int end_year = savedInstanceState.getInt(END_YEAR);
    int end_month = savedInstanceState.getInt(END_MONTH);
    int end_day = savedInstanceState.getInt(END_DAY);
    mDatePicker_end.init(end_year, end_month, end_day, this);

  }
}

这些代码是以DatePickerDialog.java为基础修改的。总的来说,阅读源码是一种好习惯。这里面最需要注意的是hidDay方法,该方法如果调用,则隐藏“日”的选择框,只能选择“年月”。这个方法的实现也比较有难度,需要通过反射,找出DatePicker中表示日的字段,并将其设置为隐藏。

还有一点需要注意的是,为了让控件显示更加好看,我用了一张名字为fenge.png的图片,图片在我提供的源码中可以找到。

下面就需要编辑activity_main.xml了,这个内容相当简单,只要一个显示的text和一个button即可,代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/LinearLayout01"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"
  android:orientation="vertical" >

  <EditText
    android:id="@+id/et"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:cursorVisible="false"
    android:editable="false" />

  <Button
    android:id="@+id/dateBtn"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:text="日期对话框" />

</LinearLayout>

最后,在MainActivity.java中,加入测试代码:

package com.example.doubledatepicker;

import java.util.Calendar;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.DatePicker;
import android.widget.TextView;

public class MainActivity extends Activity {

  Button btn;
  TextView et;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    btn = (Button) findViewById(R.id.dateBtn);
    et = (TextView) findViewById(R.id.et);

    btn.setOnClickListener(new View.OnClickListener() {
      Calendar c = Calendar.getInstance();

      @Override
      public void onClick(View v) {
        // 最后一个false表示不显示日期,如果要显示日期,最后参数可以是true或者不用输入
        new DoubleDatePickerDialog(MainActivity.this, 0, new DoubleDatePickerDialog.OnDateSetListener() {

          @Override
          public void onDateSet(DatePicker startDatePicker, int startYear, int startMonthOfYear,
              int startDayOfMonth, DatePicker endDatePicker, int endYear, int endMonthOfYear,
              int endDayOfMonth) {
            String textString = String.format("开始时间:%d-%d-%d\n结束时间:%d-%d-%d\n", startYear,
                startMonthOfYear + 1, startDayOfMonth, endYear, endMonthOfYear + 1, endDayOfMonth);
            et.setText(textString);
          }
        }, c.get(Calendar.YEAR), c.get(Calendar.MONTH), c.get(Calendar.DATE), true).show();
      }
    });
  }
}

可以看到,在新建DoubleDatePickerDialog时, 我们实现了一个new DoubleDatePickerDialog.OnDateSetListener()的匿名类,这个类被DoubleDatePickerDialog引用,当DoubleDatePickerDialog中的“确 定”按钮被点击时,就会调用匿名类的onDateSet方法。(这也是事件绑定的基本原理)。

DoubleDatePickerDialog构造函数的最后一个参数,true为显示日期,false为不显示日期。

当最后一个参数为true时,显示效果如下:

当最后一个参数为false时,显示如下

源码下载地址:http://xiazai.jb51.net/201701/yuanma/DoubleDatePicker_jb51.rar

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 轻松实现可扩展自定义的Android滚轮时间选择控件

    项目需求中有个功能模块需要用到时间选择控件,但是android系统自带的太丑了,只能自己优化下,结合WheelView实现滚轮选择日期,好像网上也挺多这种文章的.但是适用范围还是不同,希望这个能够对需求相同的朋友有一定帮助.控件标题还有年月日时分秒这些可以自己控制是否显示,先来看效果. 1.有年月日时分的开始时间 2.只有年月日的结束时间 3.用于有时身份证到期的时间选择(分为勾选长期和直接选择时间两种,另外长期后面自己也可以进行扩展) 4.项目结构 5.直接贴代码,代码里面注释很详细 <spa

  • Android高仿IOS 滚轮选择控件

    最近根据项目需要,整理了一个相对比较全面的 WheelView 使用控件,借用之前看到的一句话来说,就是站在巨人肩膀上,进行了一些小调整. 这里先贴上效果图 一般常用的时间选择格式,,单项选择,以及城市联动,这里基本都可以满足了. 这里把 单项选择,和 日期时间选择 给提出到 Util 类中,代码如下: public class Util { /** * 时间选择回调 */ public interface TimerPickerCallBack { void onTimeSelect(Stri

  • 可支持快速搜索筛选的Android自定义选择控件

    Android 自定义支持快速搜索筛选的选择控件使用方法,具体如下 项目中遇到选择控件选项过多,需要快速查找匹配的情况. 做了简单的Demo,效果图如下: 源码地址:https://github.com/whieenz/SearchSelect 这个控件是由Dialog+SearchView+ListView实现的.Dialog用来承载选择控件,SearchView实现输入,ListView展示结果.设计概要图如下: 一.自定义Dialog Dialog布局文件 <?xml version=&quo

  • Android之日期时间选择控件DatePicker和TimePicker实例

    这个月根据需求在项目中做了一个时间选择器,虽然没有用到Android原生的时间选择控件,但我羞愧地发现自己竟然从来没有用过这方面控件!趁现在有时间,赶紧查缺补漏,写一篇博客吧. (注:为了便于区分,本文将选择年月日的控件称为日期选择控件,将选择时分的控件称为时间选择控件.) 1.创建项目 新建一个项目,MainActivity的布局如下: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

  • Android自定义控件之日期选择控件使用详解

    Android日期选择控件效果如下: 调用的代码: @OnClick(R.id.btn0) public void btn0() { final AlertDialog dialog = new AlertDialog.Builder(context).create(); dialog.show(); Window window = dialog.getWindow(); window.setContentView(R.layout.dialog_change_date); window.set

  • Android 实现IOS 滚轮选择控件的实例(源码下载)

     Android 实现IOS 滚轮选择控件的实例 最近根据项目需要,整理了一个相对比较全面的 WheelView 使用控件,借用之前看到的一句话来说,就是站在巨人肩膀上,进行了一些小调整. 这里先贴上效果图 一般常用的时间选择格式,,单项选择,以及城市联动,这里基本都可以满足了. 这里把 单项选择,和 日期时间选择 给提出到 Util 类中,代码如下: public class Util { /** * 时间选择回调 */ public interface TimerPickerCallBack

  • Android自定义View实现多图片选择控件

    前言 相信很多朋友在开发中都会遇到图片上传的情况,尤其是多图上传,最经典的莫过于微信的图片选择了.所有很多情况下会使用到多图选择,所以就有了这篇文章,今天抽点时间写了个控件.  •支持自定义选择图片的样式  •支持设置图片选择数量  •支持图片预览,删除  •支持图片拍照 先来看看效果 实现分析 假如不定义控件,我们要实现这样一个功能,无非是写个GridView在item点击的时候去显示图片进行选择,在返回界面的时候进行GridView的数据刷新.我们把这些逻辑写在我们自定义的GridView中

  • Android实现滑动选择控件实例代码

    前言 最近做了个滑动选择的小控件,拿出来给大家分享一下,先上图 运行效果 实现步骤 这里分解为3个动作:Down.Move.Up来进行分析,博主文采不好,大家直接看流程图吧!! 代码分析 前置知识 1.这个地方使用的是RecyclerView的代码,使用RecyclerView只能使用LinearLayoutManager,ListView的运行效果稍微要比RecyclerView差一些 //这里使用dispatchTouchEvent,因为onTouchEvent容易被OnTouchListe

  • android实现双日期选择控件(可隐藏日,只显示年月)

    在安卓开发中,会碰到选开始日期和结束日期的问题.特别是在使用Pad时,如果弹出一个Dialog,能够同时选择开始日期和结束日期,那将是极好的.我在开发中在DatePickerDialog的基础上做了修改,实现了这种Dialog.效果如下: 具体实现方法为: 先新建一个安卓项目DoubleDatePicker,在res/layout文件夹下新建date_picker_dialog.xml,内容如下: <?xml version="1.0" encoding="utf-8&

  • Android日期选择控件使用详解

    本文实例为大家分享了Android日期选择控件的使用方法,供大家参考,具体内容如下 1.创建dialog 布局 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" an

  • Ext JS框架中日期函数的用法及日期选择控件的实现

    Ext.Date是一个单例,封装了一系列日期操作函数,扩展JavaScript Date的功能,下面列出一些常用的功能. 基本函数: Ext.Date.add(date, interval, value) 给date增加或减少时间,这个函数不改变原有Date对象的值,而是返回一个新的Date对象. Ext.Date.between(date, start, end) 判断date是否在start和end之间. Ext.Date.clearTime(date, clone) 把date的时间设置成

  • Ext JS 4实现带week(星期)的日期选择控件(实战二)

    前言 JavaScript 中的日期和时间 Ext JS 4实现带week(星期)的日期选择控件(实战一) 如对本篇的一些预备知识需详尽了解,可参考以上两篇. Javascript 有提供Date 对象用于处理时间.但是Date 并没有提供获取星期的方法. 要在web 端通过js 方式获取某个时间是这一年的第几个星期,可以根据一些算法去实现. 当然, jquery 的扩展组件 等有直接提供这样的一些现成包. 像Ext js 就有提供获取星期的方法 Ext.Date.getWeekOfYear(d

  • datePicker——日期选择控件(with jquery)

    demo: http://demo.jb51.net/js/2011/jQuery_calendar/index.html down: http://www.jb51.net/jiaoben/19622.html 用法很简单,而且js文件也很小,之前也见过一些日期选择控件,但个头都比较大,影响速度可以设置日期的格式,可以选择日期的起止时间,如果不加参数的话,默认就是之前的日期不可选,而只能从今天开始选择 现在My97 DatePicker也不错,不用jquery 一款很不错的基于JavaScri

  • 基于jQuery的日期选择控件

    但是也有些问题,第一画日历有点慢,第二兼容性不太好IE Only,第三它不是基于jQuery的哈哈. 那还是老规矩,做之前先看下效果   这下是更酷的Ext风格了. 从上图我们可以看出这个控件其实有两个视图一个日期月视图,还有一个是年月选择视图. 1:还是先从HTML入手 日期控件确定HTML其实还是比较简单,因为明摆着是列表的数据格式,当然主要是采用table了. 两个视图分别用两个Div包裹,控制div的显示隐藏即可以切换视图了.完整的HTMl结构大家可以用IEDeveloper看一下Dem

  • Ext JS 4实现带week(星期)的日期选择控件(实战一)

    前言 Ext JS 3 和 Ext JS 4中都有提供日期选择的组件(当然早期版本也有). 但是有一些日期选择的需求是要看到星期,就是日期中的哪一天是这一年的第几周. 遗憾的是Ext js 并没有提供这样的配置. (针对Ext js 4来说,理想的相法是在Ext.picker.Date有一个 类似showWeek这样的配置项) 现有的解法 到网络上去看看基于Ext js 的解法:有找到两个 http://enikao.net/extjs/weeknumber/weeknumber.html 和

随机推荐