如何在 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。
如你们所知,没有内置类型转换器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。