Binding ConverterParameter

165

Apakah ada cara saya bisa melakukan ini di 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>

Saya hanya perlu mengirim orang Tagtua tingkat atas dan Tagkontrol itu sendiri ke kelas konverter saya.

dotNET
sumber

Jawaban:

303

The ConverterParameterproperti tidak bisa terikat karena tidak properti ketergantungan.

Karena Bindingtidak berasal dari DependencyObjecttidak ada sifat-sifatnya dapat menjadi sifat ketergantungan. Akibatnya, Binding tidak akan pernah bisa menjadi objek target dari Binding lain.

Namun ada solusi alternatif. Anda bisa menggunakan MultiBindingdengan converter multi-nilai bukan sebuah Binding yang normal:

<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>

Konverter multi-nilai mendapatkan array nilai sumber sebagai input:

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
sumber
36

Tidak, sayangnya ini tidak akan mungkin karena ConverterParameterbukan DependencyPropertyjadi Anda tidak akan dapat menggunakan binding

Tapi mungkin Anda bisa menipu dan menggunakan MultiBindingwith IMultiValueConverteruntuk meneruskan 2 Tagproperti.

sa_ddam213
sumber
13

Ada juga cara alternatif untuk digunakan MarkupExtensionagar dapat digunakan Bindinguntuk ConverterParameter. Dengan solusi ini Anda masih dapat menggunakan default IValueConverterbukan IMultiValueConverterkarena ConverterParameterdilewatkan ke dalam IValueConverterseperti yang Anda harapkan dalam sampel pertama Anda.

Ini adalah reusable saya 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) };
        }
    }
}

Dengan ini MarkupExtensiondi basis kode Anda, Anda dapat mengikat dengan ConverterParametercara berikut:

<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>

Yang hampir mirip dengan proposal awal Anda.

Pascalsz
sumber
4
Ini bermanfaat. Namun, MultiValueConverterAdapterhilang. Saya menemukannya di sini .
blearyeye