WPF TabItem Custom ContentTemplate - PullRequest
       29

WPF TabItem Custom ContentTemplate

1 голос
/ 19 марта 2010

Я некоторое время боролся с этим, это было бы просто сделать в WindowForms.

Я делаю IRC-клиент, для каждого канала, к которому будет подключен канал, будет несколько вкладок. На каждой вкладке должен отображаться ряд вещей: UserList, MessageHistory, Topic.

В WindowForms я бы просто унаследовал от TabItem, добавил некоторые пользовательские свойства и элементы управления, и сделал бы.

В WPF у меня возникли небольшие проблемы с разработкой, как это сделать.

Я пробовал много способов сделать это, и ниже мой текущий метод, но я не могу заставить TextBox связываться со свойством Topic.

    <Style TargetType="{x:Type t:IRCTabItem}" BasedOn="{StaticResource {x:Type TabItem}}" >
    <Setter Property="ContentTemplate">
        <Setter.Value>
            <DataTemplate>
                <Grid>
                    <Grid.ColumnDefinitions>
                        <ColumnDefinition Width="540" />
                        <ColumnDefinition />
                    </Grid.ColumnDefinitions>
                    <StackPanel Grid.Column="0">
                        <TextBox Text="{Binding Topic, RelativeSource={RelativeSource AncestorType={x:Type t:IRCTabItem}}}" />   
                    </StackPanel>
                </Grid>
            </DataTemplate>              
        </Setter.Value>
    </Setter>
</Style>

И Код позади

 public class IRCTabItem : TabItem
    {
        static IRCTabItem()
        {
            //This OverrideMetadata call tells the system that this element wants to provide a style that is different than its base class.
            //This style is defined in themes\generic.xaml
            //DefaultStyleKeyProperty.OverrideMetadata(typeof(IRCTabItem),
             //   new FrameworkPropertyMetadata(typeof(IRCTabItem)));
        }

        public static readonly RoutedEvent CloseTabEvent =
            EventManager.RegisterRoutedEvent("CloseTab", RoutingStrategy.Bubble,
                typeof(RoutedEventHandler), typeof(IRCTabItem));

        public event RoutedEventHandler CloseTab
        {
            add { AddHandler(CloseTabEvent, value); }
            remove { RemoveHandler(CloseTabEvent, value); }
        }

        public override void OnApplyTemplate()
        {
            base.OnApplyTemplate();

            Button closeButton = base.GetTemplateChild("PART_Close") as Button;
            if (closeButton != null)
                closeButton.Click += new System.Windows.RoutedEventHandler(closeButton_Click);
        }

        void closeButton_Click(object sender, System.Windows.RoutedEventArgs e)
        {
            this.RaiseEvent(new RoutedEventArgs(CloseTabEvent, this));
        }

        public bool Closeable
        {
            get { return (bool)GetValue(CloseableProperty); }
            set { SetValue(CloseableProperty, value); }
        }
        public static readonly DependencyProperty CloseableProperty = DependencyProperty.Register("Closeable", typeof(bool), typeof(IRCTabItem), new FrameworkPropertyMetadata(true, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault));


        public List<String> UserList
        {
            get { return (List<string>)GetValue(UserListProperty); }
            set { SetValue(UserListProperty, value); }
        }
        public static readonly DependencyProperty UserListProperty = DependencyProperty.Register("UserList", typeof(List<String>), typeof(IRCTabItem), new FrameworkPropertyMetadata(new List<String>(), FrameworkPropertyMetadataOptions.BindsTwoWayByDefault));

        public String Topic
        {
            get { return (string)GetValue(TopicProperty); }
            set { SetValue(TopicProperty, value); }
        }
        public static readonly DependencyProperty TopicProperty = DependencyProperty.Register("Topic", typeof(String), typeof(IRCTabItem), new FrameworkPropertyMetadata("Not Connected", FrameworkPropertyMetadataOptions.BindsTwoWayByDefault));

        public bool HasAlerts
        {
            get { return (bool)GetValue(HasAlertsProperty); }
            set { SetValue(HasAlertsProperty, value); }
        }
        public static readonly DependencyProperty HasAlertsProperty = DependencyProperty.Register("HasAlerts", typeof(bool), typeof(IRCTabItem), new FrameworkPropertyMetadata(false, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault));


    }

Итак, мои вопросы:

Правильно ли я делаю это (лучшие практики)? Если да, то как я могу связать DataTemplate со свойствами? Если нет, то как правильно достичь того, чего я пытаюсь достичь?

Редактировать 1: добавлены предложения Питера Стивенса Редактировать 2: Добавлено видимое редактировать резюме Изменить 3: Теги

Ответы [ 2 ]

0 голосов
/ 19 марта 2010

Я считаю, что в вашем выражении привязки отсутствует режим относительного источника.

Попробуйте это:

<TextBox Text="{Binding Topic, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type t:IRCTabItem}}}" />

Значение по умолчанию для режима равно нулю, поэтому, если вы не включите эту настройку, то, вероятно, не получится так хорошо.

0 голосов
/ 19 марта 2010

Одна из проблем, с которой вы столкнулись, заключается в том, что ваши свойства зависимостей реализованы неправильно.

Тема должна быть реализована так:

public String Topic 
{ 
    get { return (string) GetValue(TopicProperty); }
    set { SetValue(TopicProperty, value); }
}
public static readonly DependencyProperty TopicProperty = 
    DependencyProperty.Register("Topic", typeof(String), 
        typeof(IRCTabItem), new FrameworkPropertyMetadata("Not Connected", 
        FrameworkPropertyMetadataOptions.BindsTwoWayByDefault));

Необходимо рассматривать свойство CLR как синтаксический сахар, окружающий свойство зависимостей WPF. Пусть свойство WPF управляет хранением значения, в противном случае вы получите два свойства с двумя разными значениями.

...