A cross-platform UI framework for .NET
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.
 
 
 

540 lines
18 KiB

using System;
using System.Collections.Generic;
using System.Linq;
using Avalonia.Controls.Presenters;
using Avalonia.Controls.Primitives;
using Avalonia.Controls.Templates;
using Avalonia.Layout;
using Avalonia.Media;
using Avalonia.UnitTests;
using Avalonia.VisualTree;
using Moq;
using Xunit;
namespace Avalonia.Controls.UnitTests
{
public class ScrollViewerTests : ScopedTestBase
{
private readonly MouseTestHelper _mouse = new();
[Fact]
public void Content_Is_Created()
{
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = "Foo",
};
InitializeScrollViewer(target);
Assert.IsType<TextBlock>(target.Presenter.Child);
}
[Fact]
public void Offset_Should_Be_Coerced_To_Viewport()
{
var target = new ScrollViewer
{
Extent = new Size(20, 20),
Viewport = new Size(10, 10),
Offset = new Vector(12, 12)
};
Assert.Equal(new Vector(10, 10), target.Offset);
}
[Fact]
public void Test_ScrollToHome()
{
var target = new ScrollViewer
{
Extent = new Size(50, 50),
Viewport = new Size(10, 10),
Offset = new Vector(25, 25)
};
target.ScrollToHome();
Assert.Equal(new Vector(0, 0), target.Offset);
}
[Fact]
public void Test_ScrollToEnd()
{
var target = new ScrollViewer
{
Extent = new Size(50, 50),
Viewport = new Size(10, 10),
Offset = new Vector(25, 25)
};
target.ScrollToEnd();
Assert.Equal(new Vector(0, 40), target.Offset);
}
[Fact]
public void SmallChange_Should_Be_16()
{
var target = new ScrollViewer();
Assert.Equal(new Size(16, 16), target.SmallChange);
}
[Fact]
public void LargeChange_Should_Be_Viewport()
{
var target = new ScrollViewer
{
Viewport = new Size(104, 143)
};
Assert.Equal(new Size(104, 143), target.LargeChange);
}
[Fact]
public void SmallChange_Should_Come_From_ILogicalScrollable_If_Present()
{
var child = new Mock<Control>();
var logicalScroll = child.As<ILogicalScrollable>();
logicalScroll.Setup(x => x.IsLogicalScrollEnabled).Returns(true);
logicalScroll.Setup(x => x.ScrollSize).Returns(new Size(12, 43));
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = child.Object,
};
InitializeScrollViewer(target);
Assert.Equal(new Size(12, 43), target.SmallChange);
}
[Fact]
public void LargeChange_Should_Come_From_ILogicalScrollable_If_Present()
{
var child = new Mock<Control>();
var logicalScroll = child.As<ILogicalScrollable>();
logicalScroll.Setup(x => x.IsLogicalScrollEnabled).Returns(true);
logicalScroll.Setup(x => x.PageScrollSize).Returns(new Size(45, 67));
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = child.Object,
};
InitializeScrollViewer(target);
Assert.Equal(new Size(45, 67), target.LargeChange);
}
[Fact]
public void Changing_Extent_Should_Raise_ScrollChanged()
{
var target = new ScrollViewer();
var root = new TestRoot(target);
var raised = 0;
target.Extent = new Size(100, 100);
target.Viewport = new Size(50, 50);
target.Offset = new Vector(10, 10);
root.LayoutManager.ExecuteInitialLayoutPass();
target.ScrollChanged += (s, e) =>
{
Assert.Equal(new Vector(11, 12), e.ExtentDelta);
Assert.Equal(default, e.OffsetDelta);
Assert.Equal(default, e.ViewportDelta);
++raised;
};
target.Extent = new Size(111, 112);
Assert.Equal(0, raised);
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(1, raised);
}
[Fact]
public void Changing_Offset_Should_Raise_ScrollChanged()
{
var target = new ScrollViewer();
var root = new TestRoot(target);
var raised = 0;
target.Extent = new Size(100, 100);
target.Viewport = new Size(50, 50);
target.Offset = new Vector(10, 10);
root.LayoutManager.ExecuteInitialLayoutPass();
target.ScrollChanged += (s, e) =>
{
Assert.Equal(default, e.ExtentDelta);
Assert.Equal(new Vector(12, 14), e.OffsetDelta);
Assert.Equal(default, e.ViewportDelta);
++raised;
};
target.Offset = new Vector(22, 24);
Assert.Equal(0, raised);
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(1, raised);
}
[Fact]
public void Changing_Viewport_Should_Raise_ScrollChanged()
{
var target = new ScrollViewer();
var root = new TestRoot(target);
var raised = 0;
target.Extent = new Size(100, 100);
target.Viewport = new Size(50, 50);
target.Offset = new Vector(10, 10);
root.LayoutManager.ExecuteInitialLayoutPass();
target.ScrollChanged += (s, e) =>
{
Assert.Equal(default, e.ExtentDelta);
Assert.Equal(default, e.OffsetDelta);
Assert.Equal(new Vector(6, 8), e.ViewportDelta);
++raised;
};
target.Viewport = new Size(56, 58);
Assert.Equal(0, raised);
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(1, raised);
}
[Fact]
public void Reducing_Extent_Should_Constrain_Offset()
{
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
};
var root = new TestRoot(target);
var raised = 0;
target.Extent = new (100, 100);
target.Viewport = new(50, 50);
target.Offset = new Vector(50, 50);
root.LayoutManager.ExecuteInitialLayoutPass();
target.ScrollChanged += (s, e) =>
{
Assert.Equal(new Vector(-30, -30), e.ExtentDelta);
Assert.Equal(new Vector(-30, -30), e.OffsetDelta);
Assert.Equal(default, e.ViewportDelta);
++raised;
};
target.Extent = new(70, 70);
Assert.Equal(0, raised);
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(1, raised);
Assert.Equal(new Vector(20, 20), target.Offset);
}
[Fact]
public void Scroll_Does_Not_Jump_When_Viewport_Becomes_Smaller_While_Dragging_ScrollBar_Thumb()
{
var content = new TestContent
{
MeasureSize = new Size(1000, 10000),
};
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = content,
};
var root = new TestRoot(target);
root.LayoutManager.ExecuteInitialLayoutPass();
Assert.Equal(new Size(1000, 10000), target.Extent);
Assert.Equal(new Size(1000, 1000), target.Viewport);
// We're working in absolute coordinates (i.e. relative to the root) and clicking on
// the center of the vertical thumb.
var thumb = GetVerticalThumb(target);
var p = GetRootPoint(thumb, thumb.Bounds.Center);
// Press the mouse button in the center of the thumb.
_mouse.Down(thumb, position: p);
root.LayoutManager.ExecuteLayoutPass();
// Drag the thumb down 300 pixels.
_mouse.Move(thumb, p += new Vector(0, 300));
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(new Vector(0, 3000), target.Offset);
Assert.Equal(300, thumb.Bounds.Top);
// Now the extent changes from 10,000 to 5000.
content.MeasureSize /= 2;
content.InvalidateMeasure();
root.LayoutManager.ExecuteLayoutPass();
// Due to the extent change, the thumb moves down but the value remains the same.
Assert.Equal(600, thumb.Bounds.Top);
Assert.Equal(new Vector(0, 3000), target.Offset);
// Drag the thumb down another 100 pixels.
_mouse.Move(thumb, p += new Vector(0, 100));
root.LayoutManager.ExecuteLayoutPass();
// The drag should not cause the offset/thumb to jump *up* to the current absolute
// mouse position, i.e. it should move down in the direction of the drag even if the
// absolute mouse position is now above the thumb.
Assert.Equal(700, thumb.Bounds.Top);
Assert.Equal(new Vector(0, 3500), target.Offset);
}
[Fact]
public void Thumb_Does_Not_Become_Detached_From_Mouse_Position_When_Scrolling_Past_The_Start()
{
var content = new TestContent();
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = content,
};
var root = new TestRoot(target);
root.LayoutManager.ExecuteInitialLayoutPass();
Assert.Equal(new Size(1000, 2000), target.Extent);
Assert.Equal(new Size(1000, 1000), target.Viewport);
// We're working in absolute coordinates (i.e. relative to the root) and clicking on
// the center of the vertical thumb.
var thumb = GetVerticalThumb(target);
var p = GetRootPoint(thumb, thumb.Bounds.Center);
// Press the mouse button in the center of the thumb.
_mouse.Down(thumb, position: p);
root.LayoutManager.ExecuteLayoutPass();
// Drag the thumb down 100 pixels.
_mouse.Move(thumb, p += new Vector(0, 100));
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(new Vector(0, 200), target.Offset);
Assert.Equal(100, thumb.Bounds.Top);
// Drag the thumb up 200 pixels - 100 pixels past the top of the scrollbar.
_mouse.Move(thumb, p -= new Vector(0, 200));
root.LayoutManager.ExecuteLayoutPass();
Assert.Equal(new Vector(0, 0), target.Offset);
Assert.Equal(0, thumb.Bounds.Top);
// Drag the thumb back down 200 pixels.
_mouse.Move(thumb, p += new Vector(0, 200));
root.LayoutManager.ExecuteLayoutPass();
// We should now be back in the state after we first scrolled down 100 pixels.
Assert.Equal(new Vector(0, 200), target.Offset);
Assert.Equal(100, thumb.Bounds.Top);
}
[Fact]
public void BringIntoViewOnFocusChange_Scrolls_Child_Control_Into_View_When_Focused()
{
using var app = UnitTestApplication.Start(TestServices.RealFocus);
var content = new StackPanel
{
Children =
{
new Button
{
Width = 100,
Height = 900,
},
new Button
{
Width = 100,
Height = 900,
},
}
};
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = content,
};
var root = new TestRoot(target);
root.LayoutManager.ExecuteInitialLayoutPass();
var button = (Button)content.Children[1];
button.Focus();
Assert.Equal(new Vector(0, 800), target.Offset);
}
[Fact]
public void BringIntoViewOnFocusChange_False_Does_Not_Scroll_Child_Control_Into_View_When_Focused()
{
var content = new StackPanel
{
Children =
{
new Button
{
Width = 100,
Height = 900,
},
new Button
{
Width = 100,
Height = 900,
},
}
};
var target = new ScrollViewer
{
Template = new FuncControlTemplate<ScrollViewer>(CreateTemplate),
Content = content,
};
var root = new TestRoot(target);
root.LayoutManager.ExecuteInitialLayoutPass();
var button = (Button)content.Children[1];
button.Focus();
Assert.Equal(new Vector(0, 0), target.Offset);
}
private Point GetRootPoint(Visual control, Point p)
{
if (control.GetVisualRoot() is Visual root &&
control.TransformToVisual(root) is Matrix m)
{
return p.Transform(m);
}
throw new InvalidOperationException("Could not get the point in root coordinates.");
}
internal static Control CreateTemplate(ScrollViewer control, INameScope scope)
{
return new Grid
{
ColumnDefinitions = new ColumnDefinitions
{
new ColumnDefinition(1, GridUnitType.Star),
new ColumnDefinition(GridLength.Auto),
},
RowDefinitions = new RowDefinitions
{
new RowDefinition(1, GridUnitType.Star),
new RowDefinition(GridLength.Auto),
},
Children =
{
new ScrollContentPresenter
{
Name = "PART_ContentPresenter",
}.RegisterInNameScope(scope),
new ScrollBar
{
Name = "PART_HorizontalScrollBar",
Orientation = Orientation.Horizontal,
Template = new FuncControlTemplate<ScrollBar>(CreateScrollBarTemplate),
[~ScrollBar.VisibilityProperty] = control[~ScrollViewer.HorizontalScrollBarVisibilityProperty],
[Grid.RowProperty] = 1,
}.RegisterInNameScope(scope),
new ScrollBar
{
Name = "PART_VerticalScrollBar",
Orientation = Orientation.Vertical,
Template = new FuncControlTemplate<ScrollBar>(CreateScrollBarTemplate),
[~ScrollBar.VisibilityProperty] = control[~ScrollViewer.VerticalScrollBarVisibilityProperty],
[Grid.ColumnProperty] = 1,
}.RegisterInNameScope(scope),
},
};
}
private static Control CreateScrollBarTemplate(ScrollBar scrollBar, INameScope scope)
{
return new Border
{
Child = new Track
{
Name = "track",
IsDirectionReversed = true,
[!Track.MinimumProperty] = scrollBar[!RangeBase.MinimumProperty],
[!Track.MaximumProperty] = scrollBar[!RangeBase.MaximumProperty],
[!!Track.ValueProperty] = scrollBar[!!RangeBase.ValueProperty],
[!Track.ViewportSizeProperty] = scrollBar[!ScrollBar.ViewportSizeProperty],
[!Track.OrientationProperty] = scrollBar[!ScrollBar.OrientationProperty],
Thumb = new Thumb
{
Template = new FuncControlTemplate<Thumb>(CreateThumbTemplate),
},
}.RegisterInNameScope(scope),
};
}
private static Control CreateThumbTemplate(Thumb control, INameScope scope)
{
return new Border
{
Background = Brushes.Gray,
};
}
private Thumb GetVerticalThumb(ScrollViewer target)
{
var scrollbar = Assert.IsType<ScrollBar>(
target.GetTemplateChildren().FirstOrDefault(x => x.Name == "PART_VerticalScrollBar"));
var track = Assert.IsType<Track>(
scrollbar.GetTemplateChildren().FirstOrDefault(x => x.Name == "track"));
return Assert.IsType<Thumb>(track.Thumb);
}
private static void InitializeScrollViewer(ScrollViewer target)
{
target.ApplyTemplate();
var presenter = (ScrollContentPresenter)target.Presenter;
presenter.AttachToScrollViewer();
presenter.UpdateChild();
}
private class TestContent : Control
{
public Size MeasureSize { get; set; } = new Size(1000, 2000);
protected override Size MeasureOverride(Size availableSize)
{
return MeasureSize;
}
}
}
}