突出显示RichTextBox的当前行

Mar*_*k S 5 c# highlight richtextbox winforms

这通过将透明颜色绘制为当前行上的背景颜色来突出显示每条线的整个宽度.线路切换时,将恢复原始背景颜色.

所以,我们想要做的是:

  1. 验证前一个和当前矩形不匹配,因此不要两次绘制相同的区域
  2. 使用控件背景颜色替换最后一行的突出显示
  3. 使用透明颜色突出显示当前行
  4. 设置mLastHighlight每个应用行的索引和矩形

但是,删除突出显示时,文本会被涂上.应用突出显示时不会发生这种情况.

一种解决方案是在重置背景颜色后重新重新显示控件上的文本.虽然文本格式,选择颜色,字体样式,超链接等将是繁琐的过滤.不是很优雅.

这导致更简单的解决方案,刷新控制.虽然那会导致大量的闪烁.也不接受.

有优雅的解决方案吗?我完全不知道为什么会这样.

编辑:编辑以反映Code Gray的回应.

using System;

public class RTBHL : RichTextBox
{
    private LastHighlight mLastHighlight = new LastHighlight(0, Rectangle.Empty);

    private class LastHighlight
    {
        public int mCharIndex;
        public Rectangle mRectangle;

        public LastHighlight(int index, Rectangle r)
        {
            mCharIndex = index;
            mRectangle = r;
        }
    }

    public void PaintLineHighlight()
    {
        using (Graphics g = this.CreateGraphics)
        {
            // highlight color
            Color c = Color.Beige;
            // current pen color
            Pen cp = new Pen(Color.Beige);
            // color for removing highlight
            Pen lp = new Pen(this.BackColor);
            // brush for removing highlight
            SolidBrush lb = new SolidBrush(this.BackColor);
            // brush for applying highlight
            SolidBrush cb = new SolidBrush(Color.FromArgb(64, c.R, c.G, c.B));
            // index of the current line
            int index = this.GetFirstCharIndexOfCurrentLine;
            // rectangle to specify which region to paint too
            Rectangle r = new Rectangle();

            // specify dimensions
            r.X = 0;
            r.Y = this.GetPositionFromCharIndex(index).Y;
            r.Width = this.HorizontalScrollBarWidth;
            r.Height = Convert.ToInt32(this.Font.Height * this.ZoomFactor);

            // this will always be true unless the current line remains the same
            if (!(mLastHighlight.mCharIndex == index) && !(mLastHighlight.mRectangle == r))
            {
                // remove the last highlight. regardless of the brush specified, white is always applied, and the text is painted over
                g.DrawRectangle(lp, mLastHighlight.mRectangle);
                g.FillRectangle(lb, mLastHighlight.mRectangle);
                // apply highlight to the current line
                g.DrawRectangle(cp, r);
                g.FillRectangle(cb, r);
            }

            mLastHighlight = new LastHighlight(index, r);
        }
    }

#region RichScrollBars
    [DllImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    private static extern bool GetScrollInfo(IntPtr hWnd, int fnBar, ref SCROLLINFO si);

    [StructLayout(LayoutKind.Sequential)]
    public class SCROLLINFO
    {
        public int cbSize;
        public int fMask;
        public int nMin;
        public int nMax;
        public int nPage;
        public int nPos;
        public int nTrackPos;
        public SCROLLINFO()
        {
            this.cbSize = Marshal.SizeOf(typeof(SCROLLINFO));
        }

        public SCROLLINFO(int mask, int min, int max, int page, int pos)
        {
            this.cbSize = Marshal.SizeOf(typeof(SCROLLINFO));
            this.fMask = mask;
            this.nMin = min;
            this.nMax = max;
            this.nPage = page;
            this.nPos = pos;
        }
    }

    private const int SIF_ALL = 0X17;
    private const int SB_HORZ = 0;
    private const int SB_VERT = 1;

    public int HorizontalScrollBarWidth()
    {
        SCROLLINFO si = new SCROLLINFO() {fMask = SIF_ALL};
        GetScrollInfo(this.Handle, SB_HORZ, si);

        return Math.Max(si.nMax, this.Width);
    }

    public int VerticalScrollBarHeight()
    {
        SCROLLINFO si = new SCROLLINFO() {fMask = SIF_ALL};
        GetScrollInfo(this.Handle, SB_VERT, si);

        return Math.Max(si.nMax, this.Height);
    }
#endregion
}
Run Code Online (Sandbox Code Playgroud)

Cod*_*ray 3

这里的问题是您正在复制的代码是为 Scintilla 设计的。这些SCI_*常量由 Scintilla 标头内部定义,它们引用的消息仅对 Scintilla 控件有意义。

将这些消息发送到本机 Win32 Rich Edit 控件不会执行任何操作,因为它不是为处理这些消息而设计的。(或者更糟糕的是,一个或多个常量碰巧与富编辑控件确实SCI_*识别的一个或多个消息标识符发生冲突,从而产生一些潜在的有趣行为。)

除非您实际上在项目中使用 Scintilla 编辑控件(您说过您不想这样做),否则该代码不会执行任何有趣的操作。它不是为 Win32 rich edit 控件编写的,而是为与 Scintilla 控件交互而编写的。

Scintilla 控件不仅仅是 Win32 Rich Edit 控件的包装。它必须进行大量自定义绘图才能发挥其魔力,而所有这些代码都很难由您自己完成。这就是为什么这么多人首先使用 Scintilla。如果您需要它的功能集,我强烈建议您效仿。

不管怎样,我实际上不知道这是否可以通过 Win32 rich edit 控件实现。我不这么认为,但我不能发誓这一事实。我想你可以通过设置选择颜色来破解它,但这似乎不是一个很好的解决方案。像丹尼尔这样建议的东西在这里。我不是 Scintilla 专家,但在我未经训练的眼中,这看起来有点像基于 Scintilla 的代码,但它是为 Win32 富编辑控件编写的(通过其 .NET WinForms 包装器)。