WPF TemplateBinding против RelativeSource TemplatedParent

169

В чем разница между этими двумя привязками:

<ControlTemplate TargetType="{x:Type Button}">
   <Border BorderBrush="{TemplateBinding Property=Background}">
      <ContentPresenter />
   </Border>
</ControlTemplate>

и

<ControlTemplate TargetType="{x:Type Button}">
   <Border BorderBrush="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Background}">
      <ContentPresenter />
   </Border>
</ControlTemplate>

?

PaN1C_Showt1Me
источник
17
если вам нужна TwoWay Binding, вам нужно использовать второй вариант
Joachim Kerschbaumer

Ответы:

207

TemplateBinding не совсем то же самое. Документы MSDN часто пишутся людьми, которым приходится тестировать односложные SDE о функциях программного обеспечения, поэтому нюансы не совсем верны.

Привязки TemplateBindings оцениваются во время компиляции по типу, указанному в шаблоне элемента управления. Это позволяет гораздо быстрее создавать скомпилированные шаблоны. Просто возьмите имя в шаблонной привязке, и вы увидите, что компилятор пометит его.

Обязательная разметка разрешается во время выполнения. При более медленном выполнении привязка разрешит имена свойств, которые не видны в типе, объявленном шаблоном. Медленнее, я укажу, что это своего рода относительный, так как операция привязки занимает очень мало процессора процессора. Если вы загружаете шаблоны управления с высокой скоростью, вы можете заметить это.

На практике используйте TemplateBinding, когда можете, но не бойтесь Binding.

Грант БлахаЭрат
источник
18
Поэтому главное помнить: время компиляции против времени выполнения. TemplateBinding не будет работать, если вы попытаетесь изменить его во время выполнения. Правильно ?
PaN1C_Showt1Me
3
Также обратите внимание, что использование Binding вместо TemplateBinding может повлиять на то, что вы видите во время разработки. В некоторых конфигурациях свойства, связанные с помощью {Binding RelativeSource ...}, не отображаются в конструкторе (хотя они все еще отображаются во время выполнения), но если вы переключитесь на использование {TemplateBinding ...}, эти свойства будут оценены во время разработки.
Ифалин
Одна вещь, которую я добавлю в случае, если это поможет будущим посетителям, заключается в том, что поскольку TemplateBinding оценивается во время компиляции, вы не можете использовать TemplateBinding для привязки к определенному пользователем вложенному свойству. В случае определенных пользователем вложенных свойств вы должны использовать «{Binding RelativeSource = {RelativeSource TemplatedParent} ...}»
MNB
35

TemplateBinding - больше ограничений, чем при использовании обычного Binding

  • Более эффективен, чем Binding, но у него меньше функциональности
  • Работает только внутри визуального дерева ControlTemplate
  • Не работает со свойствами на Freezables
  • Не работает в триггере ControlTemplate
  • Предоставляет ярлык в настройке свойств (не такой подробный), например, {TemplateBinding targetProperty}

регулярное связывание - не имеет вышеуказанных ограничений TemplateBinding

  • Уважает родительские свойства
  • Сбрасывает целевые значения, чтобы очистить любые явно установленные значения
  • Пример: <Ellipse Fill = "{Binding RelativeSource = {RelativeSource TemplatedParent}, Path = Background}" />
Пол Фишер
источник
22

Еще одна вещь - TemplateBindings не позволяет конвертировать значения. Они не позволяют вам передавать Converter и, например, автоматически не конвертируют int в строку (что является нормальным для Binding).

Мирослав Недялков
источник
1
Благодаря Мирославу, с этой проблемой я столкнулся, переход на использование TemplatedParent решил проблему.
MikeKulls
17

TemplateBinding является сокращением для Binding с TemplatedParent, но он не предоставляет все возможности класса Binding, например, вы не можете управлять Binding.Mode из TemplateBinding.

Nir
источник
1

Я думал, что TemplateBinding не поддерживает Freezable типы (которые включают объекты кисти). Чтобы обойти проблему. Можно использовать TemplatedParent

Yaz
источник