Binding ConverterParameter

165

Czy jest sposób, w jaki mógłbym to zrobić w Style:

<Style TargetType="FrameworkElement">
    <Setter Property="Visibility">
        <Setter.Value>
            <Binding Path="Tag"
                RelativeSource="{RelativeSource AncestorType=UserControl}"
                Converter="{StaticResource AccessLevelToVisibilityConverter}"
                ConverterParameter="{Binding RelativeSource={RelativeSource Mode=Self}, Path=Tag}" />                        
        </Setter.Value>
    </Setter>
</Style>

Po prostu muszę wysłać Tagrodzica najwyższego poziomu i Tagsamą kontrolę do mojej klasy konwertera.

dotNET
źródło

Odpowiedzi:

303

ConverterParameterNieruchomość nie może być związany, ponieważ nie jest właściwość zależność.

Ponieważ Bindingnie pochodzi z DependencyObjectżadnej z jego właściwości, nie może być właściwościami zależności. W konsekwencji powiązanie nigdy nie może być obiektem docelowym innego powiązania.

Istnieje jednak alternatywne rozwiązanie. Możesz użyć MultiBindingz konwerterem wielu wartości zamiast normalnego Binding:

<Style TargetType="FrameworkElement">
    <Setter Property="Visibility">
        <Setter.Value>
            <MultiBinding Converter="{StaticResource AccessLevelToVisibilityConverter}">
                <Binding Path="Tag" RelativeSource="{RelativeSource Mode=FindAncestor,
                                                     AncestorType=UserControl}"/>
                <Binding Path="Tag" RelativeSource="{RelativeSource Mode=Self}"/>
            </MultiBinding>
        </Setter.Value>
    </Setter>
</Style>

Konwerter wielu wartości pobiera tablicę wartości źródłowych jako dane wejściowe:

public class AccessLevelToVisibilityConverter : IMultiValueConverter
{
    public object Convert(
        object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        return values.All(v => (v is bool && (bool)v))
            ? Visibility.Visible
            : Visibility.Hidden;
    }

    public object[] ConvertBack(
        object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        throw new NotSupportedException();
    }
}
Clemens
źródło
36

Nie, niestety nie będzie to możliwe, ponieważ ConverterParameternie jest DependencyPropertyto możliwe, więc nie będziesz mógł używać powiązań

Ale być może mógłbyś oszukać i użyć MultiBindingz, IMultiValueConverteraby przekazać 2 Tagwłaściwości.

sa_ddam213
źródło
13

Istnieje również alternatywny sposób MarkupExtensionkorzystania Bindingz pliku ConverterParameter. Dzięki temu rozwiązaniu nadal możesz używać wartości domyślnej IValueConverterzamiast metody, IMultiValueConverterponieważ ConverterParameterjest ona przekazywana do tego, IValueConverterczego oczekiwałeś w pierwszej próbce.

Oto mój wielokrotnego użytku MarkupExtension:

/// <summary>
///     <example>
///         <TextBox>
///             <TextBox.Text>
///                 <wpfAdditions:ConverterBindableParameter Binding="{Binding FirstName}"
///                     Converter="{StaticResource TestValueConverter}"
///                     ConverterParameterBinding="{Binding ConcatSign}" />
///             </TextBox.Text>
///         </TextBox>
///     </example>
/// </summary>
[ContentProperty(nameof(Binding))]
public class ConverterBindableParameter : MarkupExtension
{
    #region Public Properties

    public Binding Binding { get; set; }
    public BindingMode Mode { get; set; }
    public IValueConverter Converter { get; set; }
    public Binding ConverterParameter { get; set; }

    #endregion

    public ConverterBindableParameter()
    { }

    public ConverterBindableParameter(string path)
    {
        Binding = new Binding(path);
    }

    public ConverterBindableParameter(Binding binding)
    {
        Binding = binding;
    }

    #region Overridden Methods

    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        var multiBinding = new MultiBinding();
        Binding.Mode = Mode;
        multiBinding.Bindings.Add(Binding);
        if (ConverterParameter != null)
        {
            ConverterParameter.Mode = BindingMode.OneWay;
            multiBinding.Bindings.Add(ConverterParameter);
        }
        var adapter = new MultiValueConverterAdapter
        {
            Converter = Converter
        };
        multiBinding.Converter = adapter;
        return multiBinding.ProvideValue(serviceProvider);
    }

    #endregion

    [ContentProperty(nameof(Converter))]
    private class MultiValueConverterAdapter : IMultiValueConverter
    {
        public IValueConverter Converter { get; set; }

        private object lastParameter;

        public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
        {
            if (Converter == null) return values[0]; // Required for VS design-time
            if (values.Length > 1) lastParameter = values[1];
            return Converter.Convert(values[0], targetType, lastParameter, culture);
        }

        public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
        {
            if (Converter == null) return new object[] { value }; // Required for VS design-time

            return new object[] { Converter.ConvertBack(value, targetTypes[0], lastParameter, culture) };
        }
    }
}

Z tym MarkupExtensionw swojej bazie kodu możesz po prostu powiązać ConverterParameterw następujący sposób:

<Style TargetType="FrameworkElement">
<Setter Property="Visibility">
    <Setter.Value>
     <wpfAdditions:ConverterBindableParameter Binding="{Binding Tag, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type UserControl}"
                 Converter="{StaticResource AccessLevelToVisibilityConverter}"
                 ConverterParameterBinding="{Binding RelativeSource={RelativeSource Mode=Self}, Path=Tag}" />          
    </Setter.Value>
</Setter>

Co wygląda prawie jak twoja początkowa propozycja.

Pascalsz
źródło
4
To jest przydatne. Jednak MultiValueConverterAdapterbrakuje. Znalazłem to tutaj .
blearyeye