Подтвердить что ты не робот

Есть ли способ связать несколько конвертеров значений в XAML?

У меня есть ситуация, в которой мне нужно показать целочисленное значение, привязанное к свойству в моем контексте данных, после того, как он выполнил два отдельных преобразования:

  • Обратное значение в пределах диапазона (например, диапазон от 1 до 100, значение в datacontext равно 90, пользователь видит значение 10)
  • преобразует число в строку

Я понимаю, что могу сделать оба действия, создав свой собственный конвертер (который реализует IValueConverter). Однако у меня уже есть отдельный преобразователь значений, который делает только первый шаг, а второй шаг - Int32Converter.

Есть ли способ связать эти два существующих класса в XAML, не создавая дополнительный класс, который их объединяет?

Если мне нужно прояснить это, сообщите мне.:)

Спасибо.

4b9b3361

Ответ 1

Я использовал этот метод от Gareth Evans в моем проекте Silverlight.

Здесь моя реализация:

public class ValueConverterGroup : List<IValueConverter>, IValueConverter
{
    #region IValueConverter Members

    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return this.Aggregate(value, (current, converter) => converter.Convert(current, targetType, parameter, culture));
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        throw new NotImplementedException();
    }

    #endregion
}

Что можно затем использовать в XAML следующим образом:

<c:ValueConverterGroup x:Key="InvertAndVisibilitate">
   <c:BooleanInverterConverter/>
   <c:BooleanToVisibilityConverter/>
</c:ValueConverterGroup>

Ответ 2

Нашел то, что я искал, любезно предоставил Джош Смит: Преобразователи стоимости трубопроводов (ссылка archive.org).

Он определяет класс ValueConverterGroup, использование которого в XAML точно так, как я надеялся. Вот пример:

<!-- Converts the Status attribute text to a SolidColorBrush used to draw 
     the output of statusDisplayNameGroup. -->
<local:ValueConverterGroup x:Key="statusForegroundGroup">
  <local:IntegerStringToProcessingStateConverter  />
  <local:ProcessingStateToColorConverter />
  <local:ColorToSolidColorBrushConverter />
</local:ValueConverterGroup> 

Отличный материал. Спасибо, Джош.:)

Ответ 3

Городская реализация проекта Gareth Evans Silverlight великолепна, однако она не поддерживает различные параметры конвертера.

Я изменил его, чтобы вы могли предоставлять параметры, разделенные запятыми (если, конечно, вы не избежите их).

Преобразователь:

public class ValueConverterGroup : List<IValueConverter>, IValueConverter
{
    private string[] _parameters;

    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        if(parameter != null)
            _parameters = Regex.Split(parameter.ToString(), @"(?<!\\),");

        return (this).Aggregate(value, (current, converter) => converter.Convert(current, targetType, GetParameter(converter), culture));
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        throw new NotImplementedException();
    }

    private string GetParameter(IValueConverter converter)
    {
        if (_parameters == null)
            return null;

        var index = IndexOf(converter as IValueConverter);
        string parameter;

        try
        {
            parameter = _parameters[index];
        }

        catch (IndexOutOfRangeException ex)
        {
            parameter = null;
        }

        if (parameter != null)
            parameter = Regex.Unescape(parameter);

        return parameter;
    }
}

Примечание: ConvertBack здесь не реализован, полную версию см. в моей Gist.

Реализация:

<ContentPage xmlns="http://xamarin.com/schemas/2014/forms" xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:converters="clr-namespace:ATXF.Converters;assembly=ATXF" x:Class="ATXF.TestPage">
  <ResourceDictionary>
    <converters:ValueConverterGroup x:Key="converters">
      <converters:ConverterOne />
      <converters:ConverterTwo />
    </converters:ValueConverterGroup>
  </ResourceDictionary>

  <Label Text="{Binding InitialValue, Converter={StaticResource converters}, ConverterParameter='Parameter1,Parameter2'}" />
</ContentPage>

Ответ 4

Да, есть способы конвертировать конвертеры, но это не выглядит красиво, и вам здесь не нужно. Если вам когда-нибудь понадобится это, спросите себя, действительно ли это путь? Простой всегда работает лучше, даже если вам нужно написать собственный конвертер.

В вашем конкретном случае все, что вам нужно сделать, - это преобразовать преобразованное значение в строку. StringFormat на Binding - ваш друг здесь.

 <TextBlock Text="{Binding Value,Converter={StaticResource myConverter},StringFormat=D}" />

Ответ 5

Вот небольшое расширение Town answer для поддержки мульти-связывания:

public class ValueConverterGroup : List<IValueConverter>, IValueConverter, IMultiValueConverter
{
    #region IValueConverter Members

    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return this.Aggregate(value, (current, converter) => converter.Convert(current, targetType, parameter, culture));
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        throw new NotImplementedException();
    }

    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        return Convert(values as object, targetType, parameter, culture);
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        throw new NotImplementedException();
    }

    #endregion
}