具有自动垂直滚动的多行文本框

时间:2012-01-20 08:48:41

标签: wpf textbox multiline autoscroll

在互联网上有很多类似的问题,包括SO,但建议的解决方案在我的情况下不起作用。 场景:xaml中有一个日志文本框

 <TextBox Name="Status"
          Margin="5"
          Grid.Column="1"
          Grid.Row="5"
          HorizontalAlignment="Left"
          VerticalAlignment="Top"
          Width="600"
          Height="310"/>

代码隐藏中有一些方法可以做一些工作并在这个文本框中添加一些多行(可能就是问题?)消息:

private static void DoSomeThings(TextBox textBox)
{
   // do work
   textBox.AppendText("Work finished\r\n"); // better way than Text += according to msdn
   // do more
   textBox.AppendText("One more message\r\n");
   ...
}

private static void DoSomething2(TextBox textBox)
{
   // same as first method
}

在所有操作发生后,需要滚动到文本框的底部。尝试ScrollToEnd(),ScrollToLine,将文本框包装到ScrollViewer,Selection和Caret解决方法,将ScrollToEnd附加到TextChanged。在溢出文本框高度的执行行仍然需要手动滚动之后,这些都不起作用。对不起重复的问题,我想我错过了一些小问题,可以由对问题有新见解的人迅速解决。提前谢谢。

4 个答案:

答案 0 :(得分:80)

根据这个问题:TextBox.ScrollToEnd doesn't work when the TextBox is in a non-active tab

您必须对焦文本框,更新插入位置,然后滚动到结束:

Status.Focus();
Status.CaretIndex = Status.Text.Length;
Status.ScrollToEnd();

修改

示例TextBox:

<TextBox TextWrapping="Wrap" VerticalScrollBarVisibility="Auto" 
         AcceptsReturn="True" Name="textBox"/>

答案 1 :(得分:17)

如果你把它变成一个简单的自定义控件,那么你不需要任何代码来进行滚动。

public class ScrollingTextBox : TextBox {

    protected override void OnInitialized (EventArgs e) {
        base.OnInitialized(e);
        VerticalScrollBarVisibility = ScrollBarVisibility.Auto;
        HorizontalScrollBarVisibility = ScrollBarVisibility.Auto;
    }

    protected override void OnTextChanged (TextChangedEventArgs e) {
        base.OnTextChanged(e);
        CaretIndex = Text.Length;
        ScrollToEnd();
    }

}

如果你正在使用WPF,那么使用绑定会好得多,而不是在后面的代码中传递文本框。

答案 2 :(得分:3)

如果您不喜欢太多代码,这里有一个AttachedProperty可以解决这个问题:

namespace YourProject.YourAttachedProperties
{

    public class TextBoxAttachedProperties
    {

        public static bool GetAutoScrollToEnd(DependencyObject obj)
        {
            return (bool)obj.GetValue(AutoScrollToEndProperty);
        }

        public static void SetAutoScrollToEnd(DependencyObject obj, bool value)
        {
            obj.SetValue(AutoScrollToEndProperty, value);
        }

        // Using a DependencyProperty as the backing store for AutoScrollToEnd.  This enables animation, styling, binding, etc...
        public static readonly DependencyProperty AutoScrollToEndProperty =
        DependencyProperty.RegisterAttached("AutoScrollToEnd", typeof(bool), typeof(TextBoxAttachedProperties), new PropertyMetadata(false, AutoScrollToEndPropertyChanged));

        private static void AutoScrollToEndPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            if(d is TextBox textbox && e.NewValue is bool mustAutoScroll)
            {
                textbox.TextChanged += (s, ee)=> AutoScrollToEnd(s, ee, textbox);
            }
        }

        private static void AutoScrollToEnd(object sender, TextChangedEventArgs e, TextBox textbox)
        {
            textbox.ScrollToEnd();
        }
    }
}

然后在您的xaml中执行:

<TextBox
    AcceptsReturn="True"
    myAttachedProperties:TextBoxAttachedProperties.AutoScrollToEnd="True"/>

别忘了在xaml文件的顶部添加

xmlns:myAttachedProperties="clr-namespace:YourProject.YourAttachedProperties"

还有,瞧

答案 3 :(得分:2)

谢谢!我已添加此内容以记住原始焦点:

var oldFocusedElement = FocusManager.GetFocusedElement(this);

this.textBox.Focus();
this.textBox.CaretIndex = this.textBox.Text.Length;
this.textBox.ScrollToEnd();

FocusManager.SetFocusedElement(this, oldFocusedElement);