如何同步两个多行文本框的滚动?

Posted

技术标签:

【中文标题】如何同步两个多行文本框的滚动?【英文标题】:How can I sync the scrolling of two multiline textboxes? 【发布时间】:2011-04-18 21:37:47 【问题描述】:

如何在 C# (WinForms) 中同步两个多行文本框的滚动?

当您在 TextBox A 中向上/向下滚动一行时,TextBox B 也应该向上/向下滚动。 反之亦然。

如果没有自定义控件,这是否可以实现?

【问题讨论】:

除非您告诉我们您使用的是什么类型的 GUI 框架,否则无法回答。 【参考方案1】:

是的,您必须创建一个自定义文本框,以便检测它的滚动。诀窍是将滚动消息传递给另一个文本框,使其同步滚动。这只有在其他文本框大小相同且行数相同时才有效。

向您的项目添加一个新类并粘贴如下所示的代码。编译。将工具箱顶部的两个新控件拖放到表单上。将 Buddy 属性设置为两者上的另一个控件。运行,在它们中输入一些文本,然后在拖动滚动条时观察它们同步滚动。

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

class SyncTextBox : TextBox 
    public SyncTextBox() 
        this.Multiline = true;
        this.ScrollBars = ScrollBars.Vertical;
    
    public Control Buddy  get; set; 

    private static bool scrolling;   // In case buddy tries to scroll us
    protected override void WndProc(ref Message m) 
        base.WndProc(ref m);
        // Trap WM_VSCROLL message and pass to buddy
        if (m.Msg == 0x115 && !scrolling && Buddy != null && Buddy.IsHandleCreated) 
            scrolling = true;
            SendMessage(Buddy.Handle, m.Msg, m.WParam, m.LParam);
            scrolling = false;
        
    
    [DllImport("user32.dll", CharSet = CharSet.Auto)]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, IntPtr lp);

【讨论】:

@Hans Passant :我想实现类似但在两个列表视图的情况下。我尝试使用此代码,但它不起作用。我需要在其中添加一些东西吗? @Hans Passant:Gr8 伙计,实际上我在错误的地方设置了好友财产。 gr8 工作。 对编辑答案以包括鼠标滚轮和使用光标键滚动(+主页/向上/向下等)有任何想法吗? 已经可以了,唯一的错误就是不尝试。 当您拖动滚动幻灯片并按下鼠标左键时,您如何上下移动幻灯片,好友列表视图不会对滚动做出反应?【参考方案2】:

你可以改变这一行:

if (m.Msg == 0x115) && !scrolling && Buddy != null && Buddy.IsHandleCreated)

到这里:

if ((m.Msg == 0x115 || m.Msg==0x20a) && !scrolling && Buddy != null && Buddy.IsHandleCreated)

它也将支持使用鼠标滚轮滚动。

【讨论】:

不幸的是,虽然这似乎可行,但滚动与鼠标滚轮不同步,并且主文本框的滚动速度比伙伴快。 当您拖动滚动幻灯片并按下鼠标左键时,您如何上下移动幻灯片,好友列表视图不会对滚动做出反应?【参考方案3】:

Hans Passant 的解决方案很有魅力,但我需要一个带有水平和垂直滚动条的 RichTextBox。如果您扩展 RichTextBox 而不是 TextBox,则需要相应地更改 ScrollBars 属性(我使用的是 RichTextBoxScrollBars.Both)。

如果您还想同步水平滚动,请查找(m.Msg == 0x115) || (m.Msg == 0x114)

【讨论】:

【参考方案4】:

Hans Passant 的解决方案非常棒。但是我需要同步三个文本框而不仅仅是两个。

所以我对其进行了一些修改 - 但所有的信任都应该归功于 Hans,如果没有他的工作,我什至不可能接近 - 我想我会把它发回这里以防其他人需要。

SyncBox 类:

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

class SyncTextBox : TextBox

    public SyncTextBox()
    
        this.Multiline = true;
        this.ScrollBars = ScrollBars.Vertical;
    

    public Control[] Buddies  get; set; 

    private static bool scrolling;   // In case buddy tries to scroll us
    protected override void WndProc(ref Message m)
    
        base.WndProc(ref m);
        // Trap WM_VSCROLL message and pass to buddy
        if (Buddies != null)
        
            foreach (Control ctr in Buddies)
            
                if (ctr != this)
                
                    if ((m.Msg == 0x115 || m.Msg == 0x20a) && !scrolling && ctr.IsHandleCreated)
                    
                        scrolling = true;
                        SendMessage(ctr.Handle, m.Msg, m.WParam, m.LParam);
                        scrolling = false;
                    
                
            
        
    
    [DllImport("user32.dll", CharSet = CharSet.Auto)]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, IntPtr lp);

然后在表单初始化器中:

// add the required controls into scroll sync
Control[] syncedCtrls = new Control[]  ctrl1, ctrl2, ..., ctrln ;
foreach (SyncTextBox ctr in syncedCtrls)

    ctr.Buddies = syncedCtrls;

【讨论】:

【参考方案5】:

这是最终帮助我使用鼠标滚轮修复多个文本框同步问题的方法。

我以非常有用的 Hans 示例为基础。

int WM_MOUSEWHEEL   = 0x20a; // or 522
int WM_VSCROLL      = 0x115; // or 277

protected override void WndProc(ref Message m)

        //Trap WM_VSCROLL and WM_MOUSEWHEEL message and pass to buddy
        if (Buddies != null)
        

            if (m.Msg == WM_MOUSEWHEEL)  //mouse wheel 
            

                if ((int)m.WParam < 0)  //mouse wheel scrolls down
                    SendMessage(this.Handle, (int)0x0115, new IntPtr(1), new IntPtr(0)); //WParam: 1- scroll down, 0- scroll up
                else if ((int)m.WParam > 0)
                    SendMessage(this.Handle, (int)0x0115, new IntPtr(0), new IntPtr(0));



                return; //prevent base.WndProc() from messing synchronization up 
            
            else if (m.Msg == WM_VSCROLL)
            
                foreach (Control ctr in Buddies)
                
                    if (ctr != this && !scrolling && ctr != null && ctr.IsHandleCreated)
                    
                        scrolling = true;
                        SendMessage(ctr.Handle, m.Msg, m.WParam, m.LParam);
                        scrolling = false;
                    

                

            
        

    //do the usual
    base.WndProc(ref m);

【讨论】:

以上是关于如何同步两个多行文本框的滚动?的主要内容,如果未能解决你的问题,请参考以下文章

两个多行文本框的乘积并将其显示到另一个多行文本框

在 C# 中滚动到单行文本框的末尾

在多行文本框上设置最大字符(可见性允许)

怎样写CSS设置多行文本框的显示行数?

如何不断滚动到多行文本框中的文本末尾? [复制]

始终在底部滚动文本框[重复]