Android Setting 日期相关API,自定义DatePicker

2024-05-30 12:39

本文主要是介绍Android Setting 日期相关API,自定义DatePicker,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在做Setting日期相关的工作,整理一下相关的API(相关代码源自Android 8.0高通代码)

注意:以下API来自Setting,使用的地方也在Setting中,如果其他非系统应用使用,可能需要加权限

设置日期API

void setDate(int year, int month, int day) {Calendar c = Calendar.getInstance();Log.d(TAG, "setDate: year " + year + " month " + month + " day " + day);c.set(Calendar.YEAR, year);c.set(Calendar.MONTH, month);c.set(Calendar.DAY_OF_MONTH, day);long when = Math.max(c.getTimeInMillis(), DatePreferenceController.DatePreferenceHost.MIN_DATE);if (when / 1000 < Integer.MAX_VALUE) {((AlarmManager) mContext.getSystemService(Context.ALARM_SERVICE)).setTime(when);}
}

DatePreferenceController.DatePreferenceHost.MIN_DATE:是一个长整型常量,表示日期可以设置的最小值。

DateFormat API

private void set24Hour(boolean is24Hour) {Log.d(TAG, "set24Hour: ");Settings.System.putString(mContext.getContentResolver(),Settings.System.TIME_12_24,is24Hour ? HOURS_24 : HOURS_12);
}

指定12小时 24小时制

获取当前DateFormat API

public boolean is24Hour() {Log.d(TAG, "is24Hour: ");return DateFormat.is24HourFormat(mContext);
}

设置时间API

void setTime(int hourOfDay, int minute) {Calendar c = Calendar.getInstance();c.set(Calendar.HOUR_OF_DAY, hourOfDay);c.set(Calendar.MINUTE, minute);c.set(Calendar.SECOND, 0);c.set(Calendar.MILLISECOND, 0);long when = Math.max(c.getTimeInMillis(), TimePreferenceHost.MIN_DATE);if (when / 1000 < Integer.MAX_VALUE) {((AlarmManager) mContext.getSystemService(Context.ALARM_SERVICE)).setTime(when);}
}

设置时区API

final AlarmManager alarm = (AlarmManager) activity.getSystemService(Context.ALARM_SERVICE);
alarm.setTimeZone(tzId);

这里tzId是一个字符串,形式为 大洲/City
比如Asia/Shanghai
在Android 8.0高通源码中一共有89个城市,设置的tzId必须在这89个数据中,否则底层会抛出异常,找不到指定的时区,应用crash

自定义DatePicker

由于客户给出的设计图和Android原生的DatePicker相差过大,只能自定义DatePicker了。
注意:这里使用的NumberPicker在framework层的源码做过少量修改。设计图给出的样式比较丑,还没有原生的漂亮。给出的设计大概是这样的
这里写图片描述

Java代码

package com.android.settings.datetime;import android.app.AlarmManager;
import android.app.AlertDialog;
import android.content.Context;
import android.os.Build;
import android.support.annotation.NonNull;
import android.support.annotation.RequiresApi;
import android.util.Log;
import android.view.KeyEvent;
import android.view.ViewGroup.LayoutParams;
import android.view.WindowManager;
import android.widget.NumberPicker;import com.android.settings.R;import java.util.Calendar;/*** Created by caihuijian on 18-5-18.*/public class CustomDatePickerDialog extends AlertDialog {private static final String TAG = "CustomDatePickerDialog";private NumberPicker mNumberYear;private NumberPicker mNumberMonth;private NumberPicker mNumberDay;private static final int MAX_YEAR = 37;//暂定 系统某处定义了private static final int MIN_YEAR = 7;//暂定 系统某处定义了private static final int BIG_MONTH_DAY = 31;//大月31天private static final int SMALL_MONTH_DAY = 30;//小月30天private static final int LEAP_FEB_DAY = 29;//闰年二月29天private static final int NOT_LEAP_FEB_DAY = 28;//平年二月28天private static final int MIN_DAY = 1;//最小日private static final int MAX_MONTH = 12;//最大月数private static final int MIN_MONTH = 1;//最小月数private Context mContext = null;final Calendar calendar = Calendar.getInstance();protected CustomDatePickerDialog(@NonNull Context context) {super(context);this.mContext = context;}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)@Overridepublic void show() {super.show();//全屏dialogWindowManager.LayoutParams layoutParams = getWindow().getAttributes();layoutParams.width = LayoutParams.MATCH_PARENT;layoutParams.height = LayoutParams.MATCH_PARENT;getWindow().getDecorView().setPadding(0, 0, 0, 0);getWindow().setAttributes(layoutParams);getWindow().getDecorView().setBackgroundResource(0x00000000);setCustomLayout();}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private void setCustomLayout() {setContentView(R.layout.cus_date_picker_dialog);//控件初始化mNumberMonth = (NumberPicker) findViewById(R.id.month);mNumberDay = (NumberPicker) findViewById(R.id.day);mNumberYear = (NumberPicker) findViewById(R.id.year);setViewMaxMin();initAllView();mNumberYear.setOnValueChangedListener(yearChangeListener);mNumberMonth.setOnValueChangedListener(monthChangeListener);mNumberDay.setOnValueChangedListener(dayChangeListener);}//取得当前时间 设置到view@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private void initAllView() {int year, monthOfYear, dayOfMonth;if (calendar != null) {//因为控件只显示年的后两位 所以对100取余year = calendar.get(Calendar.YEAR) % 100;monthOfYear = calendar.get(Calendar.MONTH);dayOfMonth = calendar.get(Calendar.DAY_OF_MONTH);mNumberYear.setValue(year);mNumberMonth.setValue(monthOfYear);mNumberDay.setValue(dayOfMonth);}}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private int getMonthDay(int month) {switch (month) {case 1:case 3:case 5:case 7:case 8:case 10:case 12:return BIG_MONTH_DAY;case 4:case 6:case 9:case 11:return SMALL_MONTH_DAY;case 2:int year = mNumberYear.getValue();boolean isLeap = isLeapYear(year);if (isLeap) {return LEAP_FEB_DAY;} else {return NOT_LEAP_FEB_DAY;}}return 0;}//UI中,年只有2位,可以使用这两位进行闰年判断boolean isLeapYear(int year) {if (year % 4 == 0 && year % 100 != 0 || year % 400 == 0) {Log.d(TAG, "isLeapYear: " + year + "年是闰年");return true;} else {Log.d(TAG, "isLeapYear: " + year + "年不是闰年");return false;}}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private void setViewMaxMin() {mNumberMonth.setMaxValue(MAX_MONTH);mNumberMonth.setMinValue(MIN_MONTH);mNumberDay.setMaxValue(BIG_MONTH_DAY);mNumberDay.setMinValue(MIN_DAY);mNumberYear.setMaxValue(MAX_YEAR);mNumberYear.setMinValue(MIN_YEAR);}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)@Overridepublic boolean onKeyUp(int keyCode, KeyEvent event) {switch (keyCode) {case KeyEvent.KEYCODE_DPAD_CENTER:if (mNumberYear.isFocused()) {mNumberMonth.requestFocus();} else if (mNumberMonth.isFocused()) {mNumberDay.requestFocus();} else if (mNumberDay.isFocused()) {setDate(2000 + mNumberYear.getValue(), mNumberMonth.getValue(), mNumberDay.getValue());this.dismiss();}return true;case KeyEvent.KEYCODE_BACK:if (mNumberDay.isFocused()) {mNumberMonth.requestFocus();return true;} else if (mNumberMonth.isFocused()) {mNumberYear.requestFocus();return true;}}//手动输入数字 填充年月日if (keyCode >= KeyEvent.KEYCODE_0 && keyCode <= KeyEvent.KEYCODE_9) {int singleDigit = keyCode - KeyEvent.KEYCODE_0;Log.d(TAG, "onKeyUp: keyCode" + keyCode);if (mNumberYear.hasFocus()) {Log.d(TAG, "onKeyUp: year");//手动输入不会触发ValueChangeListener,需要自己处理int yearChangeTo = manualInput(mNumberYear, singleDigit);yearTextChange(yearChangeTo);} else if (mNumberMonth.hasFocus()) {Log.d(TAG, "onKeyUp: month");//手动输入不会触发ValueChangeListener,需要自己处理int monthChange = manualInput(mNumberMonth, singleDigit);monthTextChange(monthChange);} else if (mNumberDay.hasFocus()) {manualInput(mNumberDay, singleDigit);}return true;}return super.onKeyUp(keyCode, event);}@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private int manualInput(NumberPicker numberPicker, int userInput) {//手动输入逻辑://取个位数放到十位上//输入值作为个位数//十位 个位相加//判断该值是否在最大与最小值之间 小于最小值取最小值 大于最大值取最大值 十位取最大值的十位数,个位为用户输入int temp = numberPicker.getValue();int value = temp % 10 * 10 + userInput;int maxValue = numberPicker.getMaxValue();int minValue = numberPicker.getMinValue();if (value < minValue) {value = minValue;} else if (value > maxValue) {value = maxValue - maxValue % 10 + userInput;if(value > maxValue){value = maxValue;}}Log.d(TAG, "manualInput: value" + value);numberPicker.setValue(value);return value;}//设置日期API@RequiresApi(api = Build.VERSION_CODES.FROYO)void setDate(int year, int month, int day) {Calendar c = Calendar.getInstance();Log.d(TAG, "setDate: year " + year + " month " + month + " day " + day);c.set(Calendar.YEAR, year);c.set(Calendar.MONTH, month);c.set(Calendar.DAY_OF_MONTH, day);long when = Math.max(c.getTimeInMillis(), DatePreferenceController.DatePreferenceHost.MIN_DATE);if (when / 1000 < Integer.MAX_VALUE) {((AlarmManager) mContext.getSystemService(Context.ALARM_SERVICE)).setTime(when);}}NumberPicker.OnValueChangeListener yearChangeListener = new NumberPicker.OnValueChangeListener() {@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)@Overridepublic void onValueChange(NumberPicker numberPicker, int yearOld, int yearNew) {yearTextChange(yearNew);}};NumberPicker.OnValueChangeListener monthChangeListener = new NumberPicker.OnValueChangeListener() {@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)@Overridepublic void onValueChange(NumberPicker numberPicker, int monthOld, int monthNew) {monthTextChange(monthNew);}};NumberPicker.OnValueChangeListener dayChangeListener = new NumberPicker.OnValueChangeListener() {@Overridepublic void onValueChange(NumberPicker numberPicker, int dayOld, int dayNew) {}};//年的控件变化需要做的处理@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)public void yearTextChange(int yearNew) {//年的变化会导致2月是否有29日变化boolean isLeapYear = isLeapYear(yearNew);int month = mNumberMonth.getValue();if (isLeapYear && month == 2) {//闰年二月mNumberDay.setMaxValue(29);} else if (!isLeapYear && month == 2) {//非闰年二月mNumberDay.setMaxValue(28);}}//月的控件变化需要做的处理@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)private void monthTextChange(int monthNew) {int monthDay = getMonthDay(monthNew);mNumberDay.setMaxValue(monthDay);}}

xml

<?xml version="1.0" encoding="utf-8"?><!-- Layout of time picker-->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:background="#1E1E1E"android:gravity="center"android:orientation="horizontal"><!-- year --><NumberPicker
        android:id="@+id/year"android:layout_width="80dip"android:layout_height="wrap_content"android:custtomSytle="true"android:focusable="true"android:focusableInTouchMode="true" /><!-- divider1 --><TextView
        android:id="@+id/divider1"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:layout_marginEnd="2dip"android:layout_marginStart="2dip"android:importantForAccessibility="no"android:text="/"android:textColor="#808080"android:textSize="55dp" /><!-- month --><NumberPicker
        android:id="@+id/month"android:layout_width="80dip"android:layout_height="wrap_content"android:custtomSytle="true"android:focusable="true"android:focusableInTouchMode="true" /><!-- divider2 --><TextView
        android:id="@+id/divider2"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:layout_marginEnd="2dip"android:layout_marginStart="2dip"android:importantForAccessibility="no"android:text="/"android:textColor="#808080"android:textSize="55dp" /><!-- day --><NumberPicker
        android:id="@+id/day"android:layout_width="80dip"android:layout_height="wrap_content"android:custtomSytle="true"android:focusable="true"android:focusableInTouchMode="true" />
</LinearLayout>

使用

                CustomDatePickerDialog datePickerDialog = new CustomDatePickerDialog(getActivity());datePickerDialog.show();

这篇关于Android Setting 日期相关API,自定义DatePicker的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1016229

相关文章

RecastNavigation之Poly相关类

Poly分成正常的Poly 和 OffMeshPoly。 正常的Poly 又分成 原始的Poly 和 Detail化的Poly,本文介绍这两种。 Poly的边分成三种类型: 1. 正常边:有tile内部的poly与之相邻 2.border边:没有poly与之相邻 3.Portal边:与之相邻的是外部tile的poly   由firstLink索引 得到第一个连接的Poly  通

通过高德api查询所有店铺地址信息

通过高德api查询所有店铺地址电话信息 需求:通过高德api查询所有店铺地址信息需求分析具体实现1、申请高德appkey2、下载types city 字典值3、具体代码调用 需求:通过高德api查询所有店铺地址信息 需求分析 查询现有高德api发现现有接口关键字搜索API服务地址: https://developer.amap.com/api/webservice/gui

ROS话题通信流程自定义数据格式

ROS话题通信流程自定义数据格式 需求流程实现步骤定义msg文件编辑配置文件编译 在 ROS 通信协议中,数据载体是一个较为重要组成部分,ROS 中通过 std_msgs 封装了一些原生的数据类型,比如:String、Int32、Int64、Char、Bool、Empty… 但是,这些数据一般只包含一个 data 字段,结构的单一意味着功能上的局限性,当传输一些复杂的数据,比如:

DDei在线设计器-API-DDeiSheet

DDeiSheet   DDeiSheet是代表一个页签,一个页签含有一个DDeiStage用于显示图形。   DDeiSheet实例包含了一个页签的所有数据,在获取后可以通过它访问其他内容。DDeiFile中的sheets属性记录了当前文件的页签列表。   一个DDeiFile实例至少包含一个DDeiSheet实例。   本篇最后提供的示例可以在DDei文档直接预览 属性 属性名说明数

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st

Android多线程下载见解

通过for循环开启N个线程,这是多线程,但每次循环都new一个线程肯定很耗内存的。那可以改用线程池来。 就以我个人对多线程下载的理解是开启一个线程后: 1.通过HttpUrlConnection对象获取要下载文件的总长度 2.通过RandomAccessFile流对象在本地创建一个跟远程文件长度一样大小的空文件。 3.通过文件总长度/线程个数=得到每个线程大概要下载的量(线程块大小)。