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

總所周知,關(guān)于ListView嵌套GridView,最主要問(wèn)題莫過(guò)于嵌套狀態(tài)下滑動(dòng)沖突問(wèn)題,具體怎么解決,喜歡冗長(zhǎng)無(wú)注釋的代碼的,請(qǐng)點(diǎn)擊這里這篇文章跟其他的都大同小異了,不過(guò)在缺少注釋的情況下,我發(fā)現(xiàn)了一點(diǎn)點(diǎn)小問(wèn)題:
/** * 創(chuàng)建日期:2017/3/21. * 說(shuō)明:構(gòu)造方法會(huì)根據(jù)你的SDK最低版本不同而要求不同,如18的至少必須重寫(xiě)前 * 三個(gè),第四個(gè)SDK要求最低21,可以不重寫(xiě),但前三個(gè)必須寫(xiě),否則這個(gè)自定義的 * MyGridView 在運(yùn)用時(shí)會(huì)報(bào)錯(cuò); * onMeasure:自定義GridView 控件,實(shí)現(xiàn)無(wú)法滾動(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);// } /** * 重寫(xiě)測(cè)量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); } /** * 重寫(xiě)事件分發(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中的,則這樣寫(xiě) * 設(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就算自定義好了,接下來(lái)我們簡(jiǎn)單貼上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都能獨(dú)立,因此只能通過(guò)new 出適配器來(lái)//單獨(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) {//監(jiān)聽(tīng)最后一個(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)該都很熟了,我就不注釋了哈,請(qǐ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; }}布局我就不給了,就簡(jiǎn)單的幾個(gè)item的布局,請(qǐng)同學(xué)們自己定義吧…
總結(jié):解決滑動(dòng)沖突的方法或者涉及點(diǎn)擊、移動(dòng)等屬性的,用自定義控件再重寫(xiě)事件分發(fā),可以很好的解決。但是對(duì)于同樣是ViewGroup就沒(méi)辦法像View的組件那樣,通過(guò)調(diào)用onInterceptTouchEvent(MotionEvent ev)進(jìn)行攔截了,因此只能另尋他法,而這關(guān)鍵就是重寫(xiě)onMeasure(int widthMeasureSpec, int heightMeasureSpec)這個(gè)方法,使得GridView的“控件大小”被固定,這樣就不會(huì)與另一個(gè)滑動(dòng)事件沖突了。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持VEVB武林網(wǎng)。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注