在項目中實現了ListView置頂的功能,所以自己打算寫一下博客記錄下來。
其實實現起來還是挺簡單的,核心思想是改變其adapter里的數據排序。
效果如圖哈
那么開始吧
1 首先你要實現ListView吧,這個很簡單,就不多說了,其次是要自己繼承 Arrayadapter<T>,因為要用到數據的排序,所以使用Arrayadapter<T>綁定你的數據。
public SessionItemAdapter extends ArrayAdapter<Session> {
Context mContext;
/**
* 不建議使用這種方式 將數據與adapter進行綁定,如果要進行數據更新等操作
* 因為數據引用是相同的情況,會同步影響數據的變更。例如使用clean()方法消除數據
* 不僅僅消除了adapter里面的數據,還會清除了相同內存地址的數據源
*/
public SessionItemAdapter(Context context, List<Session> sessions) {
super(context, 0, sessions);
mContext = context;
}
public SessionItemAdapter(Context context) {
super(context, 0, new ArrayList<Session>());
mContext = context;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = new ItemView(mContext);
}
ItemView itemView = (ItemView) convertView;
itemView.setText(String.valueOf(getItem(position).getTop()));
itemView.setAvatar(getItem(position).getAvatar());
if (getItem(position).getTop() == 1) {
itemView.setBackgroundResource(R.drawable.bg_top_item_selector);
} else {
itemView.setBackgroundResource(R.drawable.bg_item_selector);
}
return convertView;
}
public void updateData(List<Session> sessionList) {
clear();
addAll(sessionList);
}
}
這是普通的listview的adapter,只是要注意的是構造函數那里的注釋,因為如果一開始將數據源綁定在ArrayAdapter,將會影響數據源的更新,所以使用第二種構造方法先傳入空的list,在updateData()方法中實時更新數據,才能呈現出置頂效果。
2 我使用了Dialogfragment作為切換置頂的一種形式,里面使用了回調接口,我們在回調接口里改變數據,實現置頂。
public class PopupDialogFragment extends DialogFragment {
private DialogItemOnClickListener itemOnClickListener;
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.popview, null);
TextView onTopTv = (TextView) view.findViewById(R.id.on_top_tv);
TextView cancelTv = (TextView) view.findViewById(R.id.cancel_top_tv);
Bundle bundle = getArguments();
int isTop = bundle.getInt(MainActivity.TOP_STATES);
if (isTop == 1) {
onTopTv.setVisibility(View.GONE);
cancelTv.setVisibility(View.VISIBLE);
} else if (isTop == 0) {
onTopTv.setVisibility(View.VISIBLE);
cancelTv.setVisibility(View.GONE);
}
onTopTv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
getDialog().dismiss();
itemOnClickListener.onTop();
}
});
cancelTv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
getDialog().dismiss();
itemOnClickListener.onCancel();
}
});
getDialog().getWindow().requestFeature(STYLE_NO_TITLE);
setStyle(STYLE_NO_FRAME, android.R.style.Theme_Light);
setCancelable(true);
getDialog().getWindow().setBackgroundDrawableResource(R.color.write_bg);
return view;
}
public void setItemOnClickListener(DialogItemOnClickListener itemOnClickListener) {
this.itemOnClickListener = itemOnClickListener;
}
public interface DialogItemOnClickListener {
void onTop();
void onCancel();
}
}
可以從代碼中看見數據里的top字段是實現置頂的判斷,分別是1為置頂,0為不置頂。然后在下面代碼里實現回調,設置數據改變。
mListView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
@Override
public boolean onItemLongClick(AdapterView<?> parent, View view, final int position, long id) {
final Session session = (Session) parent.getItemAtPosition(position);
Bundle bundle = new Bundle();
bundle.putInt(TOP_STATES, session.getTop());
PopupDialogFragment popupDialog = new PopupDialogFragment();
popupDialog.setArguments(bundle);
popupDialog.setItemOnClickListener(new PopupDialogFragment.DialogItemOnClickListener() {
@Override
public void onTop() {
session.setTop(ON_TOP);
//設置置頂時間
session.setTime(System.currentTimeMillis());
refreshView();
}
@Override
public void onCancel() {
session.setTop(CANCEL_TOP);
refreshView();
}
});
;
popupDialog.show(getFragmentManager(), "POPUP");
return true;
}
});
實時更新數據
private void refreshView() {
//如果不調用sort方法,是不會進行排序的,也就不會調用compareTo
Collections.sort(sessionList);
itemAdapter.updateData(sessionList);
}
3 重點應該是傳入ArrayAdapter的數據Session,它跟普通的數據類型不會有太大的區別,只是讓它實現了Comparable接口,重寫了compareTo()方法,從而實現了置頂功能。
public class Session implements Serializable, Comparable {
/**
* 是否置頂
*/
public int top;
/**
* 置頂時間
**/
public long time;
/**
* 頭像
*/
public int avatar;
public long getTime() {
return time;
}
public void setTime(long time) {
this.time = time;
}
public int getTop() {
return top;
}
public void setTop(int top) {
this.top = top;
}
public int getAvatar() {
return avatar;
}
public void setAvatar(int avatar) {
this.avatar = avatar;
}
@Override
public int compareTo(Object another) {
if (another == null || !(another instanceof Session)) {
return -1;
}
Session otherSession = (Session) another;
/**置頂判斷 ArrayAdapter是按照升序從上到下排序的,就是默認的自然排序
* 如果是相等的情況下返回0,包括都置頂或者都不置頂,返回0的情況下要
* 再做判斷,拿它們置頂時間進行判斷
* 如果是不相等的情況下,otherSession是置頂的,則當前session是非置頂的,
* 應該在otherSession下面,所以返回1
* 同樣,session是置頂的,則當前otherSession是非置頂的,
* 應該在otherSession上面,所以返回-1
* */
int result = 0 - (top - otherSession.getTop());
if (result == 0) {
result = 0 - compareToTime(time, otherSession.getTime());
}
return result;
}
/**
* 根據時間對比
* */
public static int compareToTime(long lhs, long rhs) {
Calendar cLhs = Calendar.getInstance();
Calendar cRhs = Calendar.getInstance();
cLhs.setTimeInMillis(lhs);
cRhs.setTimeInMillis(rhs);
return cLhs.compareTo(cRhs);
}
}
相信注釋已經寫的很清楚了,通過1,0,-1的判斷實現置頂。
這里是demo源碼 GitHub地址