Модифицировать изображение программно с помощью Caliburn.Micro
Я хочу применить эффект / фильтр к изображению, отображаемому на моем экране, в памяти (без сохранения файла изображения).
Я использую Caliburn.Micro в качестве основы MVVM.
У меня есть это мнение:
<UserControl x:Class="TestApplication.Views.MainView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
mc:Ignorable="d"
d:DesignHeight="300"
d:DesignWidth="300">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
<ColumnDefinition Width="0.3*" />
</Grid.ColumnDefinitions>
<Image
Grid.Column="0"
Stretch="Uniform"
Source="{Binding PreviewUrl}"
Margin="5"
x:Name="Preview" />
<ContentControl
Grid.Column="1"
Margin="5"
x:Name="ImageManagement" />
</Grid>
</UserControl>
И эта точка зрения модели:
namespace TestApplication.ViewModels
{
using System;
using System.ComponentModel.Composition;
using System.Linq;
using Caliburn.Micro;
using ImageProcessor;
using ImageProcessor.Plugins.Popart;
using PropertyChanged;
using TestApplication.Events;
using TestApplication.Models;
[ImplementPropertyChanged]
[Export(typeof(MainViewModel))]
public class MainViewModel : PropertyChangedBase, IHandle<FileSelectedEvent>, IHandle<FilterChangedEvent>
{
private readonly IEventAggregator events;
private bool hasCustomImage = false;
[ImportingConstructor]
public MainViewModel(IEventAggregator events)
{
this.events = events;
this.events.Subscribe(this);
this.ImageManagement = new ImageManagementViewModel(events);
this.ImageManagement.SettingsEnabled = false;
this.PreviewUrl = "pack://application:,,,/Resources/placeholder.jpg";
}
public ImageManagementViewModel ImageManagement { get; set; }
public String PreviewUrl { get; set; }
public void Handle(FilterChangedEvent message)
{
this.ApplyFilter(new Filter(message));
}
public void Handle(FileSelectedEvent message)
{
this.PreviewUrl = message.FilePath;
this.hasCustomImage = true;
this.ImageManagement.SettingsEnabled = true;
this.ApplyFilter(Filter.Default);
}
private void ApplyFilter(Filter filter)
{
if (this.hasCustomImage)
{
using (ImageFactory factory = new ImageFactory())
{
factory.Load(this.PreviewUrl);
// TODO: apply a filter using factory
// TODO: use this processed image
}
}
}
}
}
В методе ApplyFilter мне нужно привязать обработанный System.Drawing.Image
к источнику изображения представления, но я абсолютно не знаю, как это сделать (я еще не знаком со всеми аспектами wpf).
1 ответ
Решение
Используя комментарии @Charleh и другие посты, я смог найти, как это должно работать:
<Image
Grid.Column="0"
Stretch="Uniform"
Margin="5"
Source="{Binding Preview}" />
public class MainViewModel : PropertyChangedBase, IHandle<FileSelectedEvent>, IHandle<WarholFilterChangedEvent>
{
private readonly IEventAggregator events;
private bool hasCustomImage = false;
private string sourceUrl;
[ImportingConstructor]
public MainViewModel(IEventAggregator events)
{
this.events = events;
this.events.Subscribe(this);
this.ImageManagement = new ImageManagementViewModel(events);
this.ImageManagement.WarholSettingsEnabled = false;
this.Preview = new BitmapImage(new Uri("pack://application:,,,/Resources/placeholder.jpg"));
}
public ImageManagementViewModel ImageManagement { get; set; }
public BitmapImage Preview { get; set; }
public void Handle(WarholFilterChangedEvent message)
{
this.ApplyFilter(new WarholFilter(message));
}
public void Handle(FileSelectedEvent message)
{
this.sourceUrl = message.FilePath;
this.hasCustomImage = true;
this.ImageManagement.WarholSettingsEnabled = true;
this.ApplyFilter(WarholFilter.Default);
}
private void ApplyFilter(WarholFilter filter)
{
if (this.hasCustomImage)
{
using (ImageFactory factory = new ImageFactory())
{
factory.Load(this.sourceUrl);
PolychromaticParameters parameters = new PolychromaticParameters();
parameters.Number = filter.ColorsNumber;
parameters.Colors = filter.Colors.Select(c => System.Drawing.Color.FromArgb(c.R, c.G, c.B)).ToArray();
parameters.Thresholds = filter.Thresholds.ToArray();
factory.Polychromatic(parameters);
BitmapImage img = new BitmapImage();
using (MemoryStream str = new MemoryStream())
{
img.BeginInit();
img.CacheOption = BitmapCacheOption.OnLoad;
factory.Save(str);
str.Seek(0, SeekOrigin.Begin);
img.StreamSource = str;
img.EndInit();
}
this.Preview = img;
}
}
}
}