c# - 使用带有固定位置文本的 ScrollableControl 的自定义控件

标签 c# .net winforms user-controls

我正在构建将用于显示图 block map 的自定义用户控件,我选择了 ScrollableControl 作为基类,因为我希望在控件中包含滚动条。

我已经成功创建了绘制逻辑,负责仅绘制所需的元素。

现在我正在尝试添加始终在同一位置可见的静态文本(在我的例子中是左上角带有红色文本的白色框):

enter image description here

这在上面的 gif 中并不清晰可见,但是当我使用鼠标或滚动条滚动时,白色框会闪烁并跳跃一点。

我的问题是我应该如何更改代码以在可滚动内容之上包含可滚动内容和固定位置内容?

ScrollableControl 作为基类是不错的选择吗?

下面是我的代码:

class TestControl : ScrollableControl
{
    private int _tileWidth = 40;
    private int _tileHeight = 40;
    private int _tilesX = 20;
    private int _tilesY = 20;

    public TestControl()
    {
        SetStyle(ControlStyles.ResizeRedraw, true);
        SetStyle(ControlStyles.UserPaint, true);
        SetStyle(ControlStyles.AllPaintingInWmPaint, true);
        SetStyle(ControlStyles.Opaque, true);
        SetStyle(ControlStyles.OptimizedDoubleBuffer, true);
        UpdateStyles();
        ResizeRedraw = true;
        AutoScrollMinSize = new Size(_tilesX * _tileWidth, _tilesY * _tileHeight);

        Scroll += (sender, args) => { Invalidate(); };
    }

    protected override void OnPaint(PaintEventArgs e)
    {
        base.OnPaint(e);
        e.Graphics.FillRectangle(new SolidBrush(BackColor), ClientRectangle);
        e.Graphics.TranslateTransform(AutoScrollPosition.X, AutoScrollPosition.Y);
        e.Graphics.SmoothingMode = SmoothingMode.AntiAlias;

        var offsetX = (AutoScrollPosition.X * -1) / _tileWidth;
        var offsetY = (AutoScrollPosition.Y * -1) / _tileHeight;

        var visibleX = Width / _tileWidth + 2;
        var visibleY = Height / _tileHeight + 2;

        var x = Math.Min(visibleX + offsetX, _tilesX);
        var y = Math.Min(visibleY + offsetY, _tilesY);

        for (var i = offsetX; i < x; i++)
        {
            for (var j = offsetY; j < y; j++)
            {
                e.Graphics.FillRectangle(Brushes.Beige, new Rectangle(i*_tileWidth, j*_tileHeight, _tileWidth, _tileHeight));
                e.Graphics.DrawString(string.Format("{0}:{1}", i, j), Font, Brushes.Black, new Rectangle(i * _tileWidth, j * _tileHeight, _tileWidth, _tileHeight));
            }
        }

        using (var p = new Pen(Color.Black))
        {
            for (var i = offsetX + 1; i < x; i++)
            {
                e.Graphics.DrawLine(p, i*_tileWidth, 0, i*_tileWidth, y*_tileHeight);
            }

            for (var i = offsetY + 1; i < y; i++)
            {
                e.Graphics.DrawLine(p, 0, i*_tileHeight, x*_tileWidth, i*_tileHeight);
            }
        }

        e.Graphics.FillRectangle(Brushes.White, AutoScrollPosition.X * -1, AutoScrollPosition.Y * -1, 35, 14);
        e.Graphics.DrawString("TEST", DefaultFont, new SolidBrush(Color.Red), AutoScrollPosition.X * -1, AutoScrollPosition.Y * -1);
    }
}

编辑:
我搜索了一下,发现 UserControl 具有类似的功能 - https://www.codeproject.com/Articles/16009/A-Much-Easier-to-Use-ListView在阅读了 Control 的作者博客 http://objectlistview.sourceforge.net/cs/blog1.html#blog-overlays 后,我发现他使用的是位于控件顶部的透明表单。 我真的很想避免这种情况,但我的控件上仍然有覆盖层。

最佳答案

您正在与名为“拖动时显示窗口内容”的 Windows 系统选项进行斗争。默认情况下始终打开,this web page展示如何关闭它。

解决了问题,但它不是您可以依赖的东西,因为它会影响所有应用中的所有可滚动窗口。要求用户为你关闭它是不现实的,用户喜欢这个选项,所以他们会忽略你。他们没有提供在特定窗口关闭它的选项,这是一个相当大的疏忽。在信息亭应用程序中,这是一个不错的解决方案。

简单来说,该选项的工作方式是 Windows 本身使用 ScrollWindowEx() 滚动窗口内容。 winapi函数。使用窗口内容的 bitblt 来移动像素,并且只为滚动显示的窗口部分生成绘制请求。通常只有几行像素,因此完成速度非常快。问题是,bitblt 也会移动你的固定像素。重画将它们移回原处。值得注意的是,人眼对这样的运动非常敏感,这有助于在过去的几百万年里避免成为狮子的午餐。

您必须消除 ScrollWindowsEx() 的麻烦,防止它移动像素,即使您无法阻止它被调用。这需要一把沉重的大锤,LockWindowUpdate() 。您将在 this post 中找到代码.

using System.Runtime.InteropServices;
...

    protected override void OnScroll(ScrollEventArgs e) {
        if (e.Type == ScrollEventType.First) {
            LockWindowUpdate(this.Handle);
        }
        else {
            LockWindowUpdate(IntPtr.Zero);
            this.Update();
            if (e.Type != ScrollEventType.Last) LockWindowUpdate(this.Handle);
        }
    }

    [DllImport("user32.dll", SetLastError = true)]
    private static extern bool LockWindowUpdate(IntPtr hWnd);

不太漂亮,使用单独的 Label 控件应该开始听起来很有吸引力。

关于c# - 使用带有固定位置文本的 ScrollableControl 的自定义控件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43208630/

相关文章:

c# - 如果需要,将方案添加到 URL

c# - Kinect 语音没有得到识别器

C# 隐式转换和从 Object 的转换

c# - 在父级的中心显示对话框

c# - 关于DLR的几个问题

c# - Function C++ 到 C#(安全代码)

.net - 使用自定义异常类抛出多个异常

.net - 寻找一种可以轻松与 .NET 集成的简单脚本语言

c# - 我的 WinForm 正在获取 SQL 注入(inject)

c# - 在解决方案中显示类名的 ComboBox 项