listView.addHeader

本文详细介绍了如何在ListView中使用addHeaderView方法添加自定义视图,包括addHeaderView与setAdapter的调用顺序、焦点处理技巧、重载方法使用及常见的异常处理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、 概述:

在代码中使用 listView.addHeaderView(…) 方法可以在ListView组件上方添加上其他组件,并且连结在一起像是一个新组件。如果多次使用 .addHeaderView(…) ,则最先添加的组件在最上方,按添加的先后顺序由上到下罗列。 
此时listView 的 position = 0 的位置对应的是view1,而不再是原来listView中的第一条了。如下图所示: 
这里写图片描述

该方法帮我们解决了布局复杂的多布局麻烦,例如美团的listView顶部的复杂多布局实现:

这里写图片描述这里写图片描述

另外,若要对做原来的ListView做不可见设置,可使用将listView的adapter中数据置空的方法,在可见时再还原数据。

二、addHeaderView和setAdapter的顺序问题:

特别注意:在调用addHeaderView和setAdapter的顺序上,有时候会报这样的一个异常:

java.lang.IllegalStateException: 
Cannot add header view to list --setAdapter has already been called.
  • 1
  • 2

产生原因:addHeaderView()调用在setAdapter()之后,并且该代码运行在Android4.3之前的系统版本。就是因为我们在addHeaderView之前调用了setAdapter。

三、注意事项:

1,焦点之争:

item内如果有button等控件时,在监听listview的onitemclick事件时,焦点会被item内的button、imagebutton等控件抢走,从而导致在listview设置了onitemclick事件后不会被触发。解决方法是在初始化item的时候屏蔽掉其内部button等控件的焦点获取,具体方法可以在自定义item的根控件中调用:

setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS);
  • 1

这样就能阻塞字控件抢夺焦点,listview的onitemclick就能被正确触发,同时对item内部的button等控件也没有影响,他们在被点击时照样可以触发自身的点击事件。

2、重载方法:

当listview需要添加headerview时,可以通过调用listview的addHeaderView(headView, null, false) 方法,该方法还有一个重载方法 addHeaderView(headView);这两个方法的区别是前一个方法可以控制header是否可以被 selected,如果不想被selected则将第三个参数设置成false;

3、顺序问题详解:

接着上面说的添加header,添加header时调用的addHeaderView方法必须放在listview.setadapter前面,意思很明 确就是如果想给listview添加头部则必须在给其绑定adapter前添加,否则会报错。原因是当我们在调用setAdapter方法时会 android会判断当前listview是否已经添加header,如果已经添加则会生成一个新的tempadapter,这个新的 tempadapter包含我们设置的adapter所有内容以及listview的header和footer。所以当我们在给listview添加了 header后在程序中调用listview.getadapter时返回的是tempadapter而不是我们通过setadapter传进去的 adapter。如果没有设置adapter则tempadapter与我们自己的adapter是一样的。listview.getadapter().getcount()方法返回值会比我们预期的要大,原因是添加了header。

4、位置问题:

接着上面的tempadapter说,我们自定义adapter里面的getitem方法里面返回的position是不包括header的,是我们自定义adapter中数据position编号从0开始,也就是说与我们传进去的list的位置是一样的。 
而listview的onitemclick方法中:

public void onItemClick(AdapterView
  • 1

headView和footerView都可以响应onItemClick方法,headView的position为0,footerView的position最大。 
不过可以给headView和footerView设置OnClickListener来覆盖OnItemClick,这样,你点击headview或者footerView将触发OnClickListener而不是onItemClick(). 
在使用ListView.addHeaderView(),牵涉到布局管理问题,往往还会遇到类型转换异常: 
《java.lang.ClassCastException:android.widget.LinearLayoutLayoutParamscannotbecasttoandroid.widget.AbsListViewLayoutParamscannotbecasttoandroid.widget.AbsListViewLayoutParams》

其中本文中部分文字参考:http://892848153.iteye.com/blog/1923680,谢谢大神赐教!

#Requires AutoHotkey v2.0 #SingleInstance Force ; 创建GUI窗口 mainGui := Gui() ; 创建ListView(修正:使用数组作为列标题) headers := ["窗口标题", "窗口类", "窗口ID", "进程名", "进程路径", "样式", "扩展样式", "是否可见", "是否启用", "是否最大化", "是否最小化", "是否活动", "是否触摸键盘"] listView := mainGui.Add('ListView', 'w800 h400 r15 Grid Sort', headers) ; 添加其他控件 mainGui.Add('Button', 'w120 gExitApp', '退出程序') mainGui.Add('Text', 'x140 y+10', '每5秒自动刷新,数据已保存到:') textCtrl := mainGui.Add('Text', 'x270 yp w400', A_ScriptDir '\WindowList.csv') ; 显示窗口 mainGui.Show('Center') ; 初始化列宽 listView.ModifyCol(1, 200) listView.ModifyCol(2, 100) listView.ModifyCol(3, 80) listView.ModifyCol(4, 120) listView.ModifyCol(5, 250) listView.ModifyCol(6, 80) listView.ModifyCol(7, 80) listView.ModifyCol(8, 70) listView.ModifyCol(9, 70) listView.ModifyCol(10, 90) listView.ModifyCol(11, 90) listView.ModifyCol(12, 70) listView.ModifyCol(13, 70) ; 创建定时器,每5秒刷新一次 SetTimer RefreshWindowList, 5000 RefreshWindowList() ; 立即执行一次 ; 刷新窗口列表 RefreshWindowList() { global listView, textCtrl try { listView.Delete() ; 清空列表 ; 创建/覆盖CSV文件并写入标题行 FileDelete(A_ScriptDir '\WindowList.csv') FileAppend('窗口标题,窗口类,窗口ID,进程名,进程路径,样式,扩展样式,是否可见,是否启用,是否最大化,是否最小化,是否活动,是否触摸键盘`n', A_ScriptDir '\WindowList.csv', "CP936") ; 获取所有顶级窗口 windowList := WinGetList() ; 遍历所有窗口 for winID in windowList { ; 获取窗口标题和类名 winTitle := WinGetTitle(winID) winClass := WinGetClass(winID) ; 获取进程信息 processID := WinGetPID(winID) processName := ProcessExist(processID) processPath := ProcessPath(processID) ; 获取窗口样式 style := WinGetStyle(winID) exStyle := WinGetExStyle(winID) ; 获取窗口状态 WinGetPos(, , , , winID) ; 移除未使用的变量 winVisible := WinExist('ahk_id ' winID ' +Visible') winEnabled := WinExist('ahk_id ' winID ' +Enabled') winActive := WinExist('ahk_id ' winID ' +Active') ; 检查窗口是否最大化或最小化 isMaximized := (style & 0x00200000) ? "是" : "否" isMinimized := (style & 0x01000000) ? "是" : "否" ; 检查是否为TabTip(触摸键盘) isTabTip := (winClass = "TabTip") ? "是" : "否" ; 添加ListView(修正:使用数组添加行) listView.Add([winTitle, winClass, winID, processName, processPath, style, exStyle, winVisible ? "是" : "否", winEnabled ? "是" : "否", isMaximized, isMinimized, winActive ? "是" : "否", isTabTip]) ; 写入CSV文件(带引号处理特殊字符) FileAppend('"' winTitle '","' winClass '","' winID '","' processName '","' processPath '","' style '","' exStyle '","' winVisible '","' winEnabled '","' isMaximized '","' isMinimized '","' winActive '","' isTabTip '"`n', A_ScriptDir '\WindowList.csv', "CP936") } ; 更新时间戳 textCtrl.Value := "最后更新: " A_Now } catch as e { MsgBox "刷新错误: " e.Message } } ; 退出程序 ExitApp(*) { ExitApp } ; 窗口关闭事件 mainGui.OnEvent('Close', ExitApp) ; 辅助函数:设置ListView字体 LV_SetFont(listViewObj, fontName:='Segoe UI', fontSize:=9) { hFont := DllCall("CreateFont", "int", fontSize * 96 / 72, ; nHeight "int", 0, ; nWidth "int", 0, ; nEscapement "int", 0, ; nOrientation "int", 0, ; fnWeight "uint", 0, ; fdwItalic "uint", 0, ; fdwUnderline "uint", 0, ; fdwStrikeOut "uint", 0, ; fdwCharSet "uint", 3, ; fdwOutputPrecision "uint", 2, ; fdwClipPrecision "uint", 1, ; fdwQuality "uint", 0, ; fdwPitchAndFamily "str", fontName, ; lpszFace "ptr") DllCall("SendMessage", "ptr", listViewObj.Hwnd, "uint", 0x1103, "ptr", hFont, "ptr", 0) return hFont }
最新发布
06-07
简化下树形菜单加载的代码【 /// <summary> /// 加载数据初始化 /// </summary> private void LoadShowtab() { //dataGridView绑定 table.TableName = "DocData"; table.Columns.Add("DocID"); table.Columns.Add("DocName"); table.Columns.Add("DocAuto"); table.Rows.Add(Guid.NewGuid().ToString("N"), "激活", "未绑定列"); table.Rows.Add(Guid.NewGuid().ToString("N"), "未激活", "未绑定列"); dataGridView1.AutoGenerateColumns = false; dataGridView1.DataSource = table; //下拉框绑定 comboBox1.DataSource = table; comboBox1.ValueMember = "DocID"; comboBox1.DisplayMember = "DocName"; //listview成组 listView1.View = View.Details; listView1.ShowGroups = true; listView1.FullRowSelect = true; listView1.Groups.Add(new ListViewGroup("技术部", HorizontalAlignment.Left)); listView1.Groups.Add(new ListViewGroup("市场部", HorizontalAlignment.Left)); listView1.Columns.Clear(); listView1.Columns.Add("姓名", 120); listView1.Columns.Add("年龄", 80); listView1.Columns.Add("11", 80); ListViewItem item = new ListViewItem(new[] { "1","2","3"}); var group = listView1.Groups.Cast<ListViewGroup>().FirstOrDefault(g => g.Header == "技术部"); if (group != null) { item.Group = group; } listView1.Items.Add(item); //树形菜单递归 trees = new List<treeViewClass>() { new treeViewClass{父节点="A0",节点="A1",Name="食物" }, new treeViewClass{父节点="A1",节点="A_1",Name="馒头" }, new treeViewClass{父节点="A1",节点="A_2",Name="面" }, new treeViewClass{父节点="B0",节点="B1",Name="水果" }, new treeViewClass{父节点="B1",节点="B1_1",Name="榴莲" }, new treeViewClass{父节点="B1_1",节点="B1_2",Name="榴莲核" }, }; // 构建父节点字典(Key为父节点ID,Value为子节点集合) var parentDict = trees .GroupBy(t => t.父节点) .ToDictionary( g => g.Key, g => g.ToList() ); var allNodeIds = new HashSet<string>(trees.Select(t => t.节点)); var trees_ = trees.Where(t => !allNodeIds.Contains(t.父节点)); // 添加根节点(父节点不在所有节点中的项) foreach (var rootItem in trees_) { TreeNode rootNode = new TreeNode(rootItem.Name) { Tag = rootItem.节点 }; treeView1.Nodes.Add(rootNode); BuildChildNodes(rootNode, parentDict); } } private void BuildChildNodes(TreeNode parentNode, Dictionary<string, List<treeViewClass>> parentDict) { var currentId = parentNode.Tag?.ToString(); if (string.IsNullOrEmpty(currentId)) return; if (parentDict.TryGetValue(currentId, out var children)) { foreach (var child in children) { TreeNode childNode = new TreeNode(child.Name) { Tag = child.节点 }; parentNode.Nodes.Add(childNode); BuildChildNodes(childNode, parentDict); // 递归构建子树 } } }】
04-01
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值