ユーザーがDatePickerから日付を選択し、リストがいくつかの値で更新されるアプリを作成しようとしています。
私のGUIは次のようになります。
<LinearLayout
Android:id="@+id/linearLayout1"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content"
Android:layout_gravity="center" >
<DatePicker
Android:id="@+id/datePicker"
Android:layout_width="wrap_content"
Android:layout_height="wrap_content" />
</LinearLayout>
<LinearLayout
Android:id="@+id/linearLayout2"
Android:layout_width="match_parent"
Android:layout_height="wrap_content" >
<ListView
Android:id="@+id/list"
Android:layout_width="match_parent"
Android:layout_height="wrap_content"
Android:layout_weight="1" >
</ListView>
</LinearLayout>
DatePickerの初期化と処理は次のようになります。
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
datePicker = (DatePicker) findViewById(R.id.datePicker);
Calendar c = Calendar.getInstance();
year = c.get(Calendar.YEAR);
month = c.get(Calendar.MONTH);
day = c.get(Calendar.DAY_OF_MONTH);
datePicker.init(year, month, day, dateSetListener);
}
private DatePicker.OnDateChangedListener dateSetListener = new DatePicker.OnDateChangedListener() {
public void onDateChanged(DatePicker view, int year, int monthOfYear,
int dayOfMonth) {
Calendar c = Calendar.getInstance();
c.set(year, monthOfYear, dayOfMonth);
System.out.println ("TEST");
}
};
CatLogでは、ウィジェットの+/-ボタンで遊ぶたびに、「TEST」文字列が2回出力されることがわかります。何が問題なのでしょうか?
注: here のように、問題がListViewに関連していないことを確認するために、リスト更新コードを意図的に「無効」にしました。
アプリケーションをテストするとき、日付の選択を受け入れた後にメソッドonDateSetが2回呼び出され、キャンセルしたときに1回呼び出されました。
パラメータビューを使用してonDateSetメソッドに検証を追加しました。次のようなものです。
@Override
public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth){
if (view.isShown()) {
updateDate(year, monthOfYear, dayOfMonth);
}
}
仕えたらいいな
私はまだきちんとした修正を見つけることができませんでした。回避策を見つけたので、イベントは2回発生しません。回避策は次のとおりです。
int timesCalled = 1;
private DatePicker.OnDateChangedListener dateSetListener = new DatePicker.OnDateChangedListener() {
public void onDateChanged(DatePicker view, int year, int monthOfYear,
int dayOfMonth) {
Calendar c = Calendar.getInstance();
c.set(year, monthOfYear, dayOfMonth);
timesCalled += 1;
if ((timesCalled % 2) == 0) {
System.out.println ("TEST");
}
}};
OnDateSetコールバックの代わりに、DatePickerDialogダイアログのonClickメソッドをオーバーライドし、BUTTON_POSITIVEクリックのみを処理しました。
public static class DatePickerFragment extends DialogFragment {
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
final Calendar c = Calendar.getInstance();
int initialYear = c.get(Calendar.YEAR);
int initialMonth = c.get(Calendar.MONTH);
int initialDay = c.get(Calendar.DAY_OF_MONTH);
return new DatePickerDialog(getActivity(), null, initialYear, initialMonth, initialDay) {
@Override
public void onClick(DialogInterface dialog, int which) {
if (which == BUTTON_POSITIVE) {
int year = getDatePicker().getYear();
int month = getDatePicker().getMonth();
int day = getDatePicker().getDayOfMonth();
//TODO: Do your logic here
}
super.onClick(dialog, which);
}
};
}
}
コードは明確なままですが、onDateSetが2回呼び出される理由はまだ明確ではありません。
おそらく、インスタンス変数を追加してリスナーを変更します。そうすれば、メソッドが最後に呼び出されたときと異なるかどうかを確認できます。
final DatePickerDialog datePickerDialog = new DatePickerDialog (this, new DatePickerDialog.OnDateSetListener()
{
private int year;
private int month;
private int day;
@Override
public void onDateSet (DatePicker view, int year, int monthOfYear, int dayOfMonth)
{
if (this.year == year && this.month == monthOfYear && this.day == dayOfMonth)
return;
this.year = year;
this.month = monthOfYear;
this.day = dayOfMonth;
calendar.set (year, month, day);
int index = officeCalendar.indexOfKey (calendar.getTimeInMillis ());
if (index > -1)
viewPager.setCurrentItem (index);
}
}, calendar.get (Calendar.YEAR), calendar.get (Calendar.MONTH), calendar.get (Calendar.DATE));
private DatePicker.OnDateChangedListener dateSetListener = new DatePicker.OnDateChangedListener() {
boolean fired = false;
public void onDateChanged(DatePicker view, int year, int monthOfYear, int dayOfMonth) {
if (fired == true){
//Twince
}
else{
Calendar c = Calendar.getInstance();
c.set(year, monthOfYear, dayOfMonth);
System.out.println ("TEST");
}
}
};
ここで私の答えを確認してください。 TimePickerDialog用ですが、DatePickerDialogにも適用できます。カスタムダイアログを作成できます。これは、ほぼ同等に高速でシンプルで、機能します。
private DatePicker.OnDateChangedListener dateSetListener = new DatePicker.OnDateChangedListener() {
public void onDateChanged(DatePicker view, int year, int monthOfYear,
int dayOfMonth) {
Calendar c = Calendar.getInstance();
c.set(year, monthOfYear, dayOfMonth);
updateDisplay();
}
};
TextView datetext;
datetext = (TextView) findViewById(R.id.selected_date);
private void updateDisplay() {
this.datetext.setText(new StringBuilder()
// Month is 0 based so add 1
.append(mMonth + 1).append("-").append(mDay).append("-")
.append(mYear).append(" "));
}
Jelly Beanの下にこのコードを追加する必要はありません
@Override
public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth){
if (view.isShown()) {
updateDate(year, monthOfYear, dayOfMonth);
}
}