如何在 Xamarin.Forms 中制作可绑定标记扩展

How to make a bindable markup extension in Xamarin.Forms

如你们所知,没有内置类型转换器class可以将字符串转换为 ImageSource 对象。

当我想让我的模型有一个字符串 属性 表示嵌入图像的 ResouceID,然后 XAML 的 Image.Source 属性绑定到属性 通过数据绑定。

最简单的解决方案是在模型中添加另一个 属性 ImageSource 类型,它只获取基于字符串 ResourceID 属性生成的图像源对象,然后将其绑定到 Image.Source。

它运行良好,但问题是我必须向定义模型的项目添加 Xamarin.Forms 依赖项,该模型是 .NET Stardard class 库。

我按项目拆分视图和 model/viewmodel,以便模型可以在其他应用程序中重复使用。因此,该模型被设计为只有基本类型,如 int、string 等。 最简单的解决方案是让模型 class 依赖于 Xamarin.Forms,我担心以后使用其他框架时会不会带来一些问题。

为了解决问题,我尝试在共享项目中通过引用 MS 文档声明具有绑定功能的标记扩展 class,但不起作用。

扩展class代码如下


// This class is added to the shared project, where a page class exsists.
public class BindableImageSourceExtension : BindableObject, IMarkupExtension
{
    // This is a unique property of BindableImageSource element
    public string ResourceID
    {
        get { return (string)GetValue(ResouceIDProperty); }
        set { SetValue(ResouceIDProperty, value); }
    }

    public object ProvideValue(IServiceProvider serviceProvider)
    {
        if(ResourceID == null) return null;

        // Images are embeded to a project named BusinessSession that holds Data class.
        return ImageSource.FromResource(ResourceID, typeof(BusinessSession).GetTypeInfo().Assembly); 
    }

    public static readonly BindableProperty ResouceIDProperty =
            BindableProperty.Create("ResourceID", typeof(string),
                                    typeof(BindableImageSourceExtension),
                                    default(string));
}

有人可以帮忙吗?

不需要创建bindableproperty,解决方法在document中列出:

Because there is no built-in type converter from string to ResourceImageSource, these types of images cannot be natively loaded by XAML. Instead, a simple custom XAML markup extension can be written to load images using a Resource ID specified in XAML:

[ContentProperty (nameof(Source))]
public class ImageResourceExtension : IMarkupExtension
{
 public string Source { get; set; }

 public object ProvideValue (IServiceProvider serviceProvider)
 {
   if (Source == null)
   {
     return null;
   }

   // Do your translation lookup here, using whatever method you require
   var imageSource = ImageSource.FromResource(Source, typeof(ImageResourceExtension).GetTypeInfo().Assembly);

   return imageSource;
 }
}

并且在 Xaml 中:

 <StackLayout VerticalOptions="Center" HorizontalOptions="Center">
   <!-- use a custom Markup Extension -->
   <Image Source="{local:ImageResource WorkingWithImages.beach.jpg}" />
 </StackLayout>

您也可以创建自己的 ValueConverter:

public class stringToImageSourceConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        return ImageSource.FromResource(value as string, typeof(MainPage).GetTypeInfo().Assembly);
    }

    public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    {
        return value;
    }
}

并且在 Xaml 中:

<Image Source="{Binding Path, Converter={StaticResource stringToImage}}" VerticalOptions="CenterAndExpand" HorizontalOptions="CenterAndExpand" />

我上传了一个示例项目here