2 列の ListView を含むアプリを作成しています。最初の列にはカウントダウンが表示され、2 番目の列にはカウントダウンの内容を説明する追加のテキストが表示されます。以下に、動作する私のコードを示します...多かれ少なかれ。タイマーが刻々と過ぎている複数の行を持つリストビューがあります。問題の 1 つは、ランナブルの set.Text() がすべての行をオーバーライドしているように見えることです。たとえば、行 1 の runnable は行 2 と 3 にもテキストを設定し、行 2 の runnable は行 1 と 3 にもテキストを設定します。これには、最初の列が点滅するという効果があります (正しい値と他の行の値で)。リストビューの特定の行にテキストを設定するにはどうすればよいですか?
次の問題: ハンドラーからコールバックを削除しても、ランナブルが継続的に実行されます。しかし、アクティビティがバックグラウンドまたはクローズされている場合、タイマーの刻みは必要なく、システム リソースを無駄にしたくありません。
マイ アクティビティ:
public class TimerActivity extends ListActivity {
MyTimerAdapter myTimerAdapter = null;
ArrayList<Long> timerList = new ArrayList<Long>();
ArrayList<String> textList = new ArrayList<String>();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.listactivity);
myTimerAdapter = new MyTimerAdapter(this, R.layout.row, R.id.tv_timer, R.id.tv_text);
setListAdapter(myTimerAdapter);
}
@Override
protected void onResume() {
super.onResume();
refreshView();
}
@Override
protected void onPause() {
myTimerAdapter.clear();
myTimerAdapter.notifyDataSetChanged();
super.onPause();
}
private void refreshView() {
myTimerAdapter.clear();
timerList.clear();
textList.clear();
// some code to read database and fill
// array timerList with a long value (used for displaying a countdown)
// and textList with some additional text
myTimerAdapter.add(timerList, textList);
myTimerAdapter.notifyDataSetChanged();
}
}
私のアダプター:
public class MyTimerAdapter extends ArrayAdapter<String> {
private Activity mContext;
private ArrayList<Long> mTimer;
private ArrayList<String> mText;
private int mViewId;
private int mViewIdFieldTimer;
private int mViewIdFieldText;
private int listSize;
private ArrayList<Handler> handlerList = new ArrayList<Handler>();
private ArrayList<TimerRunnable> runList = new ArrayList<TimerRunnable>();
public MyTimerAdapter(Activity context, int textViewResId, int tv1, int tv2) {
super(context, textViewResId);
mContext = context;
mViewId = textViewResId;
mViewIdFieldTimer = tv1;
mViewIdFieldText = tv2;
listSize = 0;
}
public void add(ArrayList<Long> timer, ArrayList<String> text) {
mTimer = timer;
mText = text;
listSize = mText.size();
handlerList.clear();
runList.clear();
}
@Override
public void clear() {
super.clear();
int i;
for (i=0; i<listSize; i++) {
handlerList.get(i).removeCallbacksAndMessages(runList.get(i));
runList.get(i).stopHandler();
}
}
@Override
public int getCount() {
return listSize;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View v = convertView;
if (v == null) {
LayoutInflater vi = mContext.getLayoutInflater();
v = vi.inflate(mViewId, null);
}
long timerLine = mTimer.get(position);
if (timerLine != 0) {
TextView tvTimer = (TextView) v.findViewById(mViewIdFieldTimer);
if (tvTimer != null) {
tvTimer.setTag(position);
final Handler mTimerHandler = new Handler();
TimerRunnable timerTask = new TimerRunnable(tvTimer, tvTimer.getTag().toString(), timerLine);
mTimerHandler.post(timerTask);
// save in array to stop later
handlerList.add(mTimerHandler);
runList.add(timerTask);
}
}
String textLine = mText.get(position);
if (textLine != null) {
TextView tvText = (TextView) v.findViewById(mViewIdFieldText);
if (tvText != null) {
tvText.setText(textLine);
}
}
return v;
}
}
私のランナブル:
public class TimerRunnable implements Runnable {
private TextView tv;
final Handler mTimerHandler = new Handler();
String tag;
long endtime;
long sec;
public TimerRunnable (TextView tv, String tag, long endtime) {
this.tv = tv;
this.tag = tag;
this.endtime = endtime;
}
public void run() {
if (tv.getTag().toString().equals(tag)) {
Calendar cal = Calendar.getInstance();
sec = endtime - (cal.getTimeInMillis() / 1000); //endtime - aktuelle Zeit
if (sec >= 0) {
// some code formatting the time in seconds to something like hh:mm:ss (var String txt)
tv.setText(txt);
System.out.println(txt); // only for tests; so I could see that runnable is still running
mTimerHandler.postDelayed(this, 1000);
}
}
}
public void stopHandler() {
mTimerHandler.removeCallbacksAndMessages(null);
}
}