Browse Source

Merge branch 'master' into fixes/2420-slider-leak

pull/2438/head
danwalmsley 7 years ago
committed by GitHub
parent
commit
09c47b7f4d
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 4
      azure-pipelines.yml
  2. 2
      nukebuild/Numerge
  3. 4
      src/Avalonia.Controls.DataGrid/Collections/DataGridCollectionView.cs
  4. 3
      src/Avalonia.Controls.DataGrid/Properties/AssemblyInfo.cs
  5. 21
      src/Avalonia.Controls/Grid.cs
  6. 11
      src/Avalonia.Controls/Image.cs
  7. 5
      src/Avalonia.Controls/Platform/InProcessDragSource.cs
  8. 72
      src/Avalonia.Controls/Shapes/Shape.cs
  9. 54
      src/Avalonia.Controls/StackPanel.cs
  10. 2
      src/Avalonia.Controls/Viewbox.cs
  11. 62
      src/Avalonia.Layout/LayoutExtensions.cs
  12. 2
      src/Avalonia.Layout/Layoutable.cs
  13. 26
      src/Avalonia.Themes.Default/ScrollBar.xaml
  14. 10
      src/Avalonia.Visuals/Media/BrushExtensions.cs
  15. 72
      src/Avalonia.Visuals/Media/Pen.cs
  16. 3
      src/Avalonia.Visuals/Media/PenLineCap.cs
  17. 19
      src/Skia/Avalonia.Skia/DrawingContextImpl.cs
  18. 8
      src/Windows/Avalonia.Direct2D1/PrimitiveExtensions.cs
  19. 160
      tests/Avalonia.Controls.UnitTests/StackPanelTests.cs
  20. 8
      tests/Avalonia.RenderTests/Shapes/PathTests.cs
  21. 3
      tests/Avalonia.RenderTests/Shapes/PolylineTests.cs
  22. BIN
      tests/TestFiles/Direct2D1/Shapes/Path/Path_With_PenLineCap.expected.png
  23. BIN
      tests/TestFiles/Skia/Shapes/Path/Path_With_PenLineCap.expected.png

4
azure-pipelines.yml

@ -32,7 +32,7 @@ jobs:
- job: macOS - job: macOS
pool: pool:
vmImage: 'xcode9-macos10.13' vmImage: 'macOS-10.14'
steps: steps:
- task: DotNetCoreInstaller@0 - task: DotNetCoreInstaller@0
inputs: inputs:
@ -49,7 +49,7 @@ jobs:
inputs: inputs:
actions: 'build' actions: 'build'
scheme: '' scheme: ''
sdk: 'macosx10.13' sdk: 'macosx10.14'
configuration: 'Release' configuration: 'Release'
xcWorkspacePath: '**/*.xcodeproj/project.xcworkspace' xcWorkspacePath: '**/*.xcodeproj/project.xcworkspace'
xcodeVersion: 'default' # Options: 8, 9, default, specifyPath xcodeVersion: 'default' # Options: 8, 9, default, specifyPath

2
nukebuild/Numerge

@ -1 +1 @@
Subproject commit 4464343aef5c8ab7a42fcb20a483a6058199f8b8 Subproject commit aef10ae67dc55c95f49b52a505a0be33bfa297a5

4
src/Avalonia.Controls.DataGrid/Collections/DataGridCollectionView.cs

@ -927,7 +927,7 @@ namespace Avalonia.Collections
/// <remarks> /// <remarks>
/// <p> /// <p>
/// Clear a sort criteria by assigning SortDescription.Empty to this property. /// Clear a sort criteria by assigning SortDescription.Empty to this property.
/// One or more sort criteria in form of <seealso cref="SortDescription"/> /// One or more sort criteria in form of <seealso cref="DataGridSortDescription"/>
/// can be used, each specifying a property and direction to sort by. /// can be used, each specifying a property and direction to sort by.
/// </p> /// </p>
/// </remarks> /// </remarks>
@ -4312,4 +4312,4 @@ namespace Avalonia.Collections
} }
} }
} }
} }

3
src/Avalonia.Controls.DataGrid/Properties/AssemblyInfo.cs

@ -8,7 +8,6 @@ using Avalonia.Metadata;
[assembly: InternalsVisibleTo("Avalonia.Controls.UnitTests")] [assembly: InternalsVisibleTo("Avalonia.Controls.UnitTests")]
[assembly: InternalsVisibleTo("Avalonia.DesignerSupport")] [assembly: InternalsVisibleTo("Avalonia.DesignerSupport")]
[assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia")]
[assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls")] [assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls")]
[assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls.Primitives")]
[assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls.Collections")] [assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls.Collections")]
[assembly: XmlnsDefinition("https://github.com/avaloniaui", "Avalonia.Controls.Primitives")]

21
src/Avalonia.Controls/Grid.cs

@ -177,6 +177,17 @@ namespace Avalonia.Controls
return element.GetValue(RowSpanProperty); return element.GetValue(RowSpanProperty);
} }
/// <summary>
/// Gets the value of the IsSharedSizeScope attached property for a control.
/// </summary>
/// <param name="element">The control.</param>
/// <returns>The control's IsSharedSizeScope value.</returns>
public static bool GetIsSharedSizeScope(AvaloniaObject element)
{
return element.GetValue(IsSharedSizeScopeProperty);
}
/// <summary> /// <summary>
/// Sets the value of the Column attached property for a control. /// Sets the value of the Column attached property for a control.
/// </summary> /// </summary>
@ -217,6 +228,16 @@ namespace Avalonia.Controls
element.SetValue(RowSpanProperty, value); element.SetValue(RowSpanProperty, value);
} }
/// <summary>
/// Sets the value of IsSharedSizeScope property for a control.
/// </summary>
/// <param name="element">The control.</param>
/// <param name="value">The IsSharedSizeScope value.</param>
public static void SetIsSharedSizeScope(AvaloniaObject element, bool value)
{
element.SetValue(IsSharedSizeScopeProperty, value);
}
/// <summary> /// <summary>
/// Gets the result of the last column measurement. /// Gets the result of the last column measurement.
/// Use this result to reduce the arrange calculation. /// Use this result to reduce the arrange calculation.

11
src/Avalonia.Controls/Image.cs

@ -81,23 +81,22 @@ namespace Avalonia.Controls
protected override Size MeasureOverride(Size availableSize) protected override Size MeasureOverride(Size availableSize)
{ {
var source = Source; var source = Source;
var result = new Size();
if (source != null) if (source != null)
{ {
Size sourceSize = new Size(source.PixelSize.Width, source.PixelSize.Height); Size sourceSize = new Size(source.PixelSize.Width, source.PixelSize.Height);
if (double.IsInfinity(availableSize.Width) || double.IsInfinity(availableSize.Height)) if (double.IsInfinity(availableSize.Width) || double.IsInfinity(availableSize.Height))
{ {
return sourceSize; result = sourceSize;
} }
else else
{ {
return Stretch.CalculateSize(availableSize, sourceSize); result = Stretch.CalculateSize(availableSize, sourceSize);
} }
} }
else
{ return result.Constrain(availableSize);
return new Size();
}
} }
/// <inheritdoc/> /// <inheritdoc/>

5
src/Avalonia.Controls/Platform/InProcessDragSource.cs

@ -147,7 +147,10 @@ namespace Avalonia.Platform
e.Handled = true; e.Handled = true;
} }
else if (e.Key == Key.LeftCtrl || e.Key == Key.RightCtrl || e.Key == Key.LeftAlt || e.Key == Key.RightAlt) else if (e.Key == Key.LeftCtrl || e.Key == Key.RightCtrl || e.Key == Key.LeftAlt || e.Key == Key.RightAlt)
RaiseEventAndUpdateCursor(RawDragEventType.DragOver, _lastRoot, _lastPosition, e.Modifiers); {
if (_lastRoot != null)
RaiseEventAndUpdateCursor(RawDragEventType.DragOver, _lastRoot, _lastPosition, e.Modifiers);
}
} }
private void ProcessMouseEvents(RawMouseEventArgs e) private void ProcessMouseEvents(RawMouseEventArgs e)

72
src/Avalonia.Controls/Shapes/Shape.cs

@ -21,13 +21,19 @@ namespace Avalonia.Controls.Shapes
public static readonly StyledProperty<AvaloniaList<double>> StrokeDashArrayProperty = public static readonly StyledProperty<AvaloniaList<double>> StrokeDashArrayProperty =
AvaloniaProperty.Register<Shape, AvaloniaList<double>>(nameof(StrokeDashArray)); AvaloniaProperty.Register<Shape, AvaloniaList<double>>(nameof(StrokeDashArray));
public static readonly StyledProperty<double> StrokeDashOffsetProperty = public static readonly StyledProperty<double> StrokeDashOffsetProperty =
AvaloniaProperty.Register<Shape, double>(nameof(StrokeDashOffset)); AvaloniaProperty.Register<Shape, double>(nameof(StrokeDashOffset));
public static readonly StyledProperty<double> StrokeThicknessProperty = public static readonly StyledProperty<double> StrokeThicknessProperty =
AvaloniaProperty.Register<Shape, double>(nameof(StrokeThickness)); AvaloniaProperty.Register<Shape, double>(nameof(StrokeThickness));
public static readonly StyledProperty<PenLineCap> StrokeLineCapProperty =
AvaloniaProperty.Register<Shape, PenLineCap>(nameof(StrokeLineCap), PenLineCap.Flat);
public static readonly StyledProperty<PenLineJoin> StrokeJoinProperty =
AvaloniaProperty.Register<Shape, PenLineJoin>(nameof(StrokeJoin), PenLineJoin.Miter);
private Matrix _transform = Matrix.Identity; private Matrix _transform = Matrix.Identity;
private Geometry _definingGeometry; private Geometry _definingGeometry;
private Geometry _renderedGeometry; private Geometry _renderedGeometry;
@ -36,7 +42,9 @@ namespace Avalonia.Controls.Shapes
static Shape() static Shape()
{ {
AffectsMeasure<Shape>(StretchProperty, StrokeThicknessProperty); AffectsMeasure<Shape>(StretchProperty, StrokeThicknessProperty);
AffectsRender<Shape>(FillProperty, StrokeProperty, StrokeDashArrayProperty);
AffectsRender<Shape>(FillProperty, StrokeProperty, StrokeDashArrayProperty, StrokeDashOffsetProperty,
StrokeThicknessProperty, StrokeLineCapProperty, StrokeJoinProperty);
} }
public Geometry DefiningGeometry public Geometry DefiningGeometry
@ -106,7 +114,7 @@ namespace Avalonia.Controls.Shapes
get { return GetValue(StrokeDashArrayProperty); } get { return GetValue(StrokeDashArrayProperty); }
set { SetValue(StrokeDashArrayProperty, value); } set { SetValue(StrokeDashArrayProperty, value); }
} }
public double StrokeDashOffset public double StrokeDashOffset
{ {
get { return GetValue(StrokeDashOffsetProperty); } get { return GetValue(StrokeDashOffsetProperty); }
@ -119,13 +127,17 @@ namespace Avalonia.Controls.Shapes
set { SetValue(StrokeThicknessProperty, value); } set { SetValue(StrokeThicknessProperty, value); }
} }
public PenLineCap StrokeDashCap { get; set; } = PenLineCap.Flat; public PenLineCap StrokeLineCap
{
public PenLineCap StrokeStartLineCap { get; set; } = PenLineCap.Flat; get { return GetValue(StrokeLineCapProperty); }
set { SetValue(StrokeLineCapProperty, value); }
public PenLineCap StrokeEndLineCap { get; set; } = PenLineCap.Flat; }
public PenLineJoin StrokeJoin { get; set; } = PenLineJoin.Miter; public PenLineJoin StrokeJoin
{
get { return GetValue(StrokeJoinProperty); }
set { SetValue(StrokeJoinProperty, value); }
}
public override void Render(DrawingContext context) public override void Render(DrawingContext context)
{ {
@ -133,8 +145,8 @@ namespace Avalonia.Controls.Shapes
if (geometry != null) if (geometry != null)
{ {
var pen = new Pen(Stroke, StrokeThickness, new DashStyle(StrokeDashArray, StrokeDashOffset), var pen = new Pen(Stroke, StrokeThickness, new DashStyle(StrokeDashArray, StrokeDashOffset),
StrokeDashCap, StrokeStartLineCap, StrokeEndLineCap, StrokeJoin); StrokeLineCap, StrokeJoin);
context.DrawGeometry(Fill, pen, geometry); context.DrawGeometry(Fill, pen, geometry);
} }
} }
@ -169,11 +181,11 @@ namespace Avalonia.Controls.Shapes
protected void InvalidateGeometry() protected void InvalidateGeometry()
{ {
this._renderedGeometry = null; _renderedGeometry = null;
this._definingGeometry = null; _definingGeometry = null;
InvalidateMeasure(); InvalidateMeasure();
} }
protected override Size MeasureOverride(Size availableSize) protected override Size MeasureOverride(Size availableSize)
{ {
bool deferCalculateTransform; bool deferCalculateTransform;
@ -203,10 +215,10 @@ namespace Avalonia.Controls.Shapes
return CalculateShapeSizeAndSetTransform(availableSize); return CalculateShapeSizeAndSetTransform(availableSize);
} }
} }
protected override Size ArrangeOverride(Size finalSize) protected override Size ArrangeOverride(Size finalSize)
{ {
if(_calculateTransformOnArrange) if (_calculateTransformOnArrange)
{ {
_calculateTransformOnArrange = false; _calculateTransformOnArrange = false;
CalculateShapeSizeAndSetTransform(finalSize); CalculateShapeSizeAndSetTransform(finalSize);
@ -312,25 +324,25 @@ namespace Avalonia.Controls.Shapes
private static void AffectsGeometryInvalidate(AvaloniaPropertyChangedEventArgs e) private static void AffectsGeometryInvalidate(AvaloniaPropertyChangedEventArgs e)
{ {
var control = e.Sender as Shape; if (!(e.Sender is Shape control))
{
return;
}
if (control != null) // If the geometry is invalidated when Bounds changes, only invalidate when the Size
// portion changes.
if (e.Property == BoundsProperty)
{ {
// If the geometry is invalidated when Bounds changes, only invalidate when the Size var oldBounds = (Rect)e.OldValue;
// portion changes. var newBounds = (Rect)e.NewValue;
if (e.Property == BoundsProperty)
{
var oldBounds = (Rect)e.OldValue;
var newBounds = (Rect)e.NewValue;
if (oldBounds.Size == newBounds.Size) if (oldBounds.Size == newBounds.Size)
{ {
return; return;
}
} }
control.InvalidateGeometry();
} }
control.InvalidateGeometry();
} }
} }
} }

54
src/Avalonia.Controls/StackPanel.cs

@ -4,6 +4,7 @@
using System; using System;
using System.Linq; using System.Linq;
using Avalonia.Input; using Avalonia.Input;
using Avalonia.Layout;
namespace Avalonia.Controls namespace Avalonia.Controls
{ {
@ -219,30 +220,16 @@ namespace Avalonia.Controls
measuredWidth -= (hasVisibleChild ? spacing : 0); measuredWidth -= (hasVisibleChild ? spacing : 0);
} }
return new Size(measuredWidth, measuredHeight); return new Size(measuredWidth, measuredHeight).Constrain(availableSize);
} }
/// <summary> /// <inheritdoc/>
/// Arranges the control's children.
/// </summary>
/// <param name="finalSize">The size allocated to the control.</param>
/// <returns>The space taken.</returns>
protected override Size ArrangeOverride(Size finalSize) protected override Size ArrangeOverride(Size finalSize)
{ {
var orientation = Orientation; var orientation = Orientation;
double arrangedWidth = finalSize.Width; var spacing = Spacing;
double arrangedHeight = finalSize.Height; var finalRect = new Rect(finalSize);
double spacing = Spacing; var pos = 0.0;
bool hasVisibleChild = Children.Any(c => c.IsVisible);
if (Orientation == Orientation.Vertical)
{
arrangedHeight = 0;
}
else
{
arrangedWidth = 0;
}
foreach (Control child in Children) foreach (Control child in Children)
{ {
@ -251,32 +238,21 @@ namespace Avalonia.Controls
if (orientation == Orientation.Vertical) if (orientation == Orientation.Vertical)
{ {
double width = Math.Max(childWidth, arrangedWidth); var rect = new Rect(0, pos, childWidth, childHeight)
Rect childFinal = new Rect(0, arrangedHeight, width, childHeight); .Align(finalRect, child.HorizontalAlignment, VerticalAlignment.Top);
ArrangeChild(child, childFinal, finalSize, orientation); ArrangeChild(child, rect, finalSize, orientation);
arrangedWidth = Math.Max(arrangedWidth, childWidth); pos += childHeight + spacing;
arrangedHeight += childHeight + (child.IsVisible ? spacing : 0);
} }
else else
{ {
double height = Math.Max(childHeight, arrangedHeight); var rect = new Rect(pos, 0, childWidth, childHeight)
Rect childFinal = new Rect(arrangedWidth, 0, childWidth, height); .Align(finalRect, HorizontalAlignment.Left, child.VerticalAlignment);
ArrangeChild(child, childFinal, finalSize, orientation); ArrangeChild(child, rect, finalSize, orientation);
arrangedWidth += childWidth + (child.IsVisible ? spacing : 0); pos += childWidth + spacing;
arrangedHeight = Math.Max(arrangedHeight, childHeight);
} }
} }
if (orientation == Orientation.Vertical) return finalSize;
{
arrangedHeight = Math.Max(arrangedHeight - (hasVisibleChild ? spacing : 0), finalSize.Height);
}
else
{
arrangedWidth = Math.Max(arrangedWidth - (hasVisibleChild ? spacing : 0), finalSize.Width);
}
return new Size(arrangedWidth, arrangedHeight);
} }
internal virtual void ArrangeChild( internal virtual void ArrangeChild(

2
src/Avalonia.Controls/Viewbox.cs

@ -49,7 +49,7 @@ namespace Avalonia.Controls
var scale = GetScale(availableSize, childSize, Stretch); var scale = GetScale(availableSize, childSize, Stretch);
return childSize * scale; return (childSize * scale).Constrain(availableSize);
} }
return new Size(); return new Size();

62
src/Avalonia.Layout/LayoutExtensions.cs

@ -0,0 +1,62 @@
using System;
namespace Avalonia.Layout
{
/// <summary>
/// Extension methods for layout types.
/// </summary>
public static class LayoutExtensions
{
/// <summary>
/// Aligns a rect in a constraining rect according to horizontal and vertical alignment
/// settings.
/// </summary>
/// <param name="rect">The rect to align.</param>
/// <param name="constraint">The constraining rect.</param>
/// <param name="horizontalAlignment">The horizontal alignment.</param>
/// <param name="verticalAlignment">The vertical alignment.</param>
/// <returns></returns>
public static Rect Align(
this Rect rect,
Rect constraint,
HorizontalAlignment horizontalAlignment,
VerticalAlignment verticalAlignment)
{
switch (horizontalAlignment)
{
case HorizontalAlignment.Center:
rect = rect.WithX((constraint.Width - rect.Width) / 2);
break;
case HorizontalAlignment.Right:
rect = rect.WithX(constraint.Width - rect.Width);
break;
case HorizontalAlignment.Stretch:
rect = new Rect(
0,
rect.Y,
Math.Max(constraint.Width, rect.Width),
rect.Height);
break;
}
switch (verticalAlignment)
{
case VerticalAlignment.Center:
rect = rect.WithY((constraint.Height - rect.Height) / 2);
break;
case VerticalAlignment.Bottom:
rect = rect.WithY(constraint.Height - rect.Height);
break;
case VerticalAlignment.Stretch:
rect = new Rect(
rect.X,
0,
rect.Width,
Math.Max(constraint.Height, rect.Height));
break;
}
return rect;
}
}
}

2
src/Avalonia.Layout/Layoutable.cs

@ -314,7 +314,7 @@ namespace Avalonia.Layout
try try
{ {
_measuring = true; _measuring = true;
desiredSize = MeasureCore(availableSize).Constrain(availableSize); desiredSize = MeasureCore(availableSize);
} }
finally finally
{ {

26
src/Avalonia.Themes.Default/ScrollBar.xaml

@ -30,13 +30,7 @@
Classes="repeattrack" Classes="repeattrack"
Focusable="False"/> Focusable="False"/>
</Track.IncreaseButton> </Track.IncreaseButton>
<Thumb Name="thumb"> <Thumb Name="thumb"/>
<Thumb.Template>
<ControlTemplate>
<Border Background="{DynamicResource ThemeControlHighBrush}" />
</ControlTemplate>
</Thumb.Template>
</Thumb>
</Track> </Track>
<RepeatButton Name="PART_LineDownButton" <RepeatButton Name="PART_LineDownButton"
Classes="repeat" Classes="repeat"
@ -84,13 +78,7 @@
Classes="repeattrack" Classes="repeattrack"
Focusable="False"/> Focusable="False"/>
</Track.IncreaseButton> </Track.IncreaseButton>
<Thumb Name="thumb"> <Thumb Name="thumb"/>
<Thumb.Template>
<ControlTemplate>
<Border Background="{DynamicResource ThemeControlHighBrush}" />
</ControlTemplate>
</Thumb.Template>
</Thumb>
</Track> </Track>
<RepeatButton Name="PART_LineDownButton" <RepeatButton Name="PART_LineDownButton"
Classes="repeat" Classes="repeat"
@ -106,6 +94,16 @@
</ControlTemplate> </ControlTemplate>
</Setter> </Setter>
</Style> </Style>
<Style Selector="ScrollBar /template/ Thumb#thumb">
<Setter Property="Background" Value="{DynamicResource ThemeControlHighBrush}"/>
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<Border Background="{TemplateBinding Background}"/>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
<Style Selector="ScrollBar:horizontal /template/ Thumb#thumb"> <Style Selector="ScrollBar:horizontal /template/ Thumb#thumb">
<Setter Property="MinWidth" Value="{DynamicResource ScrollBarThickness}" /> <Setter Property="MinWidth" Value="{DynamicResource ScrollBarThickness}" />
</Style> </Style>

10
src/Avalonia.Visuals/Media/BrushExtensions.cs

@ -34,16 +34,14 @@ namespace Avalonia.Media
{ {
Contract.Requires<ArgumentNullException>(pen != null); Contract.Requires<ArgumentNullException>(pen != null);
var brush = pen?.Brush?.ToImmutable(); var brush = pen.Brush?.ToImmutable();
return pen == null || ReferenceEquals(pen?.Brush, brush) ? return ReferenceEquals(pen.Brush, brush) ?
pen : pen :
new Pen( new Pen(
brush, brush,
thickness: pen.Thickness, thickness: pen.Thickness,
dashStyle: pen.DashStyle, dashStyle: pen.DashStyle,
dashCap: pen.DashCap, lineCap: pen.LineCap,
startLineCap: pen.StartLineCap,
endLineCap: pen.EndLineCap,
lineJoin: pen.LineJoin, lineJoin: pen.LineJoin,
miterLimit: pen.MiterLimit); miterLimit: pen.MiterLimit);
} }

72
src/Avalonia.Visuals/Media/Pen.cs

@ -11,63 +11,45 @@ namespace Avalonia.Media
/// <summary> /// <summary>
/// Initializes a new instance of the <see cref="Pen"/> class. /// Initializes a new instance of the <see cref="Pen"/> class.
/// </summary> /// </summary>
/// <param name="brush">The brush used to draw.</param> /// <param name="color">The stroke color.</param>
/// <param name="thickness">The stroke thickness.</param> /// <param name="thickness">The stroke thickness.</param>
/// <param name="dashStyle">The dash style.</param> /// <param name="dashStyle">The dash style.</param>
/// <param name="dashCap">The dash cap.</param> /// <param name="lineCap">Specifies the type of graphic shape to use on both ends of a line.</param>
/// <param name="startLineCap">The start line cap.</param>
/// <param name="endLineCap">The end line cap.</param>
/// <param name="lineJoin">The line join.</param> /// <param name="lineJoin">The line join.</param>
/// <param name="miterLimit">The miter limit.</param> /// <param name="miterLimit">The miter limit.</param>
public Pen( public Pen(
IBrush brush, uint color,
double thickness = 1.0, double thickness = 1.0,
DashStyle dashStyle = null, DashStyle dashStyle = null,
PenLineCap dashCap = PenLineCap.Flat, PenLineCap lineCap = PenLineCap.Flat,
PenLineCap startLineCap = PenLineCap.Flat, PenLineJoin lineJoin = PenLineJoin.Miter,
PenLineCap endLineCap = PenLineCap.Flat, double miterLimit = 10.0) : this(new SolidColorBrush(color), thickness, dashStyle, lineCap, lineJoin, miterLimit)
PenLineJoin lineJoin = PenLineJoin.Miter,
double miterLimit = 10.0)
{ {
Brush = brush;
Thickness = thickness;
DashCap = dashCap;
StartLineCap = startLineCap;
EndLineCap = endLineCap;
LineJoin = lineJoin;
MiterLimit = miterLimit;
DashStyle = dashStyle;
} }
/// <summary> /// <summary>
/// Initializes a new instance of the <see cref="Pen"/> class. /// Initializes a new instance of the <see cref="Pen"/> class.
/// </summary> /// </summary>
/// <param name="color">The stroke color.</param> /// <param name="brush">The brush used to draw.</param>
/// <param name="thickness">The stroke thickness.</param> /// <param name="thickness">The stroke thickness.</param>
/// <param name="dashStyle">The dash style.</param> /// <param name="dashStyle">The dash style.</param>
/// <param name="dashCap">The dash cap.</param> /// <param name="lineCap">The line cap.</param>
/// <param name="startLineCap">The start line cap.</param>
/// <param name="endLineCap">The end line cap.</param>
/// <param name="lineJoin">The line join.</param> /// <param name="lineJoin">The line join.</param>
/// <param name="miterLimit">The miter limit.</param> /// <param name="miterLimit">The miter limit.</param>
public Pen( public Pen(
uint color, IBrush brush,
double thickness = 1.0, double thickness = 1.0,
DashStyle dashStyle = null, DashStyle dashStyle = null,
PenLineCap dashCap = PenLineCap.Flat, PenLineCap lineCap = PenLineCap.Flat,
PenLineCap startLineCap = PenLineCap.Flat, PenLineJoin lineJoin = PenLineJoin.Miter,
PenLineCap endLineCap = PenLineCap.Flat,
PenLineJoin lineJoin = PenLineJoin.Miter,
double miterLimit = 10.0) double miterLimit = 10.0)
{ {
Brush = new SolidColorBrush(color); Brush = brush;
Thickness = thickness; Thickness = thickness;
StartLineCap = startLineCap; LineCap = lineCap;
EndLineCap = endLineCap;
LineJoin = lineJoin; LineJoin = lineJoin;
MiterLimit = miterLimit; MiterLimit = miterLimit;
DashStyle = dashStyle; DashStyle = dashStyle;
DashCap = dashCap;
} }
/// <summary> /// <summary>
@ -78,18 +60,26 @@ namespace Avalonia.Media
/// <summary> /// <summary>
/// Gets the stroke thickness. /// Gets the stroke thickness.
/// </summary> /// </summary>
public double Thickness { get; } = 1.0; public double Thickness { get; }
/// <summary>
/// Specifies the style of dashed lines drawn with a <see cref="Pen"/> object.
/// </summary>
public DashStyle DashStyle { get; } public DashStyle DashStyle { get; }
public PenLineCap DashCap { get; } /// <summary>
/// Specifies the type of graphic shape to use on both ends of a line.
public PenLineCap StartLineCap { get; } = PenLineCap.Flat; /// </summary>
public PenLineCap LineCap { get; }
public PenLineCap EndLineCap { get; } = PenLineCap.Flat;
public PenLineJoin LineJoin { get; } = PenLineJoin.Miter; /// <summary>
/// Specifies how to join consecutive line or curve segments in a <see cref="PathFigure"/> (subpath) contained in a <see cref="PathGeometry"/> object.
/// </summary>
public PenLineJoin LineJoin { get; }
public double MiterLimit { get; } = 10.0; /// <summary>
/// The limit on the ratio of the miter length to half this pen's Thickness.
/// </summary>
public double MiterLimit { get; }
} }
} }

3
src/Avalonia.Visuals/Media/PenLineCap.cs

@ -4,7 +4,6 @@ namespace Avalonia.Media
{ {
Flat, Flat,
Round, Round,
Square, Square
Triangle
} }
} }

19
src/Skia/Avalonia.Skia/DrawingContextImpl.cs

@ -578,25 +578,17 @@ namespace Avalonia.Skia
// Need to modify dashes due to Skia modifying their lengths // Need to modify dashes due to Skia modifying their lengths
// https://docs.microsoft.com/en-us/xamarin/xamarin-forms/user-interface/graphics/skiasharp/paths/dots // https://docs.microsoft.com/en-us/xamarin/xamarin-forms/user-interface/graphics/skiasharp/paths/dots
// TODO: Still something is off, dashes are now present, but don't look the same as D2D ones. // TODO: Still something is off, dashes are now present, but don't look the same as D2D ones.
float dashLengthModifier;
float gapLengthModifier;
switch (pen.StartLineCap) switch (pen.LineCap)
{ {
case PenLineCap.Round: case PenLineCap.Round:
paint.StrokeCap = SKStrokeCap.Round; paint.StrokeCap = SKStrokeCap.Round;
dashLengthModifier = -paint.StrokeWidth;
gapLengthModifier = paint.StrokeWidth;
break; break;
case PenLineCap.Square: case PenLineCap.Square:
paint.StrokeCap = SKStrokeCap.Square; paint.StrokeCap = SKStrokeCap.Square;
dashLengthModifier = -paint.StrokeWidth;
gapLengthModifier = paint.StrokeWidth;
break; break;
default: default:
paint.StrokeCap = SKStrokeCap.Butt; paint.StrokeCap = SKStrokeCap.Butt;
dashLengthModifier = 0.0f;
gapLengthModifier = 0.0f;
break; break;
} }
@ -622,13 +614,12 @@ namespace Avalonia.Skia
for (var i = 0; i < srcDashes.Count; ++i) for (var i = 0; i < srcDashes.Count; ++i)
{ {
var lengthModifier = i % 2 == 0 ? dashLengthModifier : gapLengthModifier; dashesArray[i] = (float) srcDashes[i] * paint.StrokeWidth;
// Avalonia dash lengths are relative, but Skia takes absolute sizes - need to scale
dashesArray[i] = (float) srcDashes[i] * paint.StrokeWidth + lengthModifier;
} }
var pe = SKPathEffect.CreateDash(dashesArray, (float) pen.DashStyle.Offset); var offset = (float)(pen.DashStyle.Offset * pen.Thickness);
var pe = SKPathEffect.CreateDash(dashesArray, offset);
paint.PathEffect = pe; paint.PathEffect = pe;
rv.AddDisposable(pe); rv.AddDisposable(pe);

8
src/Windows/Avalonia.Direct2D1/PrimitiveExtensions.cs

@ -122,14 +122,16 @@ namespace Avalonia.Direct2D1
/// <returns>The Direct2D brush.</returns> /// <returns>The Direct2D brush.</returns>
public static StrokeStyle ToDirect2DStrokeStyle(this Avalonia.Media.Pen pen, Factory factory) public static StrokeStyle ToDirect2DStrokeStyle(this Avalonia.Media.Pen pen, Factory factory)
{ {
var d2dLineCap = pen.LineCap.ToDirect2D();
var properties = new StrokeStyleProperties var properties = new StrokeStyleProperties
{ {
DashStyle = DashStyle.Solid, DashStyle = DashStyle.Solid,
MiterLimit = (float)pen.MiterLimit, MiterLimit = (float)pen.MiterLimit,
LineJoin = pen.LineJoin.ToDirect2D(), LineJoin = pen.LineJoin.ToDirect2D(),
StartCap = pen.StartLineCap.ToDirect2D(), StartCap = d2dLineCap,
EndCap = pen.EndLineCap.ToDirect2D(), EndCap = d2dLineCap,
DashCap = pen.DashCap.ToDirect2D() DashCap = d2dLineCap
}; };
float[] dashes = null; float[] dashes = null;
if (pen.DashStyle?.Dashes != null && pen.DashStyle.Dashes.Count > 0) if (pen.DashStyle?.Dashes != null && pen.DashStyle.Dashes.Count > 0)

160
tests/Avalonia.Controls.UnitTests/StackPanelTests.cs

@ -1,7 +1,8 @@
// Copyright (c) The Avalonia Project. All rights reserved. // 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. // Licensed under the MIT license. See licence.md file in the project root for full license information.
using Avalonia.Controls; using System.Linq;
using Avalonia.Layout;
using Xunit; using Xunit;
namespace Avalonia.Controls.UnitTests namespace Avalonia.Controls.UnitTests
@ -147,6 +148,151 @@ namespace Avalonia.Controls.UnitTests
Assert.Equal(new Rect(50, 0, 50, 120), target.Children[2].Bounds); Assert.Equal(new Rect(50, 0, 50, 120), target.Children[2].Bounds);
} }
[Fact]
public void Arranges_Vertical_Children_With_Correct_Bounds()
{
var target = new StackPanel
{
Orientation = Orientation.Vertical,
Children =
{
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Left,
MeasureSize = new Size(50, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Left,
MeasureSize = new Size(150, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Center,
MeasureSize = new Size(50, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Center,
MeasureSize = new Size(150, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Right,
MeasureSize = new Size(50, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Right,
MeasureSize = new Size(150, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Stretch,
MeasureSize = new Size(50, 10),
},
new TestControl
{
HorizontalAlignment = HorizontalAlignment.Stretch,
MeasureSize = new Size(150, 10),
},
}
};
target.Measure(new Size(100, 150));
Assert.Equal(new Size(100, 80), target.DesiredSize);
target.Arrange(new Rect(target.DesiredSize));
var bounds = target.Children.Select(x => x.Bounds).ToArray();
Assert.Equal(
new[]
{
new Rect(0, 0, 50, 10),
new Rect(0, 10, 150, 10),
new Rect(25, 20, 50, 10),
new Rect(-25, 30, 150, 10),
new Rect(50, 40, 50, 10),
new Rect(-50, 50, 150, 10),
new Rect(0, 60, 100, 10),
new Rect(0, 70, 150, 10),
}, bounds);
}
[Fact]
public void Arranges_Horizontal_Children_With_Correct_Bounds()
{
var target = new StackPanel
{
Orientation = Orientation.Horizontal,
Children =
{
new TestControl
{
VerticalAlignment = VerticalAlignment.Top,
MeasureSize = new Size(10, 50),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Top,
MeasureSize = new Size(10, 150),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Center,
MeasureSize = new Size(10, 50),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Center,
MeasureSize = new Size(10, 150),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Bottom,
MeasureSize = new Size(10, 50),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Bottom,
MeasureSize = new Size(10, 150),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Stretch,
MeasureSize = new Size(10, 50),
},
new TestControl
{
VerticalAlignment = VerticalAlignment.Stretch,
MeasureSize = new Size(10, 150),
},
}
};
target.Measure(new Size(150, 100));
Assert.Equal(new Size(80, 100), target.DesiredSize);
target.Arrange(new Rect(target.DesiredSize));
var bounds = target.Children.Select(x => x.Bounds).ToArray();
Assert.Equal(
new[]
{
new Rect(0, 0, 10, 50),
new Rect(10, 0, 10, 150),
new Rect(20, 25, 10, 50),
new Rect(30, -25, 10, 150),
new Rect(40, 50, 10, 50),
new Rect(50, -50, 10, 150),
new Rect(60, 0, 10, 100),
new Rect(70, 0, 10, 150),
}, bounds);
}
[Theory] [Theory]
[InlineData(Orientation.Horizontal)] [InlineData(Orientation.Horizontal)]
[InlineData(Orientation.Vertical)] [InlineData(Orientation.Vertical)]
@ -185,5 +331,17 @@ namespace Avalonia.Controls.UnitTests
Assert.Equal(sizeWithTwoChildren, sizeWithThreeChildren); Assert.Equal(sizeWithTwoChildren, sizeWithThreeChildren);
} }
private class TestControl : Control
{
public Size MeasureConstraint { get; private set; }
public Size MeasureSize { get; set; }
protected override Size MeasureOverride(Size availableSize)
{
MeasureConstraint = availableSize;
return MeasureSize;
}
}
} }
} }

8
tests/Avalonia.RenderTests/Shapes/PathTests.cs

@ -334,11 +334,7 @@ namespace Avalonia.Direct2D1.RenderTests.Shapes
CompareImages(); CompareImages();
} }
#if AVALONIA_SKIA_SKIP_FAIL
[Fact(Skip = "FIXME")]
#else
[Fact] [Fact]
#endif
public async Task Path_With_PenLineCap() public async Task Path_With_PenLineCap()
{ {
Decorator target = new Decorator Decorator target = new Decorator
@ -351,10 +347,8 @@ namespace Avalonia.Direct2D1.RenderTests.Shapes
StrokeThickness = 10, StrokeThickness = 10,
HorizontalAlignment = HorizontalAlignment.Center, HorizontalAlignment = HorizontalAlignment.Center,
VerticalAlignment = VerticalAlignment.Center, VerticalAlignment = VerticalAlignment.Center,
StrokeDashCap = PenLineCap.Triangle,
StrokeDashArray = new AvaloniaList<double>(3, 1), StrokeDashArray = new AvaloniaList<double>(3, 1),
StrokeStartLineCap = PenLineCap.Round, StrokeLineCap = PenLineCap.Round,
StrokeEndLineCap = PenLineCap.Square,
Data = StreamGeometry.Parse("M 20,20 L 180,180"), Data = StreamGeometry.Parse("M 20,20 L 180,180"),
} }
}; };

3
tests/Avalonia.RenderTests/Shapes/PolylineTests.cs

@ -61,8 +61,7 @@ namespace Avalonia.Direct2D1.RenderTests.Shapes
Points = polylinePoints, Points = polylinePoints,
Stretch = Stretch.Uniform, Stretch = Stretch.Uniform,
StrokeJoin = PenLineJoin.Round, StrokeJoin = PenLineJoin.Round,
StrokeStartLineCap = PenLineCap.Round, StrokeLineCap = PenLineCap.Round,
StrokeEndLineCap = PenLineCap.Round,
StrokeThickness = 10 StrokeThickness = 10
} }
}; };

BIN
tests/TestFiles/Direct2D1/Shapes/Path/Path_With_PenLineCap.expected.png

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

BIN
tests/TestFiles/Skia/Shapes/Path/Path_With_PenLineCap.expected.png

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Loading…
Cancel
Save