2

我有一个未绑定的 DataGridView(在 VS 2008 中),其中一列包含一个文件路径。我想在 ColumnWidthChanged 事件上使用 TextRenderer 类来格式化字符串,而无需实际修改基础值。问题是表格的内容在表单关闭时被保存,我不想保存格式化的值。我想我太深了,看不到明显的解决方案,所以我依靠你们来指出:-)。

这个想法是显示这个:

C:\Program Files\Microsoft Visual Studio 8\SDK\v2.0\Bin\gacutil.exe

...这样(取决于列的宽度):

C:\Program Files\Microso…\gacutil.exe


看来我说得太早了。我从 TextRenderer.MeasureText() 得到了一些非常奇怪的结果。如果我将路径值硬编码为“C:\Documents and Settings\jluce\My Documents\Downloads”,它最终会变成 C:\Documents and Settings\jluce\M...\Downloads\0wnloads”。如果我不这样做t 对其进行硬编码(如下所示),每次我调整列大小时它都会进一步损坏。

这是几次调整大小后的样子: 截图

这是我目前正在做的事情。

  if (e.ColumnIndex == 1)
  {
    foreach (DataGridViewRow Row in mappingsDataGrid.Rows)
    {
      string Path = (string)Row.Cells[1].Value;
      Path = Path.Trim();

      TextRenderer.MeasureText(Path, e.CellStyle.Font,
        new Size(mappingsDataGrid.Columns[e.ColumnIndex].Width, Row.Height),
          TextFormatFlags.ModifyString | TextFormatFlags.PathEllipsis);

      e.Value = Path;
    }
  }

这只会越来越奇怪!

我设法通过遍历每个字符并删除坏字符来解决损坏字符串的问题。然而,现在我遇到了一个更疯狂的问题。我在事件处理程序中分配的局部变量在调用之间保留其值。

以下是相关代码:

     string Path = ""; // <-- #1
     Path = "C:\\Documents and Settings\\jluce\\My Documents\\Downloads"; // <-- #2

      TextRenderer.MeasureText(Path, Row.Cells[1].Style.Font,
        new Size((mappingsDataGrid.Columns[e.Column.Index].Width), Row.Height),
          TextFormatFlags.ModifyString | TextFormatFlags.PathEllipsis);

      // Left out code that strips invalid chars

      Row.Cells[1].Value = Path; // <-- #3
      Path = null;

第一次调整列大小(请参阅上面评论中的#):

  1. 在此行之后路径包含“”。
  2. 在此行之后,路径包含字符串,就像上面显示的那样。
  3. 路径包含应截断的文件路径(即“C:\Documents and Setti...\Downloads”)

第二次调整大小:

  1. 在此行之后路径包含“”,因为它应该。
  2. 在此行之后,路径包含“C:\Documents and Set...\Downloads\0 Documents\Downloads”,这是在我去除无效字符之前的上一次迭代中的无效值(此处显示为 '\0')! !
  3. 现在的路径是 FUBAR,因为我从一个搞砸的字符串开始,它只会变得更糟。

当我明确地为它分配一个值时,为什么会从上一个函数调用中为 Path 分配无效值(在正确分配一个空字符串之后!)?!!!!!!

4

2 回答 2

2

您需要使用 CellFormatting 事件在打印之前更改给定值(原始对象值不会被修改)。在您的情况下,您可以通过验证 e.ColumnIndex 变量并更改 e.Value 文本来检查它是否是正确的列,如下所示:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        dataGridView1.DataSource = new List<Person>(new Person[] { new Person() { Name = "André", Adress = "Brazil" } });
    }

    private void dataGridView1_CellFormatting(object sender, DataGridViewCellFormattingEventArgs e)
    {
        e.Value = e.Value + " modified";
    }
}

class Person
{
    public String Name { get; set; }
    public String Adress { get; set; }
}
于 2009-10-22T23:01:07.643 回答
2

TextRenderer.MeasureText方法是一个讨厌的方法——它改变了作为参数传递的实际字符串,所以它改变了 DataGridView 引用的实际字符串。它实际上使 .Net 字符串可变

似乎这种荒谬的方法并没有改变Length字符串的实际值,而只是覆盖了其中一个字符\0以指示字符串的结尾(就像纯 C 中的以 null 结尾的字符串一样)。那是一些有趣的东西!

这可能会对您的应用程序的稳定性产生严重影响。如果您考虑到 .Net 使用string interning,您可能会开始得到各种奇怪的结果,因为您确实注意到您的字符串常量似乎不再是恒定的。

第一步是创建字符串的副本(具有相同字符的新实例):

string Path = String.Copy(e.Value as string ?? "");

代替

string Path = (string)Row.Cells[1].Value;

这将确保无论做什么TextRenderer,原始字符串都将保持不变。

之后,您需要去掉修改后的字符串中的空字符。

通过做这个:

if (Path.IndexOf('\0') >= 0)
   e.Value = Path.Substring(0, Path.IndexOf('\0'));
else
   e.Value = Path;

您将创建一个干净的、修改过的字符串的新实例(使我们的临时Path副本不被垃圾收集引用)。

于 2009-10-23T16:03:14.980 回答