csharpc-sharpdotnetxamlavaloniauicross-platformcross-platform-xamlavaloniaguimulti-platformuser-interfacedotnetcore
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
515 lines
17 KiB
515 lines
17 KiB
// Copyright (c) The Avalonia Project. All rights reserved.
|
|
// Licensed under the MIT license. See licence.md file in the project root for full license information.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using Avalonia.Media;
|
|
using Avalonia.Media.Immutable;
|
|
using Avalonia.Platform;
|
|
using Avalonia.Rendering.SceneGraph;
|
|
using Avalonia.Threading;
|
|
using Avalonia.Utilities;
|
|
using Avalonia.VisualTree;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace Avalonia.Rendering
|
|
{
|
|
/// <summary>
|
|
/// A renderer which renders the state of the visual tree to an intermediate scene graph
|
|
/// representation which is then rendered on a rendering thread.
|
|
/// </summary>
|
|
public class DeferredRenderer : RendererBase, IRenderer, IRenderLoopTask, IVisualBrushRenderer
|
|
{
|
|
private readonly IDispatcher _dispatcher;
|
|
private readonly IRenderLoop _renderLoop;
|
|
private readonly IVisual _root;
|
|
private readonly ISceneBuilder _sceneBuilder;
|
|
|
|
private bool _running;
|
|
private volatile IRef<Scene> _scene;
|
|
private DirtyVisuals _dirty;
|
|
private IRef<IRenderTargetBitmapImpl> _overlay;
|
|
private object _rendering = new object();
|
|
private int _lastSceneId = -1;
|
|
private DisplayDirtyRects _dirtyRectsDisplay = new DisplayDirtyRects();
|
|
private IRef<IDrawOperation> _currentDraw;
|
|
private readonly IDeferredRendererLock _lock;
|
|
private readonly object _sceneLock = new object();
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DeferredRenderer"/> class.
|
|
/// </summary>
|
|
/// <param name="root">The control to render.</param>
|
|
/// <param name="renderLoop">The render loop.</param>
|
|
/// <param name="sceneBuilder">The scene builder to use. Optional.</param>
|
|
/// <param name="dispatcher">The dispatcher to use. Optional.</param>
|
|
/// <param name="rendererLock">Lock object used before trying to access render target</param>
|
|
public DeferredRenderer(
|
|
IRenderRoot root,
|
|
IRenderLoop renderLoop,
|
|
ISceneBuilder sceneBuilder = null,
|
|
IDispatcher dispatcher = null,
|
|
IDeferredRendererLock rendererLock = null)
|
|
{
|
|
Contract.Requires<ArgumentNullException>(root != null);
|
|
|
|
_dispatcher = dispatcher ?? Dispatcher.UIThread;
|
|
_root = root;
|
|
_sceneBuilder = sceneBuilder ?? new SceneBuilder();
|
|
Layers = new RenderLayers();
|
|
_renderLoop = renderLoop;
|
|
_lock = rendererLock ?? new ManagedDeferredRendererLock();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DeferredRenderer"/> class.
|
|
/// </summary>
|
|
/// <param name="root">The control to render.</param>
|
|
/// <param name="renderTarget">The render target.</param>
|
|
/// <param name="sceneBuilder">The scene builder to use. Optional.</param>
|
|
/// <remarks>
|
|
/// This constructor is intended to be used for unit testing.
|
|
/// </remarks>
|
|
public DeferredRenderer(
|
|
IVisual root,
|
|
IRenderTarget renderTarget,
|
|
ISceneBuilder sceneBuilder = null)
|
|
{
|
|
Contract.Requires<ArgumentNullException>(root != null);
|
|
Contract.Requires<ArgumentNullException>(renderTarget != null);
|
|
|
|
_root = root;
|
|
RenderTarget = renderTarget;
|
|
_sceneBuilder = sceneBuilder ?? new SceneBuilder();
|
|
Layers = new RenderLayers();
|
|
_lock = new ManagedDeferredRendererLock();
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public bool DrawFps { get; set; }
|
|
|
|
/// <inheritdoc/>
|
|
public bool DrawDirtyRects { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets a path to which rendered frame should be rendered for debugging.
|
|
/// </summary>
|
|
public string DebugFramesPath { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets the render layers.
|
|
/// </summary>
|
|
internal RenderLayers Layers { get; }
|
|
|
|
/// <summary>
|
|
/// Gets the current render target.
|
|
/// </summary>
|
|
internal IRenderTarget RenderTarget { get; private set; }
|
|
|
|
/// <inheritdoc/>
|
|
public void AddDirty(IVisual visual)
|
|
{
|
|
_dirty?.Add(visual);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Disposes of the renderer and detaches from the render loop.
|
|
/// </summary>
|
|
public void Dispose()
|
|
{
|
|
lock (_sceneLock)
|
|
{
|
|
var scene = _scene;
|
|
_scene = null;
|
|
scene?.Dispose();
|
|
}
|
|
|
|
Stop();
|
|
|
|
Layers.Clear();
|
|
RenderTarget?.Dispose();
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public IEnumerable<IVisual> HitTest(Point p, IVisual root, Func<IVisual, bool> filter)
|
|
{
|
|
if (_renderLoop == null && (_dirty == null || _dirty.Count > 0))
|
|
{
|
|
// When unit testing the renderLoop may be null, so update the scene manually.
|
|
UpdateScene();
|
|
}
|
|
//It's safe to access _scene here without a lock since
|
|
//it's only changed from UI thread which we are currently on
|
|
return _scene?.Item.HitTest(p, root, filter) ?? Enumerable.Empty<IVisual>();
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void Paint(Rect rect)
|
|
{
|
|
var t = (IRenderLoopTask)this;
|
|
if(t.NeedsUpdate)
|
|
UpdateScene();
|
|
Render(true);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void Resized(Size size)
|
|
{
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void Start()
|
|
{
|
|
if (!_running && _renderLoop != null)
|
|
{
|
|
_renderLoop.Add(this);
|
|
_running = true;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void Stop()
|
|
{
|
|
if (_running && _renderLoop != null)
|
|
{
|
|
_renderLoop.Remove(this);
|
|
_running = false;
|
|
}
|
|
}
|
|
|
|
bool NeedsUpdate => _dirty == null || _dirty.Count > 0;
|
|
bool IRenderLoopTask.NeedsUpdate => NeedsUpdate;
|
|
|
|
void IRenderLoopTask.Update(TimeSpan time) => UpdateScene();
|
|
|
|
void IRenderLoopTask.Render() => Render(false);
|
|
|
|
/// <inheritdoc/>
|
|
Size IVisualBrushRenderer.GetRenderTargetSize(IVisualBrush brush)
|
|
{
|
|
return (_currentDraw.Item as BrushDrawOperation)?.ChildScenes?[brush.Visual]?.Size ?? Size.Empty;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
void IVisualBrushRenderer.RenderVisualBrush(IDrawingContextImpl context, IVisualBrush brush)
|
|
{
|
|
var childScene = (_currentDraw.Item as BrushDrawOperation)?.ChildScenes?[brush.Visual];
|
|
|
|
if (childScene != null)
|
|
{
|
|
Render(context, (VisualNode)childScene.Root, null, new Rect(childScene.Size));
|
|
}
|
|
}
|
|
|
|
internal void UnitTestUpdateScene() => UpdateScene();
|
|
|
|
internal void UnitTestRender() => Render(false);
|
|
|
|
private void Render(bool forceComposite)
|
|
{
|
|
using (var l = _lock.TryLock())
|
|
{
|
|
if (l == null)
|
|
return;
|
|
|
|
IDrawingContextImpl context = null;
|
|
try
|
|
{
|
|
try
|
|
{
|
|
IDrawingContextImpl GetContext()
|
|
{
|
|
if (context != null)
|
|
return context;
|
|
if (RenderTarget == null)
|
|
RenderTarget = ((IRenderRoot)_root).CreateRenderTarget();
|
|
return context = RenderTarget.CreateDrawingContext(this);
|
|
|
|
}
|
|
|
|
var (scene, updated) = UpdateRenderLayersAndConsumeSceneIfNeeded(GetContext);
|
|
using (scene)
|
|
{
|
|
var overlay = DrawDirtyRects || DrawFps;
|
|
if (DrawDirtyRects)
|
|
_dirtyRectsDisplay.Tick();
|
|
if (overlay)
|
|
RenderOverlay(scene.Item, GetContext());
|
|
if (updated || forceComposite || overlay)
|
|
RenderComposite(scene.Item, GetContext());
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
context?.Dispose();
|
|
}
|
|
}
|
|
catch (RenderTargetCorruptedException ex)
|
|
{
|
|
Logging.Logger.Information("Renderer", this, "Render target was corrupted. Exception: {0}", ex);
|
|
RenderTarget?.Dispose();
|
|
RenderTarget = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
private (IRef<Scene> scene, bool updated) UpdateRenderLayersAndConsumeSceneIfNeeded(Func<IDrawingContextImpl> contextFactory,
|
|
bool recursiveCall = false)
|
|
{
|
|
IRef<Scene> sceneRef;
|
|
lock (_sceneLock)
|
|
sceneRef = _scene?.Clone();
|
|
if (sceneRef == null)
|
|
return (null, false);
|
|
using (sceneRef)
|
|
{
|
|
var scene = sceneRef.Item;
|
|
if (scene.Generation != _lastSceneId)
|
|
{
|
|
var context = contextFactory();
|
|
Layers.Update(scene, context);
|
|
|
|
RenderToLayers(scene);
|
|
|
|
if (DebugFramesPath != null)
|
|
{
|
|
SaveDebugFrames(scene.Generation);
|
|
}
|
|
|
|
lock (_sceneLock)
|
|
_lastSceneId = scene.Generation;
|
|
|
|
|
|
// We have consumed the previously available scene, but there might be some dirty
|
|
// rects since the last update. *If* we are on UI thread, we can force immediate scene
|
|
// rebuild before rendering anything on-screen
|
|
// We are calling the same method recursively here
|
|
if (!recursiveCall && Dispatcher.UIThread.CheckAccess() && NeedsUpdate)
|
|
{
|
|
UpdateScene();
|
|
var (rs, _) = UpdateRenderLayersAndConsumeSceneIfNeeded(contextFactory, true);
|
|
return (rs, true);
|
|
}
|
|
|
|
// Indicate that we have updated the layers
|
|
return (sceneRef.Clone(), true);
|
|
}
|
|
|
|
// Just return scene, layers weren't updated
|
|
return (sceneRef.Clone(), false);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
private void Render(IDrawingContextImpl context, VisualNode node, IVisual layer, Rect clipBounds)
|
|
{
|
|
if (layer == null || node.LayerRoot == layer)
|
|
{
|
|
clipBounds = node.ClipBounds.Intersect(clipBounds);
|
|
|
|
if (!clipBounds.IsEmpty && node.Opacity > 0)
|
|
{
|
|
var isLayerRoot = node.Visual == layer;
|
|
|
|
node.BeginRender(context, isLayerRoot);
|
|
|
|
foreach (var operation in node.DrawOperations)
|
|
{
|
|
_currentDraw = operation;
|
|
operation.Item.Render(context);
|
|
_currentDraw = null;
|
|
}
|
|
|
|
foreach (var child in node.Children)
|
|
{
|
|
Render(context, (VisualNode)child, layer, clipBounds);
|
|
}
|
|
|
|
node.EndRender(context, isLayerRoot);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void RenderToLayers(Scene scene)
|
|
{
|
|
if (scene.Layers.HasDirty)
|
|
{
|
|
foreach (var layer in scene.Layers)
|
|
{
|
|
var renderTarget = Layers[layer.LayerRoot].Bitmap;
|
|
var node = (VisualNode)scene.FindNode(layer.LayerRoot);
|
|
|
|
if (node != null)
|
|
{
|
|
using (var context = renderTarget.Item.CreateDrawingContext(this))
|
|
{
|
|
foreach (var rect in layer.Dirty)
|
|
{
|
|
context.Transform = Matrix.Identity;
|
|
context.PushClip(rect);
|
|
context.Clear(Colors.Transparent);
|
|
Render(context, node, layer.LayerRoot, rect);
|
|
context.PopClip();
|
|
|
|
if (DrawDirtyRects)
|
|
{
|
|
_dirtyRectsDisplay.Add(rect);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private void RenderOverlay(Scene scene, IDrawingContextImpl parentContent)
|
|
{
|
|
if (DrawDirtyRects)
|
|
{
|
|
var overlay = GetOverlay(parentContent, scene.Size, scene.Scaling);
|
|
|
|
using (var context = overlay.Item.CreateDrawingContext(this))
|
|
{
|
|
context.Clear(Colors.Transparent);
|
|
RenderDirtyRects(context);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_overlay?.Dispose();
|
|
_overlay = null;
|
|
}
|
|
}
|
|
|
|
private void RenderDirtyRects(IDrawingContextImpl context)
|
|
{
|
|
foreach (var r in _dirtyRectsDisplay)
|
|
{
|
|
var brush = new ImmutableSolidColorBrush(Colors.Magenta, r.Opacity);
|
|
context.FillRectangle(brush, r.Rect);
|
|
}
|
|
}
|
|
|
|
private void RenderComposite(Scene scene, IDrawingContextImpl context)
|
|
{
|
|
context.Clear(Colors.Transparent);
|
|
|
|
var clientRect = new Rect(scene.Size);
|
|
|
|
foreach (var layer in scene.Layers)
|
|
{
|
|
var bitmap = Layers[layer.LayerRoot].Bitmap;
|
|
var sourceRect = new Rect(0, 0, bitmap.Item.PixelSize.Width, bitmap.Item.PixelSize.Height);
|
|
|
|
if (layer.GeometryClip != null)
|
|
{
|
|
context.PushGeometryClip(layer.GeometryClip);
|
|
}
|
|
|
|
if (layer.OpacityMask == null)
|
|
{
|
|
context.DrawImage(bitmap, layer.Opacity, sourceRect, clientRect);
|
|
}
|
|
else
|
|
{
|
|
context.DrawImage(bitmap, layer.OpacityMask, layer.OpacityMaskRect, sourceRect);
|
|
}
|
|
|
|
if (layer.GeometryClip != null)
|
|
{
|
|
context.PopGeometryClip();
|
|
}
|
|
}
|
|
|
|
if (_overlay != null)
|
|
{
|
|
var sourceRect = new Rect(0, 0, _overlay.Item.PixelSize.Width, _overlay.Item.PixelSize.Height);
|
|
context.DrawImage(_overlay, 0.5, sourceRect, clientRect);
|
|
}
|
|
|
|
if (DrawFps)
|
|
{
|
|
RenderFps(context, clientRect, scene.Layers.Count);
|
|
}
|
|
}
|
|
|
|
private void UpdateScene()
|
|
{
|
|
Dispatcher.UIThread.VerifyAccess();
|
|
lock (_sceneLock)
|
|
{
|
|
if (_scene?.Item.Generation > _lastSceneId)
|
|
return;
|
|
}
|
|
if (_root.IsVisible)
|
|
{
|
|
var sceneRef = RefCountable.Create(_scene?.Item.CloneScene() ?? new Scene(_root));
|
|
var scene = sceneRef.Item;
|
|
|
|
if (_dirty == null)
|
|
{
|
|
_dirty = new DirtyVisuals();
|
|
_sceneBuilder.UpdateAll(scene);
|
|
}
|
|
else if (_dirty.Count > 0)
|
|
{
|
|
foreach (var visual in _dirty)
|
|
{
|
|
_sceneBuilder.Update(scene, visual);
|
|
}
|
|
}
|
|
|
|
lock (_sceneLock)
|
|
{
|
|
var oldScene = _scene;
|
|
_scene = sceneRef;
|
|
oldScene?.Dispose();
|
|
}
|
|
|
|
_dirty.Clear();
|
|
}
|
|
else
|
|
{
|
|
lock (_sceneLock)
|
|
{
|
|
var oldScene = _scene;
|
|
_scene = null;
|
|
oldScene?.Dispose();
|
|
}
|
|
}
|
|
}
|
|
|
|
private IRef<IRenderTargetBitmapImpl> GetOverlay(
|
|
IDrawingContextImpl parentContext,
|
|
Size size,
|
|
double scaling)
|
|
{
|
|
var pixelSize = size * scaling;
|
|
|
|
if (_overlay == null ||
|
|
_overlay.Item.PixelSize.Width != pixelSize.Width ||
|
|
_overlay.Item.PixelSize.Height != pixelSize.Height)
|
|
{
|
|
_overlay?.Dispose();
|
|
_overlay = RefCountable.Create(parentContext.CreateLayer(size));
|
|
}
|
|
|
|
return _overlay;
|
|
}
|
|
|
|
private void SaveDebugFrames(int id)
|
|
{
|
|
var index = 0;
|
|
|
|
foreach (var layer in Layers)
|
|
{
|
|
var fileName = Path.Combine(DebugFramesPath, $"frame-{id}-layer-{index++}.png");
|
|
layer.Bitmap.Item.Save(fileName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|