代码之家  ›  专栏  ›  技术社区  ›  Juan Perez

Windows Phone 8.1(WinRT)中可完全滚动的文本框

  •  2
  • Juan Perez  · 技术社区  · 10 年前

    很久以前,我问过一个类似的问题: Scrollable TextBox in WP7 (ala Skype and Facebook) 我希望在Windows Phone 8.1上有同样的行为。

    我有一个文本框,用户可以在其中键入注释,当键盘启动时,它会向上移动文本框,使其始终在视图中。问题是,如果注释太大,用户无法轻松滚动整个注释。

    我不想将TextBox上移,而是想调整页面大小,以便其他元素(如应用程序标题)始终可见。显然,即使注释很大,TextBox也应该很容易滚动。

    这是我的XAML:

    <Page
        x:Class="ScrollableTextBox.MainPage"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="using:ScrollableTextBox"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        mc:Ignorable="d"
        Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
    
    <!--LayoutRoot-->
    <Grid x:Name="LayoutRoot"
          Margin="21,-6.5,19,0">
    
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto"/>
            <RowDefinition Height="*"/>
        </Grid.RowDefinitions>
    
        <!--Title-->
        <TextBlock Margin="0,19,0,24"
                   Style="{ThemeResource TitleTextBlockStyle}"
                   Text="APP TITLE" />
    
        <!--ContentPanel-->
        <Grid Grid.Row="1">
    
            <ScrollViewer x:Name="NoteContentScrollViewer">
    
                <TextBox x:Name="NoteContentTextBox"
                         AcceptsReturn="True"
                         ScrollViewer.VerticalScrollMode="Disabled"
                         VerticalAlignment="Stretch"
                         GotFocus="NoteContentTextBox_GotFocus" />
    
            </ScrollViewer>
    
        </Grid>
    
    </Grid>
    

    下面是代码:

    using Windows.UI.ViewManagement;
    using Windows.UI.Xaml;
    using Windows.UI.Xaml.Controls;
    using Windows.UI.Xaml.Navigation;
    
    namespace ScrollableTextBox
    {
        public sealed partial class MainPage : Page
        {
            // Handle InputPane manually so the UI doesn't scroll when the keyboard appears
            InputPane inputPane = InputPane.GetForCurrentView();
    
            public MainPage()
            {
                this.InitializeComponent();
                this.NavigationCacheMode = NavigationCacheMode.Required;
            }
    
            private void NoteContentTextBox_GotFocus(object sender, RoutedEventArgs e)
            {
                // Subscribe InputPane events to handle UI scrolling
                inputPane.Showing += this.InputPaneShowing;
                inputPane.Hiding += this.InputPaneHiding;
            }
    
            private void InputPaneShowing(InputPane sender, InputPaneVisibilityEventArgs e)
            {
                // Set EnsuredFocusedElementInView to true so the UI doesn't scroll
                e.EnsuredFocusedElementInView = true;
    
                // Set new margins to LayoutRoot (to compensate keyboard)
                LayoutRoot.Margin = new Thickness(21, -6.5, 19, e.OccludedRect.Height);
    
                // Unsubscribe InputPane Showing event
                inputPane.Showing -= this.InputPaneShowing;
            }
    
            private void InputPaneHiding(InputPane sender, InputPaneVisibilityEventArgs e)
            {
                // Set EnsuredFocusedElementInView to false so the UI scrolls
                e.EnsuredFocusedElementInView = false;
    
                // Reset LayoutRoot margins
                LayoutRoot.Margin = new Thickness(21, -6.5, 19, 0);
    
                // Unsubscribe InputPane Hiding event to handle UI scrolling
                inputPane.Hiding -= this.InputPaneHiding;
            }
        }
    }
    

    这很好地工作,因为当键盘启动时,页面会被调整大小,用户可以在编辑注释时轻松滚动,并且其他UI元素不会移出视图。然而,缺少一个行为:当用户点击TextBox时,它应该滚动到插入符号位置,但现在它根本不滚动(正如我们所期望的那样)。

    在WindowsPhone7上,我使用ScrollViewer.ScrollToVerticalOffset()来实现这一点,但它在WinRT上不起作用。我们应该使用ScrollViewer.ChangeView(),但我无法使其工作。

    因此,简而言之,当用户点击TextBox时,我希望它滚动到插入符号位置,这样他就可以立即开始键入,而不必手动滚动(或按Enter键到达该位置)。有什么想法吗?

    3 回复  |  直到 7 年前
        1
  •  1
  •   Bryan Stump    10 年前

    滚动查看器和更改视图不起作用?我在MSDN上问了一篇关于相关内容的帖子,这样我可以稍后链接(我在移动atm上)。 要解释滚动查看器不起作用的原因,需要深入研究依赖项财产(如垂直偏移量)在UI中的优先级。 任何动画都将覆盖代码中设置的值,当您尝试设置新高度时,输入窗格将打开。必须在输入窗格动画完成后调用更改视图。尝试将调度程序计时器设置为半秒间隔,并在滴答事件处理程序中调用更改视图。这将有效地等待UI动画结束,以便正确设置新值。

    您尝试过使用textbox.select方法吗?可能有一个内置的机械师来集中选定的位置。

        2
  •  0
  •   Juan Perez    10 年前

    正如Bryan Stump所建议的那样,我通过使用DispatcherTimer解决了这个问题。以下是缺少的代码:

    // DispatcherTimer to ChangeView() of NoteContentScrollViewer
    DispatcherTimer keyboardTimer = new DispatcherTimer { Interval = TimeSpan.FromMilliseconds(50) };
    

    内部主页():

    // Subscribe keyboardTimer Tick event
    keyboardTimer.Tick += keyboardTimer_Tick;
    

    内部InputPaneShow():

    // Start() keyboardTimer to scroll to caret
    keyboardTimer.Start();
    

    最后,keyboardTimer Tick事件:

    private void keyboardTimer_Tick(object sender, object e)
    {
        // Stop timer so it doesn't repeat
         keyboardTimer.Stop();
    
        // Invoke ChangeView() on NoteContentScrollViewer, and use GetRectFromCharacterIndex to scroll to caret position
        if (NoteContentTextBox.Text != "")
            NoteContentScrollViewer.ChangeView(0, NoteContentTextBox.GetRectFromCharacterIndex(NoteContentTextBox.SelectionStart - 1, true).Y, null);
    }
    

    它们的键是TextBox的GetRectFromCharacterIndex方法,用于定位插入符号的位置。这总是确保插入符号在视图中,至少在我的测试中如此。

        3
  •  0
  •   Zak    10 年前

    XAML:

    <TextBox
        AcceptsReturn="True"
        Name="textBox"
        ScrollViewer.VerticalScrollBarVisibility="Auto"
        ScrollViewer.VerticalScrollMode="Auto"
        TextChanged="TextBox_TextChanged"
     />
    

    C编号:

    private void TextBox_TextChanged(object sender, TextChangedEventArgs e) 
    { 
        var grid = (Grid)VisualTreeHelper.GetChild(textBox, 0); 
        for (var i = 0; i <= VisualTreeHelper.GetChildrenCount(grid) - 1; i++) 
        { 
            object obj = VisualTreeHelper.GetChild(grid, i); 
            if (!(obj is ScrollViewer)) continue; 
            ((ScrollViewer)obj).ChangeView(null, ((ScrollViewer)obj).ExtentHeight, null); 
            break; 
         } 
    }