Как мне привязать ListBox в IronPython? - PullRequest
4 голосов
/ 04 июня 2011

Я только начинаю использовать IronPython с WPF, и я не совсем понимаю, как должно выполняться связывание.

Обычно в WPF я бы делал что-то вроде этого:

<ListBox Name="MyListBox">
    <ListBox.Resources>
        <Style TargetType="ListBoxItem">
            <Setter Property="ContentTemplate">
                <Setter.Value>
                    <DataTemplate>
                        <DockPanel>
                            <TextBlock Text="{Binding Path=From}" />
                            <TextBlock Text="{Binding Path=Subject}" />
                        </DockPanel>
                     </DataTemplate>
                </Setter.Value>
            </Setter>
        </Style>
    </ListBox.Resources>
</ListBox>

Тогда в моем коде позади:

MyListBox.ItemsSource = new ObservableCollection<Email>()

Но в IronPython у нас не может быть ObservableCollection объектов, только типов. Это не работает:

MyListBox.ItemsSource = new ObservableCollection[email]()

Когда выдается исключение: "ожидаемый массив [тип], полученный класс"

Что я должен делать? Помогите пожалуйста!

Ответы [ 3 ]

4 голосов
/ 04 июня 2011

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

Во-первых, вам нужен pyevent.py из каталога tutorial / в вашем каталоге IronPython.

Во-вторых, нам нужен вспомогательный класс:

class NotifyPropertyChangedBase(INotifyPropertyChanged):
    """INotifyProperty Helper"""
    PropertyChanged = None
    def __init__(self):
        (self.PropertyChanged, self._propertyChangedCaller) = make_event()

    def add_PropertyChanged(self, value):
        self.PropertyChanged += value

    def remove_PropertyChanged(self, value):
        self.PropertyChanged -= value

    def OnPropertyChanged(self, propertyName):
        self._propertyChangedCaller(self, PropertyChangedEventArgs(propertyName))

Затем вам нужно объявить свой класс данных следующим образом:

class Email(NotifyPropertyChangedBase):
    """
        use setter getter.
        IronPython 2.6 or later.
    """
    @property
    def From(self):
        return self._From

    @From.setter
    def From(self, value):
        self._From = value
        self.OnPropertyChanged("From")

    @property
    def Subject(self):
        return self._Subject

    @Subject.setter
    def Subject(self, value):
        self._Subject = value
        self.OnPropertyChanged("Subject")

Наконец, установите ItemSource ListBox:

self.data = ObservableCollection[Email]()
self.MyListBox.ItemsSource = self.data

Кредит этой ссылке за помощь: http://palepoli.skr.jp/wp/2009/06/28/wpf-listview-databinding-for-ironpython/

3 голосов
/ 12 января 2012

Расширяя ответ Бодена, вы можете немного улучшить NotifyPropertyChangedBase:

class NotifyPropertyChangedBase(INotifyPropertyChanged):
    PropertyChanged = None
    def __init__(self):
        self.PropertyChanged, self._propertyChangedCaller = pyevent.make_event()

    def add_PropertyChanged(self, value):
        self.PropertyChanged += value

    def remove_PropertyChanged(self, value):
        self.PropertyChanged -= value

    def OnPropertyChanged(self, propertyName):
        if self.PropertyChanged is not None:
            self._propertyChangedCaller(self, PropertyChangedEventArgs(propertyName))

    def init_view(self, view):
        xaml = view
        self.view = XamlLoader(xaml).Root
        self.view.DataContext = self

    def declareNotifiable(self, *symbols):
        for symbol in symbols:
            self.defineNotifiableProperty(symbol)

    def defineNotifiableProperty(self, symbol):
        dnp = """
import sys
sys.path.append(__file__)
from NotifyProperty import *
@notify_property
def {0}(self):
    return self._{0}   

@{0}.setter
def {0}(self, value):
    self._{0} = value
""".format(symbol)
        d = globals()
        exec dnp.strip() in d
        setattr(self.__class__, symbol, d[symbol])
        exec("self.{0} = ''".format(symbol))

С этим набором вы можете сделать что-то вроде:

class Email(NotifyPropertyChangedBase):
    def __init__(self):
        self.defineNotifiableProperty("From", "Subject")

С этим на местевы получите элементы @notify_property и @ property.setter, установленные для всего в вызове defineNotifiableProperty.

2 голосов
/ 04 июня 2011

IronPython чувствителен к регистру и не использует ключевое слово new.Попробуйте:

MyListBox.ItemsSource = ObservableCollection[Email]()
...