Strange shift when Selecting text in richtext box

2020-07-16 12:03发布

问题:

When I use the RichTextBox.Select(int start, int length) function to select text in a rich text box that contains hyperlinks, there is some shift in the selection when I try to select any text that is after the hyperlink.

When I remove the hyperlink, the selection works properly.

Edit: I use RichTextBox 5 (the default in Visual Studio is version 4)

public class RichText50W : RichTextBox
{
    [DllImport("kernel32.dll", CharSet = CharSet.Auto)]
    static extern IntPtr LoadLibrary(string lpFileName);
    protected override CreateParams CreateParams
    {
        get
        {
            CreateParams prams = base.CreateParams;
            if (LoadLibrary("msftedit.dll") != IntPtr.Zero)
            {
                prams.ClassName = "RICHEDIT50W";
            }
            return prams;
        }
    }
}

The problem occurs only with richtextbox 5+.

These images show the problem and the difference in behavior.

If you try this make sure to set HideSelection property of richtextboxes to "false".

Select functions are displayed on the buttons.

For RTB5, 'e' in "Text" should be selected in the above image.

Apparently RTB5 selects some hidden text.

I need it to select based on the visible text only.

RTB 4 has problems when using tables, so I don't want to use it.

I use .NET 2.0

Edit: To try this in Visual Studio, start with the default richtextbox and change its declaration to RichText50W instead of RichTextBox

Also RichEdit 6 has the same issue.

Thanks

回答1:

From my own experience, version "RICHEDIT50W" is horribly broken when used with embedded hyperlinks or hidden text (using rtf codes \v \v0).

In your v5 box, the Text.Length property reports 14 characters — what it displays. The TextLength property reports 51 characters. The SelectionStart and SelectionLength properties all report the "hidden text" numbers, but the control does not give you a way to get at the hidden text any longer. It means the "text" and related "text selection" information becomes unusable when your rich text has hidden characters.

I think the only solution is to not use the "RICHEDIT50W" version if there will be hidden characters or browse the market for a better rich text control.



回答2:

Only a little late. This may or may not help, I haven't used this control yet. Following code is copied from http://www.codeproject.com/Messages/3401956/NET-Richedit-Control.aspx. Note the // Check Unicode or ANSI system and set appropriate ClassName.

using System;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace RichEditor
{
  public class RichTextBoxEx : RichTextBox
  {
    private IntPtr mHandle = IntPtr.Zero;

    protected override CreateParams CreateParams
    {
      get
      {
        //Prevent module being loaded multiple times.
        if (this.mHandle == IntPtr.Zero)
        {
          //load the library to obtain an instance of the RichEdit50 class.
          this.mHandle = LoadLibrary("msftedit.dll");
        }

        //If module loaded, reset ClassName.
        if (this.mHandle != IntPtr.Zero)
        {
          CreateParams cParams = base.CreateParams;

          // Check Unicode or ANSI system and set appropriate ClassName.
          if (Marshal.SystemDefaultCharSize == 1)
          {
            cParams.ClassName = "RichEdit50A";
          }
          else
          {
            cParams.ClassName = "RichEdit50W";
          }

          return cParams;
        }
        else // Module wasnt loaded, return default .NET RichEdit20 CreateParams.
        {
          return base.CreateParams;
        }
      }
    }


    ~RichTextBoxEx()
    {
      //Free loaded Library.
      if (mHandle != IntPtr.Zero)
      {
        FreeLibrary(mHandle);
      }
    }

    [DllImport("kernel32.dll", SetLastError = true)]
    private static extern IntPtr LoadLibrary(String lpFileName);

    [DllImport("kernel32.dll", SetLastError = true)]
    private static extern bool FreeLibrary(IntPtr hModule);
  }
}