c# - 突出显示 RichTextBox 的当前行

标签 c# winforms richtextbox highlight

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

那么,我们要做的是:

  1. 验证之前和当前的矩形不匹配,所以不要在同一区域绘制两次
  2. 使用控件背景色替换最后一行的高亮
  3. 用透明色高亮显示当前行
  4. 为每条应用线设置 mLastHighlight 的索引和矩形

但是,当移除高亮时,文本会被覆盖。应用高亮时不会发生这种情况。

一种解决方案是在重置背景颜色后重新绘制控件上的文本。尽管文本格式、选择颜色、字体样式、超链接等过滤起来很乏味。不是很优雅。

这导致更简单的解决方案,即刷新控件。尽管那会导致大量闪烁。也不接受。

有优雅的解决方案吗?我完全不明白为什么会这样。

编辑:编辑以反射(reflect) 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
}

最佳答案

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

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

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

Scintilla 控件不仅仅是 Win32 rich edit 控件的包装器。它必须执行很多自定义绘图才能发挥其魔力,而所有这些代码都很难靠您自己正确处理。这就是为什么这么多人首先使用 Scintilla 的原因。如果您需要它的功能集,我强烈建议您效仿。

无论如何,我实际上并不知道 Win32 rich edit 控件是否可以做到这一点。我不认为是,但我不能对这个事实发誓。我猜你可以通过设置选择颜色来破解它,但这似乎不是一个很好的解决方案。类似于 Daniel suggests here .我不是 Scintilla 专家,但在我未经训练的眼中,这看起来有点像基于 Scintilla 的代码的道德等价物,但是是为 Win32 rich edit 控件编写的(通过其 .NET WinForms 包装器)。

关于c# - 突出显示 RichTextBox 的当前行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11642861/

相关文章:

c# - 正则表达式/linq 用计数替换连续字符

c# - 更高效地编辑 XML 代码注释的工具?

c# - 如何在 Xamarin.Android/Monodroid 中使用 Android KitKat EvaluateJavascript 并检索结果?

c# - 如何在T4模板中 "read"EF6 EntityTypeConfiguration?

c# - 当字符串为空时将 NULL 插入 SQL

c# - 从 RichTextBox 复制选定的文本

c# - 将子控件 MouseMove 事件无缝转发给父控件

c# - 按下按钮后运行事件

silverlight - Silverlight RichTextBox 中的 XAML 标记大小过大

c# - 禁用 RichTextBox 或 TextBox 中的选择突出显示