6
     <data:DataGridTemplateColumn Header="Name">
        <data:DataGridTemplateColumn.CellTemplate>
            <DataTemplate>
                <TextBlock Text="{Binding Name}">
            </DataTemplate>
        </data:DataGridTemplateColumn.CellTemplate> 
        <data:DataGridTemplateColumn.CellEditingTemplate>
            <DataTemplate>
                <TextBox Text="{Binding Name}">
            </DataTemplate>
        </data:DataGridTemplateColumn.CellEditingTemplate> 
    </data:DataGridTemplateColumn>              

テンプレート列のわかりやすい例ですね。それの何が問題なのですか?ユーザーが TAB キーを押して DataGrid をナビゲートする場合、TextBox のテキストを編集できるようにするには、TAB を 2 回 (!) 押す必要があります。ユーザーが列フォーカスを取得したらすぐに編集可能にするにはどうすればよいでしょうか。

Ok。私は方法を見つけました - Grid.KeyUp() に以下のコードを入れました:

 if (Grid.CurrentColumn.Header.ToString() == "UserName")
        {
            if (e.Key != Key.Escape) 
            {
                Grid.BeginEdit();

                // Simply send another TAB press
                if (Keyboard.FocusedElement is Microsoft.Windows.Controls.DataGridCell)
                {
                    var keyEvt = new KeyEventArgs(Keyboard.PrimaryDevice, Keyboard.PrimaryDevice.ActiveSource, 0, Key.Tab) { RoutedEvent = Keyboard.KeyDownEvent };
                    InputManager.Current.ProcessInput(keyEvt);
                }
            }
        } 
4

4 に答える 4

8

直面した問題は、DataGridTemplateColumn内のコントロール(TextBoxなど)がDataGridCell内に含まれていることです。デフォルトでは、DataGridCellにはタブストップ機能があります。したがって、TextBoxコントロールにフォーカスを合わせるためにTabキーを2回押す必要がある理由。解決策は、DataGridCellのタブストップ機能を無効にすることです。これは、DataGridCellのスタイルを介して実行できます。

解決策は次のとおりです。

<Style TargetType="{x:Type DataGridCell}">
     <Setter Property="KeyboardNavigation.IsTabStop" Value="False" />
</Style>
于 2012-08-05T15:26:02.187 に答える
8

あなたの問題は、各セルが最初にフォーカスを受け取るコンテンツコントロールにエディターを配置し、次にもう一度エディターにタブを付ける必要があるという事実に起因しています。GenerateEditingElement メソッドの DataGridTemplateColumn のコードを見ると、これを行うメソッド LoadTemplateContent が呼び出されます。

private FrameworkElement LoadTemplateContent(bool isEditing, object dataItem, DataGridCell cell)
{
    DataTemplate template = ChooseCellTemplate(isEditing);
    DataTemplateSelector templateSelector = ChooseCellTemplateSelector(isEditing);
    if (template != null || templateSelector != null)
    {
        ContentPresenter contentPresenter = new ContentPresenter();
        BindingOperations.SetBinding(contentPresenter, ContentPresenter.ContentProperty, new Binding());
        contentPresenter.ContentTemplate = template;
        contentPresenter.ContentTemplateSelector = templateSelector;
        return contentPresenter;
    }

    return null;
}

テンプレートを配置するための新しいコンテンツ プレゼンターがどのように作成されるかを確認してください。他の人がこの問題にさまざまな方法で対処してきました。(そのため、余分な要素を作成したり、コンテンツ プレゼンターをフォーカスを受け取らないように設定したりしません) このでは、フォーカス マネージャーを使用して同じ問題に対処しています (このコードはテストしていません) 。

<tk:DataGridTemplateColumn.CellEditingTemplate>
   <DataTemplate>
      <Grid FocusManager.FocusedElement="{Binding ElementName=txt1}">
         <TextBox Name="txt1" Text="{Binding XPath=@ISBN}" 
                  BorderThickness="0" GotFocus="TextBox_GotFocus"/>
      </Grid>
   </DataTemplate>
</tk:DataGridTemplateColumn.CellEditingTemplate>

エディターとしてユーザー コントロールがある場合は、フォーカス マネージャーでパターンを使用するか、OnLoaded イベントのイベント ハンドラーを使用できます。

于 2010-01-07T03:42:19.290 に答える
3

これが私のアプローチです。@Nalin Jayasuriyaの回答に非常に近いですが、スタイルを作成したくありませんでした。また、このソリューションは TextBox 内のテキストを選択します。とにかく、ホール DataGrid の XAML は次のようになります。

<DataGrid Name="TextBlockDataGrid" ItemsSource="{Binding Path=Rows}" Style="{StaticResource DefaultSettingsDataGrid}">
<DataGrid.Columns>
    <DataGridTextColumn Binding="{Binding Text}" IsReadOnly="True"/>
    <DataGridTemplateColumn Width="*">
        <DataGridTemplateColumn.CellStyle>
            <Style TargetType="{x:Type DataGridCell}">
                <Setter Property="KeyboardNavigation.IsTabStop" Value="False"/>
            </Style>
        </DataGridTemplateColumn.CellStyle>
        <DataGridTemplateColumn.CellTemplate>
            <DataTemplate>
                <Border BorderThickness="{Binding ErrorBorderThickness}" BorderBrush="{Binding ErrorBorderBrush}">
                    <TextBox Text="{Binding UserText, Mode=TwoWay, UpdateSourceTrigger=LostFocus}"
                             HorizontalAlignment="Right"
                             GotKeyboardFocus="TextBox_GotKeyboardFocus"
                             PreviewMouseDown="TextBox_PreviewMouseDown"
                             Style="{StaticResource DefaultTextBox}"/>
                </Border>
            </DataTemplate>
        </DataGridTemplateColumn.CellTemplate>
    </DataGridTemplateColumn>
</DataGrid.Columns>

そしてコードビハインド。

private void TextBox_GotKeyboardFocus(object sender, KeyboardFocusChangedEventArgs e)
{
    try
    {
        ((TextBox)sender).SelectAll();
    }
    catch (Exception ex) { GlobalDebug.debugForm.WriteText(ex); }
}

private void TextBox_PreviewMouseDown(object sender, MouseButtonEventArgs e)
{
    try
    {
        // If its a triple click, select all text for the user.
        if (e.ClickCount == 3)
        {
            ((TextBox)sender).SelectAll();
            return;
        }

        // Find the TextBox
        DependencyObject parent = e.OriginalSource as UIElement;
        while (parent != null && !(parent is TextBox))
        {
            parent = System.Windows.Media.VisualTreeHelper.GetParent(parent);
        }

        if (parent != null)
        {
            if (parent is TextBox)
            {
                var textBox = (TextBox)parent;
                if (!textBox.IsKeyboardFocusWithin)
                {
                    // If the text box is not yet focussed, give it the focus and
                    // stop further processing of this click event.
                    textBox.Focus();
                    e.Handled = true;
                }
            }
        }
    }
    catch (Exception ex) { GlobalDebug.debugForm.WriteText(ex); }
}

詳細については、私のブログをご覧ください: http://blog.baltz.dk/post/2014/11/28/WPF-DataGrid-set-focus-and-mark-text

于 2014-11-28T17:32:03.160 に答える
0

私のアプローチは、ロード時に目的のテンプレート要素にフォーカスを設定する TriggerAction を使用することです。

トリガーは非常に単純です。

public class TakeFocusAndSelectTextOnVisibleBehavior : TriggerAction<TextBox>
{
    protected override void Invoke(object parameter)
    {
        Dispatcher.BeginInvoke(
            DispatcherPriority.Loaded,
            new Action(() =>
            {
                AssociatedObject.Focus();
                AssociatedObject.SelectAll();
            }));
    }
}

DataTemplate は次のようになります。

<DataTemplate>
    <TextBox Text="{Binding Path=Price, Mode=TwoWay}"
                MinHeight="0"
                Padding="1,0"
                Height="20">
        <Interactivity:Interaction.Triggers>
            <Interactivity:EventTrigger EventName="Loaded">
                <Behaviors:TakeFocusAndSelectTextOnVisibleBehavior />
            </Interactivity:EventTrigger>
        </Interactivity:Interaction.Triggers>
    </TextBox>
</DataTemplate>

他の要素タイプの他のトリガーを作成できます。

于 2011-11-11T16:46:38.030 に答える