① 安卓 怎樣自定義一個時間選擇器
實現的效果就是在同一個布局上顯示日期選擇和時間選擇。
自定義一個類:
/DateTimePicker/src/com/wwj/datetimepicker/DateTimePickDialogUtil.java
package com.wwj.datetimepicker;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.widget.DatePicker;
import android.widget.DatePicker.OnDateChangedListener;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.TimePicker;
import android.widget.TimePicker.OnTimeChangedListener;
/**
* 日期時間選擇控制項 使用方法: private EditText inputDate;//需要設置的日期時間文本編輯框 private String
* initDateTime="2012年9月3日 14:44",//初始日期時間值 在點擊事件中使用:
* inputDate.setOnClickListener(new OnClickListener() {
*
* @Override public void onClick(View v) { DateTimePickDialogUtil
* dateTimePicKDialog=new
* DateTimePickDialogUtil(SinvestigateActivity.this,initDateTime);
* dateTimePicKDialog.dateTimePicKDialog(inputDate);
*
* } });
*
* @author
*/
public class DateTimePickDialogUtil implements OnDateChangedListener,
OnTimeChangedListener {
private DatePicker datePicker;
private TimePicker timePicker;
private AlertDialog ad;
private String dateTime;
private String initDateTime;
private Activity activity;
/**
* 日期時間彈出選擇框構造函數
*
* @param activity
* :調用的父activity
* @param initDateTime
* 初始日期時間值,作為彈出窗口的標題和日期時間初始值
*/
public DateTimePickDialogUtil(Activity activity, String initDateTime) {
this.activity = activity;
this.initDateTime = initDateTime;
}
public void init(DatePicker datePicker, TimePicker timePicker) {
Calendar calendar = Calendar.getInstance();
if (!(null == initDateTime || "".equals(initDateTime))) {
calendar = this.getCalendarByInintData(initDateTime);
} else {
initDateTime = calendar.get(Calendar.YEAR) + "年"
+ calendar.get(Calendar.MONTH) + "月"
+ calendar.get(Calendar.DAY_OF_MONTH) + "日 "
+ calendar.get(Calendar.HOUR_OF_DAY) + ":"
+ calendar.get(Calendar.MINUTE);
}
datePicker.init(calendar.get(Calendar.YEAR),
calendar.get(Calendar.MONTH),
calendar.get(Calendar.DAY_OF_MONTH), this);
timePicker.setCurrentHour(calendar.get(Calendar.HOUR_OF_DAY));
timePicker.setCurrentMinute(calendar.get(Calendar.MINUTE));
}
/**
* 彈出日期時間選擇框方法
*
* @param inputDate
* :為需要設置的日期時間文本編輯框
* @return
*/
public AlertDialog dateTimePicKDialog(final EditText inputDate) {
LinearLayout dateTimeLayout = (LinearLayout) activity
.getLayoutInflater().inflate(R.layout.common_datetime, null);
datePicker = (DatePicker) dateTimeLayout.findViewById(R.id.datepicker);
timePicker = (TimePicker) dateTimeLayout.findViewById(R.id.timepicker);
init(datePicker, timePicker);
timePicker.setIs24HourView(true);
timePicker.setOnTimeChangedListener(this);
ad = new AlertDialog.Builder(activity)
.setTitle(initDateTime)
.setView(dateTimeLayout)
.setPositiveButton("設置", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
inputDate.setText(dateTime);
}
})
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
inputDate.setText("");
}
}).show();
onDateChanged(null, 0, 0, 0);
return ad;
}
public void onTimeChanged(TimePicker view, int hourOfDay, int minute) {
onDateChanged(null, 0, 0, 0);
}
public void onDateChanged(DatePicker view, int year, int monthOfYear,
int dayOfMonth) {
// 獲得日歷實例
Calendar calendar = Calendar.getInstance();
calendar.set(datePicker.getYear(), datePicker.getMonth(),
datePicker.getDayOfMonth(), timePicker.getCurrentHour(),
timePicker.getCurrentMinute());
SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm");
dateTime = sdf.format(calendar.getTime());
ad.setTitle(dateTime);
}
/**
* 實現將初始日期時間2012年07月02日 16:45 拆分成年 月 日 時 分 秒,並賦值給calendar
*
* @param initDateTime
* 初始日期時間值 字元串型
* @return Calendar
*/
private Calendar getCalendarByInintData(String initDateTime) {
Calendar calendar = Calendar.getInstance();
// 將初始日期時間2012年07月02日 16:45 拆分成年 月 日 時 分 秒
String date = spliteString(initDateTime, "日", "index", "front"); // 日期
String time = spliteString(initDateTime, "日", "index", "back"); // 時間
String yearStr = spliteString(date, "年", "index", "front"); // 年份
String monthAndDay = spliteString(date, "年", "index", "back"); // 月日
String monthStr = spliteString(monthAndDay, "月", "index", "front"); // 月
String dayStr = spliteString(monthAndDay, "月", "index", "back"); // 日
String hourStr = spliteString(time, ":", "index", "front"); // 時
String minuteStr = spliteString(time, ":", "index", "back"); // 分
int currentYear = Integer.valueOf(yearStr.trim()).intValue();
int currentMonth = Integer.valueOf(monthStr.trim()).intValue() - 1;
int currentDay = Integer.valueOf(dayStr.trim()).intValue();
int currentHour = Integer.valueOf(hourStr.trim()).intValue();
int currentMinute = Integer.valueOf(minuteStr.trim()).intValue();
calendar.set(currentYear, currentMonth, currentDay, currentHour,
currentMinute);
return calendar;
}
/**
* 截取子串
*
* @param srcStr
* 源串
* @param pattern
* 匹配模式
* @param indexOrLast
* @param frontOrBack
* @return
*/
public static String spliteString(String srcStr, String pattern,
String indexOrLast, String frontOrBack) {
String result = "";
int loc = -1;
if (indexOrLast.equalsIgnoreCase("index")) {
loc = srcStr.indexOf(pattern); // 取得字元串第一次出現的位置
} else {
loc = srcStr.lastIndexOf(pattern); // 最後一個匹配串的位置
}
if (frontOrBack.equalsIgnoreCase("front")) {
if (loc != -1)
result = srcStr.substring(0, loc); // 截取子串
} else {
if (loc != -1)
result = srcStr.substring(loc + 1, srcStr.length()); // 截取子串
}
return result;
}
}
/DateTimePicker/src/com/wwj/datetimepicker/DateTimePickerActivity.java
package com.wwj.datetimepicker;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.EditText;
/**
* 時間拾取器界面
*
* @author wwj_748
*
*/
public class DateTimePickerActivity extends Activity {
/** Called when the activity is first created. */
private EditText startDateTime;
private EditText endDateTime;
private String initStartDateTime = "2013年9月3日 14:44"; // 初始化開始時間
private String initEndDateTime = "2014年8月23日 17:44"; // 初始化結束時間
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// 兩個輸入框
startDateTime = (EditText) findViewById(R.id.inputDate);
endDateTime = (EditText) findViewById(R.id.inputDate2);
startDateTime.setText(initStartDateTime);
endDateTime.setText(initEndDateTime);
startDateTime.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
DateTimePickDialogUtil dateTimePicKDialog = new DateTimePickDialogUtil(
DateTimePickerActivity.this, initEndDateTime);
dateTimePicKDialog.dateTimePicKDialog(startDateTime);
}
});
endDateTime.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
DateTimePickDialogUtil dateTimePicKDialog = new DateTimePickDialogUtil(
DateTimePickerActivity.this, initEndDateTime);
dateTimePicKDialog.dateTimePicKDialog(endDateTime);
}
});
}
}
自定義界面:
/DateTimePicker/res/layout/common_datetime.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<DatePicker
android:id="@+id/datepicker"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:calendarViewShown="false" />
<TimePicker
android:id="@+id/timepicker"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout>
② Android仿ios條件選擇器pickerview
最近怎麼老寫View,可能寫view比較方便,寫其它東西還要抽時間整理總結,寫View就直接封完寫出來就行。
准備國慶放假,無心工作,那就寫篇簡單實用一點的文章,總不能白白浪費了時間。
有時候ios端會用到條件選擇器,好像是那邊自帶的,而android這邊是沒有的,但是為了兩端統一,沒辦法,只能我們去遷就他們了(你讓一個有自帶的去寫自定義是基本不可能的事)。
最經典的是我們有選擇地址的需求,比如美團這里的:
這個android是原生是沒有的,只有能選擇日期的。那怎麼辦?自定義,好像略難,那就用三方的吧。
https://github.com/Bigkoo/Android-PickerView
我找了很多,就覺得這個庫是做得比較好,比較完整的,而且也一直有在維護,還是比較推薦,使用起來也比較方便。項目里有很清晰的文檔,建議看之前先瀏覽過文檔。
我使用的效果:
我還是順便把源碼也瀏覽了下。發現這里有3個比較重要的類,這個之後會簡單的介紹:
(1)WheelView
(2)條件選擇的WheelOptions, 我感覺這個類的封裝有點vm的意思
(3)最外層封裝的OptionsPickerView
如果只是為了選擇地址的話直接用它封裝好的就行,但是有時候可能會需要用到其它的布局或需求,那我們就要在它原有的功能上進行擴展,比如說我寫的這個時間段的現在,直接用是沒有的,需要自己擴展。
而要進行擴展的話,就要先瀏覽源碼看看它內部怎麼寫的。
可以從調用的地方找到OptionsPickerView類
然後看看OptionsPickerView類內部,你會發現很多方法,但是基本都是builder方法個getset方法,我們可以找到重要的幾個方法。
這里做的是為view設置屬性。重要的是這里
這里的意思就是把這個View給WheelOptions這個對象,讓它來做處理。然後可以看
看布局。
可以看出它裡面是寫死固定就是3列。其實我不太贊成這樣的做法,對於這樣的多情況view的封裝,我個人還是比較喜歡做動態的。由於這里固定是3列,所以我上圖中4列的情況直接使用是實現不了的,所以需要擴展。這里的WheelView就是單列
它這裡布局寫死了固定3列,那我肯定是沒法復用它的這個布局了,所以就只能重寫布局。
我只寫了LinearLayout,就是要動態去添加WheelView。
原本的OptionsPickerView中
在builder構造時就固定了布局,所以我這不好擴展,不如重寫一個OptionsPickerView,當然重寫Builder也行,但是我覺得重寫OptionsPickerView比較好。而且他原本只有兩個類
所以我們需要繼承BasePickerView重寫一個PickerView,他原本內部的邏輯沒問題,我就抄過來用好了。
修改了
(1)修改布局變成我的布局
(2)然後把創建WheelView給加擴展createWheel(optionsPicker, context, total);因為我不想每次都都寫Builder這么多參數,我把這個pickerview當成中間成來弄,讓子類繼承它來做簡單的擴展
我們重寫個WheelView,因為原本的WheelView是做固定3列的處理,我們需要做成個動態的。
(1)我多添加了個參數total表示要展示多少列
(2)用List<WheelView> wvList數組來動態創建添加WheelView
(3)用List<List<T>> items 來裝每一列的數據(我這個Wheel只做了不關聯情況下的多列,關聯情況下我沒弄)
(4)showWheelView();
這個方法做展示的規則,默認是平均展示total列,而如果需要做特殊的展示情況,像我上邊一樣的,就寫個類繼承這個類重新這個方法重新展示的規則就行,比如我的時間期間選擇器。
重寫這個方法就能展示出自己需要展示的效果
調用時也很方便。
我講這篇的目的是為了第一介紹一下這個三方庫,還是比較實用的。第二,說下擴展的重要性。第三,放假了實在工作效率低。
③ android 做一個微信那樣的地區選擇器求思路,求方法
不知微信上哪個模塊有這個地址選擇,我暫時沒發現,就說下我用的那個原理吧,界面是這樣的:
首先數據源是通過解析全國地址信息的xml文件(國家測繪局下載的)得到的,分別封裝獲取了省市縣三級集合信息,每項點擊打開是一個listView,特別注意當用戶點擊了某一具體的省時,要同步跟新界面中其市(任選一個)和該市的縣的集合即可。
④ Android DatePicker使用遇到了一些問題
讓它失去焦點就可以了:
不直接用DatePickerDialog,用AlertDialog setView的方式,
view 裡面放一個DatePicker、TimePicker和EditText(EditText的長寬為0dp),
點擊確定的時候先editText.requestFocus();再用datePicker.getYear()、 datePicker.getMonth()等就可以了