StudioLaValse.Drawable.Interaction
6.0.1
dotnet add package StudioLaValse.Drawable.Interaction --version 6.0.1
NuGet\Install-Package StudioLaValse.Drawable.Interaction -Version 6.0.1
<PackageReference Include="StudioLaValse.Drawable.Interaction" Version="6.0.1" />
paket add StudioLaValse.Drawable.Interaction --version 6.0.1
#r "nuget: StudioLaValse.Drawable.Interaction, 6.0.1"
// Install StudioLaValse.Drawable.Interaction as a Cake Addin #addin nuget:?package=StudioLaValse.Drawable.Interaction&version=6.0.1 // Install StudioLaValse.Drawable.Interaction as a Cake Tool #tool nuget:?package=StudioLaValse.Drawable.Interaction&version=6.0.1
Drawable
A lightweight front end framework to design scenes and draw them to a generic canvas in C#/.NET.
Description
StudioLaValse.Drawable allows for the simple definition of scenes that can be reused and re-rendered in different environments. It provides a strict separation of models and views without necessarily adhering to the MVVM pattern. The library uses a performant internal visual tree that supports partial re-rendering and offers native support for WPF, Avalonia, WinForms, PDF, Skia, and SVG. Additionally, it allows for dynamic scenes that users can interact with using mouse and keyboard inputs.
Getting Started
Dependencies
- StudioLaValse.Geometry (https://github.com/Studio-La-Valse/Geometry)
Installing
You can install StudioLaValse.Drawable using the NuGet feed or by cloning/forking the repository:
Using Nuget
To install StudioLaValse.Drawable via NuGet, run the following command in your package manager console:
Install-Package StudioLaValse.Drawable
or use the Nuget Package Manager.
Cloning or Forking
Alternatively, you can clone or fork the repository from GitHub:
- Clone the repository:
git clone https://github.com/yourusername/StudioLaValse.Drawable.git
- Navigate to the project directory:
cd StudioLaValse.Drawable
- Install dependencies:
dotnet restore
How to use
- Lets start with a basic scene and render it to html/svg:
using StudioLaValse.Drawable.HTML;
using StudioLaValse.Drawable.Extensions;
public class Program
{
public static void Main(string[] args)
{
var graph = new VisualGraph();
var canvas = new HTMLCanvas(width: 1920, height: 1080);
var canvasPainter = new HTMLCanvasPainter(canvas);
canvasPainter.DrawContentWrapper(graph).FinishDrawing();
var result = canvas.SVGContent();
Console.WriteLine(result);
}
}
public class VisualGraph : BaseContentWrapper
{
public override IEnumerable<BaseDrawableElement> GetDrawableElements()
{
var radius = 50;
var firstPoint = new XY(Random.Shared.Next(300), Random.Shared.Next(300));
var secondPoint = new XY(Random.Shared.Next(300), Random.Shared.Next(300));
// Draw two white circles with a random position and a radius of 50.
yield return new DrawableCircle(firstPoint, radius, ColorARGB.White);
yield return new DrawableCircle(secondPoint, radius, ColorARGB.White);
}
}
Remember that you can natively render to WPF, Avalonia, Skia, Winforms, HTML/SVG, or you can implement to your own canvas adapter. We'll use HTML/SVG strings for sake of simplicity.
- Great, but I want to base my scene on a model that can be serialized/persisted independent of it's visual representation.
using StudioLaValse.Drawable.HTML;
using StudioLaValse.Drawable.Extensions;
public class Program
{
public static void Main(string[] args)
{
// Create dependencies.
var keyGenerator = new IncrementalKeyGenerator();
// Create the models.
var firstNode = new NodeModel(keyGenerator);
var secondNode = new NodeModel(keyGenerator);
// Create the visual representation, the canvas and the canvas painter.
var graph = new VisualGraph(firstNode, secondNode);
var canvas = new HTMLCanvas(width: 1920, height: 1080);
var canvasPainter = new HTMLCanvasPainter(canvas);
canvasPainter.DrawContentWrapper(graph).FinishDrawing();
var result = canvas.SVGContent();
Console.WriteLine(result);
}
}
public class NodeModel
{
public XY Position { get; }
public int Id { get; }
public NodeModel(IKeyGenerator<int> keyGenerator)
{
Id = keyGenerator.Generate();
Position = new XY(Random.Shared.Next(300), Random.Shared.Next(300));
}
}
public class VisualNode : BaseVisualParent<int>
{
private readonly NodeModel node;
private readonly double radius = 50;
public VisualNode(NodeModel node) : base(node.Id)
{
this.node = node;
}
public override IEnumerable<BaseDrawableElement> GetDrawableElements()
{
Debug.Write($"Visual node with id {node.Id} has regenerated it's drawable elements.");
yield return new DrawableCircle(node.Position, radius, ColorARGB.White);
}
}
public class VisualGraph : BaseContentWrapper
{
private readonly NodeModel firstNode;
private readonly NodeModel secondNode;
public VisualGraph(NodeModel firstNode, NodeModel secondNode)
{
this.firstNode = firstNode;
this.secondNode = secondNode;
}
public override IEnumerable<BaseContentWrapper> GetContentWrappers()
{
yield return new VisualNode(firstNode);
yield return new VisualNode(secondNode);
}
}
- Because now the visual representation of the node is coupled to the ID of the node, we can rerender only parts of the scene, for example lets change the position of only the first node. Notice how we don't rerender the second node. We use an observable that emits the ID of the model and some additional information about the render method. If a scene manager is subscribed to the observable, it will render the visual tree branch (and its children) using the given id.
using StudioLaValse.Drawable.HTML;
using StudioLaValse.Drawable.Extensions;
public class Program
{
public static void Main(string[] args)
{
// Create dependencies.
var keyGenerator = new IncrementalKeyGenerator();
var notifyEntityChanged = SceneManager<int>.CreateObservable();
// Create the models.
var firstNode = new NodeModel(keyGenerator, notifyEntityChanged);
var secondNode = new NodeModel(keyGenerator, notifyEntityChanged);
// Create the canvas, canvas painter and scene.
var sceneId = keyGenerator.Generate();
var graph = new VisualGraph(firstNode, secondNode, sceneId);
// Create the canvas and canvas painter.
var canvas = new HTMLCanvas(width: 1920, height: 1080);
var canvasPainter = new HTMLCanvasPainter(canvas);
// Create a scene manager and subscribe to the notify entity changed, to invalidate models to the canvas.
var sceneManager = new SceneManager<int>(graph, canvasPainter)
.WithBackground(ColorARGB.Black)
.WithRerender();
using var subscription = notifyEntityChanged.Subscribe(sceneManager.CreateObserver());
var result = canvas.SVGContent();
Console.WriteLine("--");
Console.WriteLine(result);
Console.WriteLine("--");
// Change your model, and the changes will be reflected on the canvas.
firstNode.Position = new XY(100, 100);
result = canvas.SVGContent();
Console.WriteLine("--");
Console.WriteLine(result);
Console.WriteLine("--");
}
}
public class NodeModel
{
private readonly INotifyEntityChanged<int> notifyEntityChanged;
private XY position;
public XY Position
{
get => position;
set
{
position = value;
// Invalidate this node.
notifyEntityChanged.Invalidate(Id);
// Immediately render changes.
notifyEntityChanged.RenderChanges();
}
}
public int Id { get; }
public NodeModel(IKeyGenerator<int> keyGenerator, INotifyEntityChanged<int> notifyEntityChanged)
{
this.notifyEntityChanged = notifyEntityChanged;
Id = keyGenerator.Generate();
Position = new XY(Random.Shared.Next(300), Random.Shared.Next(300));
}
}
public class VisualNode : BaseVisualParent<int>
{
private readonly NodeModel node;
private readonly double radius = 50;
public VisualNode(NodeModel node) : base(node.Id)
{
this.node = node;
}
public override IEnumerable<BaseDrawableElement> GetDrawableElements()
{
Console.WriteLine("--");
Console.WriteLine($"Visual node with id {node.Id} has regenerated it's drawable elements.");
Console.WriteLine("--");
yield return new DrawableCircle(node.Position, radius, ColorARGB.White);
}
}
public class VisualGraph : BaseVisualParent<int>
{
private readonly NodeModel firstNode;
private readonly NodeModel secondNode;
public VisualGraph(NodeModel firstNode, NodeModel secondNode, int sceneId) : base(sceneId)
{
this.firstNode = firstNode;
this.secondNode = secondNode;
}
public override IEnumerable<BaseContentWrapper> GetContentWrappers()
{
yield return new VisualNode(firstNode);
yield return new VisualNode(secondNode);
}
}
- Excersise: I leave it up to the reader to connect the two nodes with a line, that moves with its parent nodes when they are moved. Bonus points if the connection or the nodes may be added/deleted and allow for mouse/keyboard input. You can find the answer under Example/Scene/VisualGraph.
Why not just use MVVM?
While the MVVM (Model-View-ViewModel) pattern is a well-established approach for building user interfaces, it may not be the ideal solution for every scenario, especially when it comes to complex, dynamic visualizations. Consider for example the effort you have to go through just to connect two ellipses with a line from center to center (try it!), or exporting your components to PDF or SVG for a static webview.
- Separation of Models and Views:
StudioLaValse.Drawable still allows for a strict separation of models and views without strictly adhering to the MVVM pattern. This flexibility enables you to create and manage visual representations of models in a way that best suits your application's requirements.
- Versatile Rendering:
StudioLaValse.Drawable supports multiple rendering targets, including WPF, Avalonia, WinForms, PDF, Skia, and SVG. This versatility allows your scenes to be rendered to different environments without being tied to a specific UI framework.
- Dynamic and Interactive Scenes:
The library is designed to handle dynamic scenes where users can interact with the visual elements using mouse and keyboard inputs. Ofcourse interactivity is supported for MVVM, but most of the time resorting to code behind for custom components is the way to go and at that point you are fully locked into the UI framework - which I personally don't like.
If the above points do not resonate with you, that's fine and it's probaly best to stick with a strict MVVM pattern, especially if you're working with WPF or Avalonia.
Help
Please get in touch if you need help or have any questions. You can also refer to the documentation: https://drawable.lavalse.net/
Authors
License
This project is licensed under the GNU GENERAL PUBLIC LICENSE - see the LICENSE.md file for details
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net8.0
- StudioLaValse.Drawable (>= 6.0.1)
- System.Reactive (>= 6.0.0)
NuGet packages (4)
Showing the top 4 NuGet packages that depend on StudioLaValse.Drawable.Interaction:
Package | Downloads |
---|---|
StudioLaValse.Drawable.WPF
Provides a WPF implementation to the StudioLaValse.Drawable namespace. |
|
StudioLaValse.Drawable.Winforms
Provides a WinForms implementation to the StudioLaValse.Drawable namespace. |
|
StudioLaValse.ScoreDocument.Drawable
Provides extensions to the score document reader so that it may be drawn to a canvas. |
|
StudioLaValse.Drawable.Avalonia
A package to invalidate entities and draw them on any generic bitmap. |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
6.0.1 | 85 | 12/29/2024 |
6.0.0 | 91 | 12/22/2024 |
5.0.0 | 107 | 11/23/2024 |
4.0.2 | 127 | 10/6/2024 |
4.0.0 | 105 | 10/6/2024 |
3.0.0 | 191 | 7/6/2024 |
2.1.1 | 133 | 5/4/2024 |
2.1.0 | 120 | 5/4/2024 |
2.0.4 | 188 | 3/2/2024 |
2.0.3 | 127 | 3/2/2024 |
2.0.2 | 122 | 3/2/2024 |
2.0.1 | 127 | 3/1/2024 |
2.0.0 | 109 | 2/18/2024 |
1.0.0 | 238 | 1/13/2024 |