การผูก ConverterParameter


165

มีวิธีที่ฉันสามารถทำได้ใน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>

ฉันเพียงแค่ต้องส่งTagผู้ปกครองระดับบนสุดและTagตัวควบคุมไปยังคลาสตัวแปลงของฉัน

คำตอบ:


303

ConverterParameterคุณสมบัติไม่สามารถผูกไว้เพราะมันเป็นไม่ได้ทรัพย์สินพึ่งพา

เนื่องจากBindingไม่ได้มาจากDependencyObjectไม่มีคุณสมบัติใดที่สามารถเป็นคุณสมบัติการพึ่งพาได้ ดังนั้นการผูกไม่สามารถเป็นวัตถุเป้าหมายของการผูกอื่น

อย่างไรก็ตามยังมีทางเลือกอื่น คุณสามารถใช้ a MultiBindingพร้อมตัวแปลงหลายค่าแทนการเชื่อมปกติ:

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

ตัวแปลงค่าหลายค่าได้รับอาร์เรย์ของค่าแหล่งข้อมูลเป็นอินพุต:

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();
    }
}

36

ไม่โชคไม่ดีที่มันเป็นไปไม่ได้เพราะConverterParameterไม่ใช่DependencyPropertyคุณจะไม่สามารถใช้การผูกได้

แต่บางทีคุณอาจจะโกงและใช้MultiBindingกับIMultiValueConverterการผ่านใน 2 Tagคุณสมบัติ


13

นอกจากนี้ยังมีทางเลือกในการใช้งานMarkupExtensionเพื่อการใช้งานสำหรับBinding ConverterParameterด้วยโซลูชันนี้คุณยังคงสามารถใช้ค่าเริ่มต้นIValueConverterแทนIMultiValueConverterเพราะConverterParameterจะถูกส่งผ่านไปในIValueConverterแบบที่คุณต้องการในตัวอย่างแรกของคุณ

นี่คือนำมาใช้ใหม่ของฉัน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) };
        }
    }
}

ด้วยสิ่งนี้MarkupExtensionในฐานรหัสของคุณคุณสามารถผูกConverterParameterวิธีต่อไปนี้:

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

ซึ่งดูเหมือนกับข้อเสนอเริ่มต้นของคุณ


4
สิ่งนี้มีประโยชน์ อย่างไรก็ตามMultiValueConverterAdapterหายไป ผมพบว่ามันนี่
blearyeye
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.