我正在为那些尚未意识到像ListBox
这样的控件不必包含字符串的人做一个例子;他一直在存储格式化的字符串并跳过复杂的解析箍以将数据从ListBox
中取出来,我想告诉他有更好的方法。
我注意到如果我在ListBox
中存储了一个对象,然后更新了影响ToString
的值,则ListBox
不会自行更新。我尝试在控件上调用Refresh
和Update
,但都不起作用。这是我正在使用的示例的代码,它要求您将列表框和按钮拖到窗体上:
Public Class Form1
Protected Overrides Sub OnLoad(ByVal e As System.EventArgs)
MyBase.OnLoad(e)
For i As Integer = 1 To 3
Dim tempInfo As New NumberInfo()
tempInfo.Count = i
tempInfo.Number = i * 100
ListBox1.Items.Add(tempInfo)
Next
End Sub
Private Sub Button1_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Button1.Click
For Each objItem As Object In ListBox1.Items
Dim info As NumberInfo = DirectCast(objItem, NumberInfo)
info.Count += 1
Next
End Sub
End Class
Public Class NumberInfo
Public Count As Integer
Public Number As Integer
Public Overrides Function ToString() As String
Return String.Format("{0}, {1}", Count, Number)
End Function
End Class
我认为问题可能是使用字段并尝试实现 INotifyPropertyChanged ,但这没有任何效果。 (我使用字段的原因是因为它是一个例子,我不想添加几十行与我正在演示的主题无关。)
老实说,我以前从未尝试过更新这样的物品;在过去,我一直在添加/删除项目,而不是编辑它们。所以我从来没有注意到我不知道如何使这项工作。
那么我错过了什么?
答案 0 :(得分:33)
当我需要一个更新的列表框时,我会使用这个类。
更新列表中的对象,然后调用其中一个包含的方法,具体取决于您是否有索引。如果要更新列表中包含的对象,但没有索引,则必须调用RefreshItems并更新所有项目。
public class RefreshingListBox : ListBox
{
public new void RefreshItem(int index)
{
base.RefreshItem(index);
}
public new void RefreshItems()
{
base.RefreshItems();
}
}
答案 1 :(得分:29)
lstBox.Items[lstBox.SelectedIndex] = lstBox.SelectedItem;
答案 2 :(得分:24)
BindingList处理自己更新绑定。
using System;
using System.ComponentModel;
using System.Windows.Forms;
namespace TestBindingList
{
public class Employee
{
public string Name { get; set; }
public int Id { get; set; }
}
public partial class Form1 : Form
{
private BindingList<Employee> _employees;
private ListBox lstEmployees;
private TextBox txtId;
private TextBox txtName;
private Button btnRemove;
public Form1()
{
InitializeComponent();
FlowLayoutPanel layout = new FlowLayoutPanel();
layout.Dock = DockStyle.Fill;
Controls.Add(layout);
lstEmployees = new ListBox();
layout.Controls.Add(lstEmployees);
txtId = new TextBox();
layout.Controls.Add(txtId);
txtName = new TextBox();
layout.Controls.Add(txtName);
btnRemove = new Button();
btnRemove.Click += btnRemove_Click;
btnRemove.Text = "Remove";
layout.Controls.Add(btnRemove);
Load+=new EventHandler(Form1_Load);
}
private void Form1_Load(object sender, EventArgs e)
{
_employees = new BindingList<Employee>();
for (int i = 0; i < 10; i++)
{
_employees.Add(new Employee() { Id = i, Name = "Employee " + i.ToString() });
}
lstEmployees.DisplayMember = "Name";
lstEmployees.DataSource = _employees;
txtId.DataBindings.Add("Text", _employees, "Id");
txtName.DataBindings.Add("Text", _employees, "Name");
}
private void btnRemove_Click(object sender, EventArgs e)
{
Employee selectedEmployee = (Employee)lstEmployees.SelectedItem;
if (selectedEmployee != null)
{
_employees.Remove(selectedEmployee);
}
}
}
}
答案 3 :(得分:16)
typeof(ListBox).InvokeMember("RefreshItems",
BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.InvokeMethod,
null, myListBox, new object[] { });
答案 4 :(得分:9)
在数据源和列表框的datasource属性之间使用datasource属性和BindingSource对象。然后刷新它。
更新添加了示例。
像这样:
Public Class Form1
Private datasource As New List(Of NumberInfo)
Private bindingSource As New BindingSource
Protected Overrides Sub OnLoad(ByVal e As System.EventArgs)
MyBase.OnLoad(e)
For i As Integer = 1 To 3
Dim tempInfo As New NumberInfo()
tempInfo.Count = i
tempInfo.Number = i * 100
datasource.Add(tempInfo)
Next
bindingSource.DataSource = datasource
ListBox1.DataSource = bindingSource
End Sub
Private Sub Button1_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Button1.Click
For Each objItem As Object In datasource
Dim info As NumberInfo = DirectCast(objItem, NumberInfo)
info.Count += 1
Next
bindingSource.ResetBindings(False)
End Sub
End Class
Public Class NumberInfo
Public Count As Integer
Public Number As Integer
Public Overrides Function ToString() As String
Return String.Format("{0}, {1}", Count, Number)
End Function
End Class
答案 5 :(得分:1)
如果从ListBox派生,则可以调用RefreshItem保护方法。只需在您自己的类型中重新公开此方法。
public class ListBox2 : ListBox {
public void RefreshItem2(int index) {
RefreshItem(index);
}
}
然后更改您的设计器文件以使用您自己的类型(在本例中为ListBox2)。
答案 6 :(得分:0)
这有点不专业,但它确实有效。 我刚删除并添加了该项目(也再次选中)。 列表是根据“显示和更改”属性排序的,所以再次对我来说很好。副作用是引发附加事件(索引已更改)。
if (objLstTypes.SelectedItem != null)
{
PublisherTypeDescriptor objType = (PublisherTypeDescriptor)objLstTypes.SelectedItem;
objLstTypes.Items.Remove(objType);
objLstTypes.Items.Add(objType);
objLstTypes.SelectedItem = objType;
}
答案 7 :(得分:0)
如果您使用类似的绘制方法:
private void listBox1_DrawItem(object sender, DrawItemEventArgs e)
{
e.DrawBackground();
e.DrawFocusRectangle();
Sensor toBeDrawn = (listBox1.Items[e.Index] as Sensor);
e.Graphics.FillRectangle(new SolidBrush(toBeDrawn.ItemColor), e.Bounds);
e.Graphics.DrawString(toBeDrawn.sensorName, new Font(FontFamily.GenericSansSerif, 14, FontStyle.Bold), new SolidBrush(Color.White),e.Bounds);
}
传感器是我的班级。
因此,如果我在某处更改了类Color
,您只需将其更新为:
int temp = listBoxName.SelectedIndex;
listBoxName.SelectedIndex = -1;
listBoxName.SelectedIndex = temp;
Color
将更新,只是另一种解决方案:)
答案 8 :(得分:-1)
我对vb.net了解不多,但在C#中你应该使用数据源,然后通过调用listbox.bind()
绑定它就可以了。
答案 9 :(得分:-3)
如果objLstTypes是您的ListBox名称 使用 objLstTypes.Items.Refresh(); 希望这有效......