Используя XAML для привязки к System.Drawing.Image в элемент управления System.Windows.Image

Я привязываю ListView к списку объектов, например:

<ListView 
    ItemsSource="{ Binding Path=. }"
    ItemTemplate="{DynamicResource EventTemplate}">   

Я привязываюсь к объекту, который объявляет два свойства;

string DisplayName { get; }
System.Drawing.Image Image { get; set; }

Я хочу заполнить DataTemplate, но я не могу понять, как; если я сделаю это в своем шаблоне;

<StackPanel Orientation="Horizontal">
    <Image Source="{ Binding Path=Image }" />
    <TextBlock Text="{ Binding Path=DisplayName }" />
</StackPanel>      

Текст появляется, но изображения нет. Что я делаю не так? Отладочный вывод показывает

System.Windows.Data Error: 1 : Cannot create default converter
to perform 'one-way' conversions between types
'System.Drawing.Image' and 'System.Windows.Media.ImageSource'.
Consider using Converter property of Binding.
BindingExpression:Path=Image; DataItem='RealElement'
(HashCode=54308798); target element is 'Image' (Name='');
target property is 'Source' (type 'ImageSource')

Ответ 1

Нашел способ, которым я доволен. Используя указатель Рида Копси и этот учебник Я завернул кода как IValueConverter.

Здесь преобразователь от System.Drawing.Image до System.Windows.Media.ImageSource;

using System;
using System.Drawing.Imaging;
using System.Globalization;
using System.IO;
using System.Windows.Data;

namespace System.Windows.Media
{
    /// <summary>
    /// One-way converter from System.Drawing.Image to System.Windows.Media.ImageSource
    /// </summary>
    [ValueConversion(typeof(System.Drawing.Image), typeof(System.Windows.Media.ImageSource))]
    public class ImageConverter : IValueConverter
    {
        public object Convert(object value, Type targetType,
            object parameter, CultureInfo culture)
        {
            // empty images are empty...
            if (value == null) { return null; }

            var image = (System.Drawing.Image)value;
            // Winforms Image we want to get the WPF Image from...
            var bitmap = new System.Windows.Media.Imaging.BitmapImage();
            bitmap.BeginInit();
            MemoryStream memoryStream = new MemoryStream();
            // Save to a memory stream...
            image.Save(memoryStream, ImageFormat.Bmp);
            // Rewind the stream...
            memoryStream.Seek(0, System.IO.SeekOrigin.Begin);
            bitmap.StreamSource = memoryStream;
            bitmap.EndInit();
            return bitmap;
        }

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

Затем вам нужно привести конвертер изображений в XAML в качестве ресурса;

xmlns:med="clr-namespace:System.Windows.Media"
...

<ListView.Resources>
    <med:ImageConverter x:Key="imageConverter" />
</ListView.Resources>

Затем вы можете использовать его в XAML для непосредственного привязки к изображению, используя новый конвертер;

<Image Source="{ Binding Path=Image, Converter={StaticResource imageConverter} }" />

Ответ 2

Вы не можете напрямую связать это - вам нужно иметь какой-то конвертер, который будет конвертировать из изображения GDI в изображение WPF.

Вот один подход - он использует поток памяти для извлечения данных из изображения GDI и создания объекта BitmapSource.