我將首先說這是一個由兩部分組成的問題。我有一個3按鈕和一個列表視圖的活動。一個按鈕用日期選擇器打開一個對話框片段,按鈕文本顯示所選日期。其他2個按鈕可讓您分別前往/下一天。 ListView應顯示對應於所選日期的記錄。根據日期選擇器更改從數據庫重新加載ListView內容
當前,加載活動時,按鈕顯示今天的日期,列表視圖顯示數據庫中今天的記錄。我已經設法創建一個方法,重新啓動加載程序並在上一個/下一個按鈕中調用它,所以當每次瀏覽一天時,listview會更新以顯示正確的記錄,但是當我通過調用datepicker來更改日期時對話框中,我似乎無法找到正確更新的方法 - 這是問題的第一部分。
第二部分有點難解釋。當我選擇4條記錄(列表視圖中有4條記錄)的日期時,所有4條記錄都會顯示,但是例如,如果我轉到只有2條記錄(列表視圖中有2條記錄)的第二天,它會正確顯示第一條記錄2條記錄,但接下來的2行仍顯示上一個日期的數據。爲了更好地解釋這一點,讓我們假設listview中的每一行都顯示DB記錄_id數字。在日期1(其中有4條記錄)我有ID 1,2,3,4顯示,但當更改爲日期2(其中有2條記錄)時,顯示ID 5,6,3,4。
我假設通過合併一個完全清除listview內容然後用新日期重新填充的方法,我可以同時解決這兩個問題。我只是不知道如何做到這一點。我的代碼如下:
public class FuncLogbook extends ListActivity implements
LoaderManager.LoaderCallbacks<Cursor>, OnItemClickListener {
Button btn_logbook_date;
LogDateDialogFragment fragDate;
Calendar now;
Date nowD, newD, PrevDate, NextDate;
int lMonth;
String strDate;
private SimpleCursorAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView (R.layout.dash_logbook);
this.getListView().setDividerHeight(2);
ListView list = getListView();
list.setOnItemClickListener(this);
btn_logbook_date = (Button) findViewById(R.id.btn_logbook_date);
now = Calendar.getInstance();
nowD = now.getTime();
SimpleDateFormat dFormatter = new SimpleDateFormat("dd MMM yyyy");
strDate = dFormatter.format(nowD);
btn_logbook_date.setText(strDate);
fillData();
}
public void modifyDate(View v) {
int id = v.getId();
switch (id) {
case R.id.btn_logbook_date :
showDialogDate();
break;
case R.id.btn_logbook_prev :
updateDatePrev();
break;
case R.id.btn_logbook_next :
updateDateNext();
break;
}
}
public interface LogDateDialogFragmentListener {
public void logbookChangedDate(int year, int month, int day);
}
public void showDialogDate() {
FragmentTransaction ftLogDate = getFragmentManager().beginTransaction();
fragDate = LogDateDialogFragment.newInstance(this, new LogDateDialogFragmentListener() {
public void logbookChangedDate(int year, int month, int day) {
now.set(Calendar.YEAR, year);
now.set(Calendar.MONTH, month);
now.set(Calendar.DAY_OF_MONTH, day);
newD = now.getTime();
SimpleDateFormat newDFormatter = new SimpleDateFormat("dd MMM yyyy");
strDate = newDFormatter.format(newD);
btn_logbook_date.setText(strDate);
}}, now);
fragDate.show(ftLogDate, "DateDialogFragment");
}
public void updateDatePrev() {
String prevVar[] = btn_logbook_date.getText().toString().split(" ");
if (prevVar[1].equalsIgnoreCase("Jan")) {
lMonth = 0;
} else if (prevVar[1].equalsIgnoreCase("Feb")) {
lMonth = 1;
} else if (prevVar[1].equalsIgnoreCase("Mar")) {
lMonth = 2;
} else if (prevVar[1].equalsIgnoreCase("Apr")) {
lMonth = 3;
} else if (prevVar[1].equalsIgnoreCase("May")) {
lMonth = 4;
} else if (prevVar[1].equalsIgnoreCase("Jun")) {
lMonth = 5;
} else if (prevVar[1].equalsIgnoreCase("Jul")) {
lMonth = 6;
} else if (prevVar[1].equalsIgnoreCase("Aug")) {
lMonth = 7;
} else if (prevVar[1].equalsIgnoreCase("Sep")) {
lMonth = 8;
} else if (prevVar[1].equalsIgnoreCase("Oct")) {
lMonth = 9;
} else if (prevVar[1].equalsIgnoreCase("Nov")) {
lMonth = 10;
} else if (prevVar[1].equalsIgnoreCase("Dec")) {
lMonth = 11;
}
int lYear = Integer.parseInt(prevVar[2]);
int lDay = Integer.parseInt(prevVar[0]);
now = Calendar.getInstance();
now.set(Calendar.YEAR, lYear);
now.set(Calendar.MONTH, lMonth);
now.set(Calendar.DAY_OF_MONTH, lDay - 1);
PrevDate = now.getTime();
SimpleDateFormat prevDFormatter = new SimpleDateFormat("dd MMM yyyy");
strDate = prevDFormatter.format(PrevDate);
btn_logbook_date.setText(strDate);
refillData();
}
public void updateDateNext() {
// Same as updateDatePrev() method, but moves forward 1 day, instead of backward
}
protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
super.onActivityResult(requestCode, resultCode, intent);
}
private void fillData() {
String[] from = new String[] { AddDBHelper.KEY_BGL, AddDBHelper.KEY_ROWID,
AddDBHelper.KEY_CATEG, AddDBHelper.KEY_TIME };
int[] to = new int[] {R.id.logBGL, R.id.logRowID, R.id.logCateg, R.id.logTime };
getLoaderManager().initLoader(0, null, this);
adapter = new SimpleCursorAdapter(this, R.layout.logbook_item, null, from, to, 0);
setListAdapter(adapter);
}
private void refillData() {
getLoaderManager().restartLoader(0, null, this);
}
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
String selection = AddDBHelper.KEY_DATE + "=?";
String[] selectionArgs = { String.valueOf(btn_logbook_date.getText().toString()) };
CursorLoader cl = new CursorLoader(this, DBProvider.CONTENT_URI,
null, selection, selectionArgs, null);
return cl;
}
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
adapter.swapCursor(data);
}
public void onLoaderReset(Loader<Cursor> loader) {
// data is not available anymore; delete reference
adapter.swapCursor(null);
}
}
UPDATE
從ContentProvider的顯示查詢的信息:
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
String sortOrder) {
// Using SQLiteQueryBuilder instead of query() method
SQLiteQueryBuilder queryBuilder = new SQLiteQueryBuilder();
// Set the DB table to query
queryBuilder.setTables(AddDBHelper.DB_TABLE);
int uriType = sURIMatcher.match(uri);
switch (uriType) {
case DBAllEntries:
//no filter or WHERE clause
break;
case DBSingleEntry:
//add ROWID filter when selecting individual item
queryBuilder.appendWhere(AddDBHelper.KEY_ROWID + "="
+ uri.getLastPathSegment());
break;
default:
throw new IllegalArgumentException("Unknown URI: " + uri);
}
SQLiteDatabase sqlDB = mDBP.getWritableDatabase();
Cursor cursor = queryBuilder.query(sqlDB, projection, selection,
selectionArgs, null, null, sortOrder);
// Ensure potential listeners are getting notified
cursor.setNotificationUri(getContext().getContentResolver(), uri);
return cursor;
}
UPDATE
以下調用refillData()
與第一日期變化工作,但沒有再次工作。重新啓動模擬器並再次嘗試後,我能夠看到2個單獨的日期更改記錄,但不會再更改數據。
public void showDialogDate() {
FragmentTransaction ftLogDate = getFragmentManager().beginTransaction();
fragDate = LogDateDialogFragment.newInstance(this, new LogDateDialogFragmentListener() {
public void logbookChangedDate(int year, int month, int day) {
now.set(Calendar.YEAR, year);
now.set(Calendar.MONTH, month);
now.set(Calendar.DAY_OF_MONTH, day);
newD = now.getTime();
SimpleDateFormat newDFormatter = new SimpleDateFormat("dd MMM yyyy");
strDate = newDFormatter.format(newD);
btn_logbook_date.setText(strDate);
refillData();
}}, now);
fragDate.show(ftLogDate, "DateDialogFragment");
}
通過下面的呼叫,數據不會更改日期更改之後,但是當我按下按鈕再次打開日期選擇器,我可以看到在後臺列表視圖更新(對話片段後面)。例如,我選擇日期2,數據不會更改。我在對話框後面打開日期選擇器和日期2數據顯示。我將日期選擇器設置爲日期3,但列表視圖仍然顯示日期2的數據。如果再次打開日期選擇器,則將日期3的數據加載到對話框片段後面的背景中。
如果我在FragDate.show(...)
行之後或在FragmentTransaction ftLogDate...
行之後撥打refillData()
行,我會得到相同的結果。
public void showDialogDate() {
FragmentTransaction ftLogDate = getFragmentManager().beginTransaction();
fragDate = LogDateDialogFragment.newInstance(this, new LogDateDialogFragmentListener() {
public void logbookChangedDate(int year, int month, int day) {
now.set(Calendar.YEAR, year);
now.set(Calendar.MONTH, month);
now.set(Calendar.DAY_OF_MONTH, day);
newD = now.getTime();
SimpleDateFormat newDFormatter = new SimpleDateFormat("dd MMM yyyy");
strDate = newDFormatter.format(newD);
btn_logbook_date.setText(strDate);
}}, now);
refillData();
fragDate.show(ftLogDate, "DateDialogFragment");
}
*我似乎無法找到一種方法讓它正確更新*以及究竟發生了什麼?你不應該從你傳遞給你的'DialogFragment'的偵聽器調用'refillData'方法嗎?關於第二個問題,除非您更改數據庫中的數據,或者您在ContentProvider的查詢方法中做了非常錯誤的操作(假設您檢查返回的「Cursor」並且它確實如此)有正確的行,沒有別的)。 – Luksprog
我必須離開才能工作,但會稍後用評論更新上面的代碼,以演示我得到的不同結果。 至於從我的聽衆調用它,我會在哪裏將它放在上面的代碼? 至於第二個問題,請參閱上文。我已經從ContentProvider添加了我的查詢。 – Ronnie
內容提供商對我而言仍然很陌生。我已經採用了通用示例並對其進行了修改以供我使用。我相信我明白它是如何工作的,但我可能是錯的。 – Ronnie