ImageEditor.axaml.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568
  1. using Avalonia;
  2. using Avalonia.Controls;
  3. using Avalonia.Controls.Shapes;
  4. using Avalonia.Data;
  5. using Avalonia.Input;
  6. using Avalonia.Interactivity;
  7. using Avalonia.Markup.Xaml;
  8. using Avalonia.Media;
  9. using Avalonia.Media.Imaging;
  10. using Avalonia.Skia.Helpers;
  11. using CommunityToolkit.Mvvm.Input;
  12. using FluentResults;
  13. using InABox.Avalonia.Components.ImageEditing;
  14. using InABox.Avalonia.Converters;
  15. using InABox.Core;
  16. using SkiaSharp;
  17. using System.Collections.ObjectModel;
  18. namespace InABox.Avalonia.Components;
  19. public enum ImageEditingMode
  20. {
  21. Polyline,
  22. Rectangle,
  23. Ellipse
  24. }
  25. public class ImageEditorModeButton(ImageEditingMode mode, Canvas? canvas)
  26. {
  27. public ImageEditingMode Mode { get; set; } = mode;
  28. public Canvas? Canvas { get; set; } = canvas;
  29. }
  30. public class ImageEditorTransparentImageBrushConverter : AbstractConverter<IBrush?, IBrush?>
  31. {
  32. public static readonly ImageEditorTransparentImageBrushConverter Instance = new ImageEditorTransparentImageBrushConverter();
  33. protected override IBrush? Convert(IBrush? value, object? parameter = null)
  34. {
  35. if (value is SolidColorBrush solid && solid.Color.A == 255) return solid;
  36. var brush = new VisualBrush
  37. {
  38. TileMode = TileMode.Tile,
  39. DestinationRect = new(0, 0, 10, 10, RelativeUnit.Absolute)
  40. };
  41. var canvas = new Canvas
  42. {
  43. Width = 10,
  44. Height = 10
  45. };
  46. var rect1 = new Rectangle { Width = 5, Height = 5 };
  47. var rect2 = new Rectangle { Width = 5, Height = 5 };
  48. Canvas.SetLeft(rect2, 5);
  49. Canvas.SetTop(rect2, 5);
  50. rect1.Fill = new SolidColorBrush(Colors.LightGray);
  51. rect2.Fill = new SolidColorBrush(Colors.LightGray);
  52. var rect3 = new Rectangle { Width = 10, Height = 10 };
  53. rect3.Fill = value;
  54. canvas.Children.Add(rect1);
  55. canvas.Children.Add(rect2);
  56. canvas.Children.Add(rect3);
  57. brush.Visual = canvas;
  58. return brush;
  59. }
  60. }
  61. public class ImageEditorRemoveOpacityConverter : AbstractConverter<IBrush?, IBrush?>
  62. {
  63. public static readonly ImageEditorRemoveOpacityConverter Instance = new();
  64. protected override IBrush? Convert(IBrush? value, object? parameter = null)
  65. {
  66. if (value is SolidColorBrush solid)
  67. {
  68. return new SolidColorBrush(new Color(255, solid.Color.R, solid.Color.G, solid.Color.B));
  69. }
  70. return value;
  71. }
  72. }
  73. // TODO: Make it so we don't re-render everything everytime 'Objects' changes.
  74. public partial class ImageEditor : UserControl
  75. {
  76. public static readonly StyledProperty<IImage?> SourceProperty =
  77. AvaloniaProperty.Register<ImageEditor, IImage?>(nameof(Source));
  78. public static readonly StyledProperty<IBrush?> PrimaryBrushProperty =
  79. AvaloniaProperty.Register<ImageEditor, IBrush?>(nameof(PrimaryBrush), new SolidColorBrush(Colors.Black));
  80. public static readonly StyledProperty<IBrush?> SecondaryBrushProperty =
  81. AvaloniaProperty.Register<ImageEditor, IBrush?>(nameof(SecondaryBrush), new SolidColorBrush(Colors.White));
  82. public static readonly StyledProperty<double> LineThicknessProperty =
  83. AvaloniaProperty.Register<ImageEditor, double>(nameof(LineThickness), 3.0);
  84. public static readonly StyledProperty<int> ImageWidthProperty =
  85. AvaloniaProperty.Register<ImageEditor, int>(nameof(ImageWidth), 100);
  86. public static readonly StyledProperty<int> ImageHeightProperty =
  87. AvaloniaProperty.Register<ImageEditor, int>(nameof(ImageHeight), 100);
  88. public static readonly StyledProperty<ImageEditingMode> ModeProperty =
  89. AvaloniaProperty.Register<ImageEditor, ImageEditingMode>(nameof(Mode), ImageEditingMode.Polyline);
  90. public static readonly StyledProperty<bool> ShowButtonsProperty =
  91. AvaloniaProperty.Register<ImageEditor, bool>(nameof(ShowButtons), true);
  92. public IImage? Source
  93. {
  94. get => GetValue(SourceProperty);
  95. set => SetValue(SourceProperty, value);
  96. }
  97. public int ImageWidth
  98. {
  99. get => GetValue(ImageWidthProperty);
  100. set => SetValue(ImageWidthProperty, value);
  101. }
  102. public int ImageHeight
  103. {
  104. get => GetValue(ImageHeightProperty);
  105. set => SetValue(ImageHeightProperty, value);
  106. }
  107. public bool ShowButtons
  108. {
  109. get => GetValue(ShowButtonsProperty);
  110. set => SetValue(ShowButtonsProperty, value);
  111. }
  112. #region Editing Properties
  113. public ImageEditingMode Mode
  114. {
  115. get => GetValue(ModeProperty);
  116. set => SetValue(ModeProperty, value);
  117. }
  118. public IBrush? PrimaryBrush
  119. {
  120. get => GetValue(PrimaryBrushProperty);
  121. set => SetValue(PrimaryBrushProperty, value);
  122. }
  123. public IBrush? SecondaryBrush
  124. {
  125. get => GetValue(SecondaryBrushProperty);
  126. set => SetValue(SecondaryBrushProperty, value);
  127. }
  128. public double LineThickness
  129. {
  130. get => GetValue(LineThicknessProperty);
  131. set => SetValue(LineThicknessProperty, value);
  132. }
  133. #endregion
  134. #region Events
  135. public event EventHandler? Changed;
  136. #endregion
  137. #region Private Properties
  138. public ObservableCollection<ImageEditorModeButton> ModeButtons { get; set; } = new();
  139. private ObservableCollection<IImageEditorObject> Objects = new();
  140. private IImageEditorObject? CurrentObject;
  141. private Stack<IImageEditorObject> RedoStack = new();
  142. private double ScaleFactor = 1.0;
  143. #endregion
  144. static ImageEditor()
  145. {
  146. SourceProperty.Changed.AddClassHandler<ImageEditor>(Source_Changed);
  147. }
  148. private static void Source_Changed(ImageEditor editor, AvaloniaPropertyChangedEventArgs args)
  149. {
  150. if(editor.Source is not null)
  151. {
  152. editor.ImageWidth = (int)Math.Floor(editor.Source.Size.Width);
  153. editor.ImageHeight = (int)Math.Floor(editor.Source.Size.Height);
  154. editor.PositionImage();
  155. }
  156. }
  157. public ImageEditor()
  158. {
  159. InitializeComponent();
  160. Objects.CollectionChanged += Objects_CollectionChanged;
  161. AddModeButtons();
  162. SetMode(Mode);
  163. OuterCanvas.LayoutUpdated += OuterCanvas_LayoutUpdated;
  164. }
  165. #region Layout
  166. private void OuterCanvas_LayoutUpdated(object? sender, EventArgs e)
  167. {
  168. PositionImage();
  169. }
  170. protected override void OnLoaded(RoutedEventArgs e)
  171. {
  172. base.OnLoaded(e);
  173. PositionImage();
  174. }
  175. private void PositionImage()
  176. {
  177. var canvasWidth = OuterCanvas.Bounds.Width;
  178. var canvasHeight = OuterCanvas.Bounds.Height;
  179. var scaleFactor = Math.Min(canvasWidth / ImageWidth, canvasHeight / ImageHeight);
  180. ScaleFactor = scaleFactor;
  181. var imageWidth = ImageWidth * scaleFactor;
  182. var imageHeight = ImageHeight * scaleFactor;
  183. ImageBorder.Width = imageWidth;
  184. ImageBorder.Height = imageHeight;
  185. Canvas.SetLeft(ImageBorder, OuterCanvas.Bounds.Width / 2 - imageWidth / 2);
  186. Canvas.SetTop(ImageBorder, OuterCanvas.Bounds.Height / 2 - imageHeight / 2);
  187. Canvas.RenderTransform = new ScaleTransform(ScaleFactor, ScaleFactor);
  188. Canvas.Width = ImageWidth;
  189. Canvas.Height = ImageHeight;
  190. }
  191. #endregion
  192. #region Editing Commands
  193. private void UpdateUndoRedoButtons()
  194. {
  195. UndoButton.IsEnabled = Objects.Count > 0;
  196. RedoButton.IsEnabled = RedoStack.Count > 0;
  197. }
  198. [RelayCommand]
  199. private void Undo()
  200. {
  201. if (Objects.Count == 0) return;
  202. RedoStack.Push(Objects[^1]);
  203. Objects.RemoveAt(Objects.Count - 1);
  204. UpdateUndoRedoButtons();
  205. Changed?.Invoke(this, new EventArgs());
  206. }
  207. [RelayCommand]
  208. private void Redo()
  209. {
  210. if (!RedoStack.TryPop(out var top)) return;
  211. Objects.Add(top);
  212. UpdateUndoRedoButtons();
  213. Changed?.Invoke(this, new EventArgs());
  214. }
  215. private void AddObject(IImageEditorObject obj)
  216. {
  217. Objects.Add(obj);
  218. RedoStack.Clear();
  219. UpdateUndoRedoButtons();
  220. Changed?.Invoke(this, new EventArgs());
  221. }
  222. [RelayCommand]
  223. private void SetMode(ImageEditingMode mode)
  224. {
  225. Mode = mode;
  226. ShapeButton.Content = CreateModeButtonContent(mode);
  227. SecondaryColour.IsVisible = HasSecondaryColour();
  228. }
  229. private bool HasSecondaryColour()
  230. {
  231. return Mode == ImageEditingMode.Rectangle || Mode == ImageEditingMode.Ellipse;
  232. }
  233. #endregion
  234. #region Mode Buttons
  235. private void AddModeButtons()
  236. {
  237. AddModeButton(ImageEditingMode.Polyline);
  238. AddModeButton(ImageEditingMode.Rectangle);
  239. AddModeButton(ImageEditingMode.Ellipse);
  240. }
  241. private void AddModeButton(ImageEditingMode mode)
  242. {
  243. ModeButtons.Add(new(mode, CreateModeButtonContent(mode)));
  244. }
  245. private Canvas? CreateModeButtonContent(ImageEditingMode mode, bool bindColour = false)
  246. {
  247. switch (mode)
  248. {
  249. case ImageEditingMode.Polyline:
  250. var canvas = new Canvas();
  251. var points = new Point[] { new(0, 0), new(20, 8), new(5, 16), new(25, 25) };
  252. var line1 = new Polyline { Points = points, Width = 25, Height = 25 };
  253. var line2 = new Polyline { Points = points, Width = 25, Height = 25 };
  254. line1.StrokeThickness = 4;
  255. line1.StrokeLineCap = PenLineCap.Round;
  256. line1.StrokeJoin = PenLineJoin.Round;
  257. line1.Stroke = new SolidColorBrush(Colors.Black);
  258. canvas.Children.Add(line1);
  259. if (bindColour)
  260. {
  261. line1.StrokeThickness = 5;
  262. line2.StrokeThickness = 4;
  263. line2.StrokeLineCap = PenLineCap.Round;
  264. line2.StrokeJoin = PenLineJoin.Round;
  265. line2.Bind(Polyline.StrokeProperty, new Binding(nameof(PrimaryBrush))
  266. {
  267. Source = this,
  268. Converter = ImageEditorRemoveOpacityConverter.Instance
  269. });
  270. canvas.Children.Add(line2);
  271. }
  272. return canvas;
  273. case ImageEditingMode.Rectangle:
  274. canvas = new Canvas();
  275. canvas.Width = 25;
  276. canvas.Height = 25;
  277. var rectangle = new Rectangle();
  278. if (bindColour)
  279. {
  280. rectangle.Bind(Rectangle.StrokeProperty, new Binding(nameof(PrimaryBrush))
  281. {
  282. Source = this,
  283. Converter = ImageEditorRemoveOpacityConverter.Instance
  284. });
  285. rectangle.Bind(Rectangle.FillProperty, new Binding(nameof(SecondaryBrush))
  286. {
  287. Source = this,
  288. Converter = ImageEditorTransparentImageBrushConverter.Instance
  289. });
  290. }
  291. else
  292. {
  293. rectangle.Stroke = new SolidColorBrush(Colors.Black);
  294. rectangle.Fill = new SolidColorBrush(Colors.White);
  295. }
  296. rectangle.StrokeThickness = 1.0;
  297. rectangle.Width = 25;
  298. rectangle.Height = 25;
  299. canvas.Children.Add(rectangle);
  300. return canvas;
  301. case ImageEditingMode.Ellipse:
  302. canvas = new Canvas();
  303. canvas.Width = 25;
  304. canvas.Height = 25;
  305. var ellipse = new Ellipse();
  306. if (bindColour)
  307. {
  308. ellipse.Bind(Rectangle.StrokeProperty, new Binding(nameof(PrimaryBrush))
  309. {
  310. Source = this,
  311. Converter = ImageEditorRemoveOpacityConverter.Instance
  312. });
  313. ellipse.Bind(Rectangle.FillProperty, new Binding(nameof(SecondaryBrush))
  314. {
  315. Source = this,
  316. Converter = ImageEditorTransparentImageBrushConverter.Instance
  317. });
  318. }
  319. else
  320. {
  321. ellipse.Stroke = new SolidColorBrush(Colors.Black);
  322. ellipse.Fill = new SolidColorBrush(Colors.White);
  323. }
  324. ellipse.StrokeThickness = 1.0;
  325. ellipse.Width = 25;
  326. ellipse.Height = 25;
  327. canvas.Children.Add(ellipse);
  328. return canvas;
  329. default:
  330. return null;
  331. }
  332. }
  333. #endregion
  334. #region Public Interface
  335. public void Reset()
  336. {
  337. Objects.Clear();
  338. RedoStack.Clear();
  339. UpdateUndoRedoButtons();
  340. Changed?.Invoke(this, new EventArgs());
  341. }
  342. public Bitmap GetImage()
  343. {
  344. var renderBitmap = new RenderTargetBitmap(new PixelSize(ImageWidth, ImageHeight));
  345. renderBitmap.Render(Image);
  346. using var context = renderBitmap.CreateDrawingContext();
  347. if(Source is not null)
  348. {
  349. context.DrawImage(Source, new(0, 0, ImageWidth, ImageHeight));
  350. }
  351. foreach (var obj in Objects)
  352. {
  353. var control = obj.GetControl();
  354. var left = Canvas.GetLeft(control);
  355. var top = Canvas.GetTop(control);
  356. if (double.IsNaN(left)) left = 0;
  357. if (double.IsNaN(top)) top = 0;
  358. using (context.PushTransform(Matrix.CreateTranslation(new(left, top))))
  359. {
  360. control.Render(context);
  361. }
  362. }
  363. return renderBitmap;
  364. }
  365. public byte[] SaveImage()
  366. {
  367. var bitmap = GetImage();
  368. var stream = new MemoryStream();
  369. bitmap.Save(stream);
  370. return stream.ToArray();
  371. }
  372. #endregion
  373. #region Editing
  374. private void Objects_CollectionChanged(object? sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
  375. {
  376. Canvas.Children.Clear();
  377. foreach(var item in Objects)
  378. {
  379. item.Update();
  380. Canvas.Children.Add(item.GetControl());
  381. }
  382. }
  383. Point ConvertToImageCoordinates(Point canvasCoordinates)
  384. {
  385. return canvasCoordinates;// new(canvasCoordinates.X / ScaleFactor, canvasCoordinates.Y / ScaleFactor);
  386. }
  387. private void Canvas_PointerPressed(object? sender, PointerPressedEventArgs e)
  388. {
  389. var position = ConvertToImageCoordinates(e.GetPosition(Canvas));
  390. switch (Mode)
  391. {
  392. case ImageEditingMode.Polyline:
  393. CurrentObject = new PolylineObject
  394. {
  395. Points = [position],
  396. PrimaryBrush = PrimaryBrush,
  397. Thickness = LineThickness
  398. };
  399. AddObject(CurrentObject);
  400. break;
  401. case ImageEditingMode.Rectangle:
  402. CurrentObject = new RectangleObject
  403. {
  404. Point1 = position,
  405. Point2 = position,
  406. PrimaryBrush = PrimaryBrush,
  407. SecondaryBrush = SecondaryBrush,
  408. Thickness = LineThickness
  409. };
  410. AddObject(CurrentObject);
  411. break;
  412. case ImageEditingMode.Ellipse:
  413. CurrentObject = new EllipseObject
  414. {
  415. Point1 = position,
  416. Point2 = position,
  417. PrimaryBrush = PrimaryBrush,
  418. SecondaryBrush = SecondaryBrush,
  419. Thickness = LineThickness
  420. };
  421. AddObject(CurrentObject);
  422. break;
  423. }
  424. }
  425. private void Canvas_PointerMoved(object? sender, PointerEventArgs e)
  426. {
  427. var position = ConvertToImageCoordinates(e.GetPosition(Canvas));
  428. switch (CurrentObject)
  429. {
  430. case PolylineObject polyline:
  431. polyline.Points.Add(position);
  432. polyline.Update();
  433. Changed?.Invoke(this, new EventArgs());
  434. break;
  435. case RectangleObject rectangle:
  436. rectangle.Point2 = position;
  437. rectangle.Update();
  438. Changed?.Invoke(this, new EventArgs());
  439. break;
  440. case EllipseObject ellipse:
  441. ellipse.Point2 = position;
  442. ellipse.Update();
  443. Changed?.Invoke(this, new EventArgs());
  444. break;
  445. }
  446. }
  447. private void Canvas_PointerReleased(object? sender, PointerReleasedEventArgs e)
  448. {
  449. var position = ConvertToImageCoordinates(e.GetPosition(Canvas));
  450. switch (CurrentObject)
  451. {
  452. case PolylineObject polyline:
  453. polyline.Points.Add(position);
  454. polyline.Update();
  455. CurrentObject = null;
  456. Changed?.Invoke(this, new EventArgs());
  457. break;
  458. case RectangleObject rectangle:
  459. rectangle.Point2 = position;
  460. rectangle.Update();
  461. CurrentObject = null;
  462. Changed?.Invoke(this, new EventArgs());
  463. break;
  464. case EllipseObject ellipse:
  465. ellipse.Point2 = position;
  466. ellipse.Update();
  467. CurrentObject = null;
  468. Changed?.Invoke(this, new EventArgs());
  469. break;
  470. }
  471. }
  472. #endregion
  473. }