4

DataGrid我使用CellTemplateand CellEditingTemplate。在这两个 DataTemplates FrameworkElement.IsLoaded 属性False中,即使我可以看到TextBlock,使用TextBoxFocus()调用也返回了True

这是一个错误吗?或者有人可以解释一下,这种行为的原因是什么?


我创建了这个示例应用程序用于演示目的。

主窗口.xaml.cs

namespace WpfApplication
{
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            this.DataContext = new List<string> { "Row1", "Row2" };
        }
    }

    public class FocusAttached
    {
        public static bool GetIsFocused(DependencyObject obj)
        {
            return (bool)obj.GetValue(IsFocusedProperty);
        }

        public static void SetIsFocused(DependencyObject obj, bool value)
        {
            obj.SetValue(IsFocusedProperty, value);
        }

        public static readonly DependencyProperty IsFocusedProperty =
             DependencyProperty.RegisterAttached("IsFocused", typeof(bool), typeof(MainWindow), new UIPropertyMetadata(false, IsFocusedChanged));

        static void IsFocusedChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e)
        {
            FrameworkElement element = obj as FrameworkElement;

            if ((bool)e.NewValue)
            {
                Console.Write(element);
                Console.Write("  IsLoaded=" + element.IsLoaded);
                Console.Write("  IsVisible=" + element.IsVisible);
                Console.Write("  Focusable=" + element.Focusable);
                // here I call Focus()
                Console.Write("  Focus() returns:" + element.Focus());
                Console.WriteLine("  IsLoaded=" + element.IsLoaded);
            }
        }
    }
}

主窗口.xaml

<Window x:Class="WpfApplication.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:c="clr-namespace:WpfApplication"
        Title="Please click on row!" SizeToContent="WidthAndHeight">
    <DataGrid ItemsSource="{Binding}" AutoGenerateColumns="False">
        <DataGrid.Columns>
            <DataGridTemplateColumn>
                <DataGridTemplateColumn.CellTemplate>
                    <DataTemplate>
                        <TextBlock Text="{Binding IsLoaded, RelativeSource={RelativeSource Self}, Mode=OneWay,
                                                  StringFormat='TextBlock in CellTemplate: IsLoaded={0}'}" />
                    </DataTemplate>
                </DataGridTemplateColumn.CellTemplate>

                <DataGridTemplateColumn.CellEditingTemplate>
                    <DataTemplate>
                        <TextBox c:FocusAttached.IsFocused="True"
                                 Text="{Binding IsLoaded, RelativeSource={RelativeSource Self}, Mode=OneWay,
                                                StringFormat='Even after call Focus(): IsLoaded={0}'}" />                                     
                    </DataTemplate>
                </DataGridTemplateColumn.CellEditingTemplate>
            </DataGridTemplateColumn>
        </DataGrid.Columns>

        <DataGrid.CellStyle>
            <Style TargetType="DataGridCell">
                <Setter Property="Focusable" Value="False" />
                <Style.Triggers>
                    <Trigger Property="IsSelected" Value="True">
                        <Setter Property="IsEditing" Value="True" />
                    </Trigger>
                </Style.Triggers>
            </Style>            
        </DataGrid.CellStyle>
    </DataGrid>
</Window>
4

1 回答 1

1

首先,您的绑定是无用的,因为 IsLoaded 不是依赖属性。没有通知,文本没有变化。

IsLoaded 是假的,因为它像测量和排列一样被延迟。该元素是可聚焦的、可见的和启用的,因此它可以被聚焦。但不能保证此时元素已经测量和渲染。这些操作在 Dispatcher 中排队。当它们被处理时, IsLoaded 将为真。试试这个:

static void IsFocusedChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e)
{
    FrameworkElement element = obj as FrameworkElement;

    if ((bool)e.NewValue)
    {
        Console.Write(element);
        Console.Write("  IsLoaded=" + element.IsLoaded);
        Console.Write("  IsVisible=" + element.IsVisible);
        Console.Write("  Focusable=" + element.Focusable);
        // here I call Focus() 
        Console.Write("  Focus() returns:" + element.Focus());
        element.Dispatcher.BeginInvoke((Action)(() =>
            {
                Console.WriteLine("  IsLoaded=" + element.IsLoaded);
            }),
            System.Windows.Threading.DispatcherPriority.Loaded);
    }
}
于 2012-04-16T09:40:14.047 回答