列表框水平滚动条无法正确更新

时间:2012-04-10 16:18:19

标签: c# winforms listbox scrollbar

我有一个表单,其中包含一个列表框控件,其中包含许多其他控件,用于确定列表框内容。将项目添加到列表框会正确设置滚动条的范围,但更新列表框中的项目(通过this.lstResources.Items [index] = myUri)会导致滚动条的范围减小超过最大项目&#39 ; s宽度,切掉最后几个字符。滚动条仍然有效,但以这种方式更新列表框会导致列表中项目的意外且不可接受的滚动范围。以下是我实现列表框的方式:

public System.Windows.Forms.ListBox lstResources;
this.lstResources = new System.Windows.Forms.ListBox();

this.lstResources.FormattingEnabled = true;
this.lstResources.HorizontalScrollbar = true;
this.lstResources.Location = new System.Drawing.Point(331, 122);
this.lstResources.Name = "lstResources";
this.lstResources.Size = new System.Drawing.Size(307, 316);
this.lstResources.TabIndex = 8;
this.lstResources.Click += new System.EventHandler(this.ResourcesPage.LstResources_Click);

我在此列表框中执行的更改操作如下:

this.parentClassReference.lstResources.Items.Add(myUri);
this.parentClassReference.lstResources.Items[index] = myUri;
this.parentClassReference.lstResources.Items.RemoveAt(index);

我在表单和列表框控件上尝试了Refresh()和Update(),但都没有任何效果。我已经查看了列表框滚动条问题,但似乎都没有这个特殊的重绘问题。

这应该是这样的:
enter image description here

这是实际发生的事情:
enter image description here

我有一种感觉,我错过了一些显而易见的东西,或者错误地更改了这些东西,但我对这个问题的想法很新鲜。我对C#和UI设计并不陌生,但我不能说我知道所有细节控制操作。非常感谢任何帮助。

编辑1:这是一个应该重现问题的示例表单。我开始怀疑StringBuilder可能与更新部分有关,但它也用在btnAdd_Click代码中......

public partial class SampleListBoxForm : Form
{
    public Dictionary<string, string> CurrentUriQuery { get; set; }

    public SampleListBoxForm()
    {
        this.CurrentUriQuery = new Dictionary<string, string>();
        this.InitializeComponent();
    }

    private void btnAdd_Click(object sender, EventArgs e)
    {
        this.UpdateParams("sampleApp");
        this.listBox1.Items.Add(this.GenerateURI());
        this.listBox1.Refresh();
    }

    private void btnUpdate_Click(object sender, EventArgs e)
    {
        int index = 0;
        this.UpdateParams("sampleApp2");
        for (; index < this.listBox1.Items.Count; index++)
        {
            this.listBox1.Items[index] = this.GenerateURI();
        }
    }

    private void UpdateParams(string filename)
    {
        this.CurrentUriQuery = new Dictionary<string, string>();
        this.CurrentUriQuery["filename"] = filename;
        this.CurrentUriQuery["url"] = @"C:\Users\me\Desktop\" + filename;
        this.CurrentUriQuery["type"] = "dynamicType";
        this.CurrentUriQuery["p1"] = "foo";
        this.CurrentUriQuery["p2"] = "bar";
        this.CurrentUriQuery["p3"] = "stuff";
        this.CurrentUriQuery["p4"] = "test";
    }

    private string GenerateURI()
    {
        StringBuilder currentUri = new StringBuilder();
        bool firstParam = true;
        currentUri.Append(this.CurrentUriQuery["filename"]);
        foreach (KeyValuePair<string, string> pair in this.CurrentUriQuery)
        {
            if (pair.Key != "url" && pair.Key != "filename" && pair.Value != string.Empty && pair.Value != "0")
            {
                if (firstParam)
                {
                    currentUri.Append("?");
                    firstParam = false;
                }
                else
                {
                    currentUri.Append("&");
                }

                currentUri.Append(pair.Key);
                currentUri.Append("=");
                currentUri.Append(pair.Value.Replace(" ", ""));
            }
        }

        return currentUri.ToString();
    }

    public string[] ExtractPathAndQueryFromUriString(string uriString)
    {
        string[] pathAndQuery = new string[2];
        if (uriString.Contains('?'))
        {
            pathAndQuery[0] = uriString.Split('?')[0];
            pathAndQuery[1] = uriString.Split('?')[1];
        }
        else if (uriString.Contains("%3F"))
        {
            string[] stringSeparator = new string[] { "%3F" };
            pathAndQuery[0] = uriString.Split(stringSeparator, StringSplitOptions.None)[0];
            pathAndQuery[1] = uriString.Split(stringSeparator, StringSplitOptions.None)[1];
        }
        else
        {
            pathAndQuery[0] = uriString;
            pathAndQuery[1] = string.Empty;
        }

        return pathAndQuery;
    }
}

2 个答案:

答案 0 :(得分:3)

问题似乎与&符号有关。添加项目时,它似乎正确测量。更新项目时,它不会。

这个丑陋的黑客似乎是:记录索引,删除项目,将项目插回到记录的索引中。

或切换到DrawMode = OwnerDrawFixed并计算您添加或更改的每个项目的HorizontalExtent值。

private void ResizeListBox() {
  int maxWidth = 0;

  for (int i = 0; i < listBox1.Items.Count; i++) {
    int testWidth = TextRenderer.MeasureText(listBox1.Items[i].ToString(), 
                                             listBox1.Font, listBox1.ClientSize,
                                             TextFormatFlags.NoPrefix).Width;
    if (testWidth > maxWidth)
      maxWidth = testWidth;
  }

  listBox1.HorizontalExtent = maxWidth;
}

不幸的是,您必须在每次更改时调用它:

private void btnAdd_Click(object sender, EventArgs e) {
  this.UpdateParams("sampleApp");
  this.listBox1.Items.Add(this.GenerateURI());
  ResizeListBox();
}

private void btnUpdate_Click(object sender, EventArgs e) {
  this.UpdateParams("sampleApp");
  for (int index = 0; index < this.listBox1.Items.Count; index++) {
    this.listBox1.Items[index] = this.GenerateURI();
  }
  ResizeListBox();
}

这是一个快速而又脏的DrawItem版本:

private void listBox1_DrawItem(object sender, DrawItemEventArgs e) {
  e.DrawBackground();
  if (e.Index > -1) {
    Color textColor = SystemColors.WindowText;
    if ((e.State & DrawItemState.Selected) == DrawItemState.Selected) {
      textColor = SystemColors.HighlightText;
    }
    TextRenderer.DrawText(e.Graphics, listBox1.Items[e.Index].ToString(), 
                          listBox1.Font, e.Bounds, 
                          textColor, Color.Empty, TextFormatFlags.NoPrefix);
  }
}

答案 1 :(得分:0)

public static void ResizeListBox()
{
    YourListView.HorizontalExtent = TextRenderer.MeasureText(YourListView.Items.ToString(),
                                                    YourListView.Font, YourListView.ClientSize,
                                                    TextFormatFlags.NoPrefix).Width;           
}

在表单加载时调用此函数...这可能会解决您的问题。 在这里,只检查第一个列表框元素的宽度,或者您可以使用循环找到最大宽度。