Android仿貼吧內(nèi)容下的簡單ListView嵌套GridView
ListView嵌套GridView的簡單實(shí)例
我的項(xiàng)目想實(shí)現(xiàn)一個(gè)listview里面的每個(gè)item都嵌套一個(gè)GridView,頂部還有主題等內(nèi)容,如

總所周知,關(guān)于ListView嵌套GridView,最主要問題莫過于嵌套狀態(tài)下滑動(dòng)沖突問題,具體怎么解決,喜歡冗長無注釋的代碼的,請點(diǎn)這篇文章跟其他的都大同小異了,不過在缺少注釋的情況下,我發(fā)現(xiàn)了一點(diǎn)點(diǎn)小問題:
/**
* 創(chuàng)建日期:2017/3/21.
* 說明:構(gòu)造方法根據(jù)你的SDK最低版本不同而要求不同,如18的至少必須重寫前
* 三個(gè),第四個(gè)SDK要求最低21,可以不重寫,但前三個(gè)必須寫,否則這個(gè)自定義的
* MyGridView 在運(yùn)用時(shí)會(huì)報(bào)錯(cuò);
* onMeasure:自定義GridView 控件,實(shí)現(xiàn)無法滾動(dòng)(拖動(dòng))的方法
*/
public class MyGridView extends GridView {
public MyGridView(Context context) {
super(context);
}
public MyGridView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MyGridView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
//public MyGridView(Context context, AttributeSet attrs, int //defStyleAttr, int defStyleRes) {
// super(context, attrs, defStyleAttr, defStyleRes);
// }
/**
* 重寫測量GridView的內(nèi)容空間(有多少數(shù)據(jù)內(nèi)容)
* @param widthMeasureSpec 占用寬度
* @param heightMeasureSpec 占用高度
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// expandSpec:拓展空間,其中MeasureSpec.AT_MOST為“最大模式”
// AT_MOST:最大模式,比喻為布局里的match_parent
// EXACTLY:精確模式,比喻為布局里的"50dp"
// UNSPECIFIED:未指定模式,比喻為布局里的wrap_content
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);
}
/**
* 重寫事件分發(fā):因?yàn)閮蓚€(gè)都是ViewGroup,這個(gè)方法不知道是否可行,
* 有興趣的朋友自行腦補(bǔ)
*/
// @Override
// public boolean dispatchTouchEvent(MotionEvent ev) {
// if (ev.getAction() == MotionEvent.ACTION_MOVE){
// //返回true直接結(jié)束當(dāng)前事件消費(fèi)
// return true;
// }
// return super.dispatchTouchEvent(ev);
// }
/**
* 如果是嵌套在ScollView中的,則這樣寫
* 設(shè)置是否有ScrollBar,當(dāng)要在ScollView中顯示時(shí),應(yīng)當(dāng)設(shè)置為
* false。 默認(rèn)為 true
*/
// boolean haveScrollbars = false;
// public void setHaveScrollbar(boolean haveScrollbar) {
// this.haveScrollbar = haveScrollbar;
// }
// @Override
// protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// if (haveScrollbars == false) {
// int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
// super.onMeasure(widthMeasureSpec, expandSpec);
// } else {
// super.onMeasure(widthMeasureSpec, heightMeasureSpec);
// }
// }
}
上面這個(gè)GridView就算自定義好了,接下來我們簡單貼上listview的適配器主要方法getview(…)
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
holder = new ViewHolder();
convertView = mInflater.inflate(R.layout.item_lv_gridview, null);
holder.mMyGridView = (MyGridView) convertView.findViewById(R.id.gridView_show_controller);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
//注意:這是重要的地方
//鑒于我們想讓每個(gè)item下的GridView都能,因此只能通過new 出適配器來
//單獨(dú)定義每個(gè)item,這樣才能讓每個(gè)listview的item內(nèi)容都有不同的GridView
DevicesAdapter devicesAdapter = new DevicesAdapter(mContext);
devicesAdapter.setDevicesList(deviceList);
holder.mMyGridView.setAdapter(devicesAdapter);
holder.mMyGridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
//最后一個(gè)item(position==list.size()-1),
//動(dòng)態(tài)更改其作用功能(比如增加一條數(shù)據(jù),或者blablabla...)
if (position == parent.getCount() - 1) {
mItemListener.onAddClick();
} else {
mItemListener.onDeviceItemClick(deviceList.get(position));
}
}
});
return convertView;
}
static class ViewHolder {
MyGridView mMyGridView;
}
只要再定義GridView的適配器就大功告成了(適配器相信到這時(shí)候大家應(yīng)該都很熟了,我就不注釋了哈,請?jiān)徫冶容^懶)
public class DevicesAdapter extends BaseAdapter {
private final LayoutInflater mInflater;
private ArrayList<DeviceInfos> devicesList;
public DevicesAdapter(Context context) {
mInflater = LayoutInflater.from(context);
}
public void setDevicesList(ArrayList<DeviceInfos> devicesList) {
this.devicesList = devicesList;
}
@Override
public int getCount() {
if (devicesList == null) {
return 1;
}
return devicesList.size() + 1;
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
convertView = mInflater.inflate(R.layout.item_grid_view, null);
holder = new ViewHolder();
holder.mImageView = (ImageView) convertView.findViewById(R.id.img_controller);
holder.mTextView = (TextView) convertView.findViewById(R.id.tv_controller_name);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
if (devicesList!=null && position < devicesList.size()) {
String childDeviceName = devicesList.get(position).getChildDeviceName();
holder.mImageView.setImageResource(R.drawable.huajidadi);
holder.mTextView.setText(childDeviceName);
} else {
holder.mImageView.setImageResource(R.drawable.add);
holder.mTextView.setVisibility(View.GONE);
}
return convertView;
}
static class ViewHolder {
ImageView mImageView;
TextView mTextView;
}
}
布局我就不給了,就簡單的幾個(gè)item的布局,請同學(xué)們自己定義吧…
總結(jié):解決滑動(dòng)沖突的方法或者涉及點(diǎn)擊、移動(dòng)等屬性的,用自定義控件再重寫事件分發(fā),可以很好的解決。但是對于同樣是ViewGroup就沒辦法像View的組件那樣,通過調(diào)用onInterceptTouchEvent(MotionEvent ev)進(jìn)行攔截了,因此只能另尋他法,而這關(guān)鍵就是重寫onMeasure(int widthMeasureSpec, int heightMeasureSpec)這個(gè)方法,使得GridView的“控件大小”被固定,這樣就不會(huì)與另一個(gè)滑動(dòng)事件沖突了。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android編程實(shí)現(xiàn)攔截短信并屏蔽系統(tǒng)Notification的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)攔截短信并屏蔽系統(tǒng)Notification的方法,較為詳細(xì)的分析了Android短信與Notification的原理及對應(yīng)的設(shè)置取消技巧,需要的朋友可以參考下2015-12-12
Jetpack Compose實(shí)現(xiàn)對話框和進(jìn)度條實(shí)例解析
對話框和進(jìn)度條其實(shí)并無多大聯(lián)系,放在一起寫是因?yàn)閮烧叩膬?nèi)容都不多,所以湊到一起,對話框是我們平時(shí)開發(fā)使用得比較多的組件,進(jìn)度條的使用頻率也很高,比如下載文件,上傳文件,處理任務(wù)時(shí)都可以使用進(jìn)度條2023-04-04
flutter InheritedWidget使用方法總結(jié)
這篇文章主要為大家介紹了flutter InheritedWidget使用方法總結(jié)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
Android開發(fā)中ListView自定義adapter的封裝
這篇文章主要為大家詳細(xì)介紹了android開發(fā)中ListView自定義adapter的封裝,ListView的模板寫法,感興趣的小伙伴們可以參考一下2016-09-09
Android 異步任務(wù)和消息機(jī)制面試題分析
這篇文章主要為大家介紹了Android 異步任務(wù)和消息機(jī)制面試題分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07
Android Studio多工程引用同一個(gè)library項(xiàng)目配置的解決方法
大家在使用android studio的時(shí)候,會(huì)遇到多個(gè)項(xiàng)目引用相同的library這篇文章主要介紹了Android Studio多工程引用同一個(gè)library項(xiàng)目配置方法,需要的朋友可以參考下2018-03-03
Android中ViewPager帶來的滑動(dòng)卡頓問題解決要點(diǎn)解析
這里我們主要針對ViewGroup的SwipeRefreshLayout中引入ViewPager所引起的滑動(dòng)沖突問題進(jìn)行討論,一起來看一下Android中ViewPager帶來的滑動(dòng)卡頓問題解決要點(diǎn)解析:2016-06-06
Android App中使用SurfaceView制作多線程動(dòng)畫的實(shí)例講解
這篇文章主要介紹了Android App中使用SurfaceView制作多線程動(dòng)畫的實(shí)例講解,SurfaceView經(jīng)常被用來制作游戲中的動(dòng)畫,不過同時(shí)要注意畫面閃爍的問題,需要的朋友可以參考下2016-04-04

