46

互联网上有很多类似的问题,包括 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

4 回答 4

87

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

您必须聚焦文本框,更新插入符号位置,然后滚动到结尾:

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

编辑

示例文本框:

<TextBox TextWrapping="Wrap" VerticalScrollBarVisibility="Auto" 
         AcceptsReturn="True" Name="textBox"/>
于 2012-01-20T09:12:53.870 回答
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,最好使用绑定而不是在后面的代码中传递文本框。

于 2014-02-13T13:03:50.607 回答
11

如果你不太喜欢后面的代码,这里有一个 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 && 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"

于 2018-08-03T14:28:20.637 回答
2

谢谢!我添加了这个来记住原来的焦点:

var oldFocusedElement = FocusManager.GetFocusedElement(this);

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

FocusManager.SetFocusedElement(this, oldFocusedElement);
于 2013-11-20T14:23:23.770 回答