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.
412 lines
13 KiB
412 lines
13 KiB
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Avalonia.Automation.Peers;
|
|
using Avalonia.Automation.Provider;
|
|
using Avalonia.Controls;
|
|
using Avalonia.Win32.Automation;
|
|
using Avalonia.Win32.Interop.Automation;
|
|
using Xunit;
|
|
using AAP = Avalonia.Automation.Provider;
|
|
|
|
namespace Avalonia.Win32.UnitTests.Automation;
|
|
|
|
public class AutomationTextRangeTests
|
|
{
|
|
public class Move : AutomationTextRangeTests
|
|
{
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Right_One_Char(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 1);
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[0][1].ToString(), range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Does_Not_Move_Right_From_Last_Char(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.Text.Length - 1;
|
|
var range = new AutomationTextRange(node, start, start + 1);
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(0, result);
|
|
Assert.Equal(peer.Text[^1].ToString(), range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Degenerate_Right_One_Char(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 0);
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(1, range.Start);
|
|
Assert.Equal(1, range.End);
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Right_To_Newline(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 1);
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[0][1].ToString(), range.GetText(-1));
|
|
}
|
|
|
|
[Theory(Skip = "Not yet implemented")]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Right_One_Surrogate_Pair(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(6).Start;
|
|
var range = new AutomationTextRange(node, start, start + 2);
|
|
|
|
Assert.Equal("🌉", range.GetText(-1));
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal("U", range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Left_One_Char(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 1);
|
|
|
|
var result = range.Move(TextUnit.Character, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[0][1].ToString(), range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Does_Not_Move_Left_From_First_Char(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 1);
|
|
|
|
var result = range.Move(TextUnit.Character, -1);
|
|
|
|
Assert.Equal(0, result);
|
|
Assert.Equal(peer.Text[0].ToString(), range.GetText(-1));
|
|
}
|
|
|
|
[Theory(Skip = "Not yet implemented")]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Left_One_Surrogate_Pair(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(6).Start + 2;
|
|
var range = new AutomationTextRange(node, start, start + 1);
|
|
|
|
var result = range.Move(TextUnit.Character, -1);
|
|
|
|
Assert.Equal(-1, result);
|
|
Assert.Equal("🌉", range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Right_One_Word_With_Trailing_Newlines(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 1);
|
|
|
|
var result = range.Move(TextUnit.Word, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
|
|
// From https://docs.microsoft.com/en-us/windows/win32/winauto/uiauto-uiautomationtextunits
|
|
// When TextUnit_Word is used to set the boundary of a text range, the resulting text range
|
|
// should include any word break characters that are present at the end of the word, but
|
|
// before the start of the next word.
|
|
Assert.Equal("text:" + newline + newline, range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Right_One_Word_With_Apostrophe(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(10).Start;
|
|
var range = new AutomationTextRange(node, start, start + 1);
|
|
|
|
var result = range.Move(TextUnit.Word, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
|
|
Assert.Equal("can't ", range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Left_To_Previous_Line_With_Trailing_Newlines(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(2).Start;
|
|
var range = new AutomationTextRange(node, start, start + 1);
|
|
|
|
var result = range.Move(TextUnit.Word, -1);
|
|
|
|
Assert.Equal(-1, result);
|
|
Assert.Equal("text:" + newline + newline, range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Down_To_Empty_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, peer.Lines[0].Length);
|
|
|
|
var result = range.Move(TextUnit.Line, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[1], range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Degenerate_Down_To_Empty_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, 0);
|
|
|
|
var result = range.Move(TextUnit.Line, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[0].Length, range.Start);
|
|
Assert.Equal(range.Start, range.End);
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Down_From_Mid_Line_To_Next_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 2, 4);
|
|
|
|
var result = range.Move(TextUnit.Line, 1);
|
|
|
|
Assert.Equal(1, result);
|
|
Assert.Equal(peer.Lines[1], range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Does_Not_Move_Down_From_Last_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(peer.Lines.Count - 1).Start;
|
|
var range = new AutomationTextRange(node, start, peer.Text.Length);
|
|
|
|
var result = range.Move(TextUnit.Line, 1);
|
|
|
|
Assert.Equal(0, result);
|
|
Assert.Equal(peer.Lines[^1], range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Up_To_Empty_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(2).Start;
|
|
var range = new AutomationTextRange(node, start, start + peer.Lines[2].Length);
|
|
|
|
var result = range.Move(TextUnit.Line, -1);
|
|
|
|
Assert.Equal(-1, result);
|
|
Assert.Equal(peer.Lines[1], range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Degenerate_Up_To_Empty_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(2).Start;
|
|
var range = new AutomationTextRange(node, start, start);
|
|
|
|
var result = range.Move(TextUnit.Line, -1);
|
|
|
|
Assert.Equal(-1, result);
|
|
Assert.Equal(peer.Lines[0].Length, range.Start);
|
|
Assert.Equal(range.Start, range.End);
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Moves_Up_From_Mid_Line_To_Next_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var start = peer.GetLineRange(2).Start + 2;
|
|
var range = new AutomationTextRange(node, start, start + 2);
|
|
|
|
var result = range.Move(TextUnit.Line, -1);
|
|
|
|
Assert.Equal(-1, result);
|
|
Assert.Equal(peer.Lines[1], range.GetText(-1));
|
|
}
|
|
|
|
[Theory]
|
|
[ClassData(typeof(Newlines))]
|
|
public void Does_Not_Move_Up_From_First_Line(string newline)
|
|
{
|
|
var (node, peer) = CreateTestNode(newline: newline);
|
|
var range = new AutomationTextRange(node, 0, peer.Lines[0].Length);
|
|
|
|
var result = range.Move(TextUnit.Line, -1);
|
|
|
|
Assert.Equal(0, result);
|
|
Assert.Equal(peer.Lines[0], range.GetText(-1));
|
|
}
|
|
}
|
|
|
|
public class MoveEndpointByUnit
|
|
{
|
|
[Fact]
|
|
public void Moving_Back_A_Word_Works_With_Only_Symbols()
|
|
{
|
|
var (node, peer) = CreateTestNode(lines: "(___) ___-____");
|
|
var range = new AutomationTextRange(node, 0, 14);
|
|
|
|
range.MoveEndpointByUnit(TextPatternRangeEndpoint.End, TextUnit.Word, -1);
|
|
}
|
|
}
|
|
|
|
private static (AutomationNode, TestPeer) CreateTestNode(string newline)
|
|
{
|
|
var lines = new[]
|
|
{
|
|
"Test text:" + newline,
|
|
newline,
|
|
"Lorem ipsum dolor sit amet, ",
|
|
"consectetur adipiscing elit, ",
|
|
"sed do eiusmod tempor incididunt ",
|
|
"ut labore et dolore magna aliqua." + newline,
|
|
"🌉Ut enim ad minim veniam, quis ",
|
|
"nostrud exercitation ullamco ",
|
|
"laboris nisi ut aliquip ex ea ",
|
|
"commodo consequat." + newline,
|
|
"We can't stop now"
|
|
};
|
|
|
|
return CreateTestNode(lines);
|
|
}
|
|
|
|
private static (AutomationNode, TestPeer) CreateTestNode(params string[] lines)
|
|
{
|
|
var peer = new TestPeer(lines);
|
|
return (new AutomationNode(peer), peer);
|
|
}
|
|
|
|
private class TestPeer : ControlAutomationPeer, AAP.ITextProvider
|
|
{
|
|
private readonly string[] _lines;
|
|
|
|
public TestPeer(string[] lines)
|
|
: base(new Control())
|
|
{
|
|
_lines = lines;
|
|
}
|
|
|
|
public bool IsReadOnly => true;
|
|
public int CaretIndex => 0;
|
|
public TextRange DocumentRange => new(0, _lines.Sum(x => x.Length));
|
|
public IList<string> Lines => _lines;
|
|
public int LineCount => _lines.Length;
|
|
public string PlaceholderText => string.Empty;
|
|
public AAP.SupportedTextSelection SupportedTextSelection { get; }
|
|
public string Text => string.Concat(_lines);
|
|
|
|
public event EventHandler? SelectedRangesChanged;
|
|
public event EventHandler? TextChanged;
|
|
|
|
public IReadOnlyList<Rect> GetBounds(TextRange range)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public int GetLineForIndex(int index)
|
|
{
|
|
var i = 0;
|
|
|
|
for (var line = 0; line < _lines.Length; ++line)
|
|
{
|
|
i += _lines[line].Length;
|
|
if (index < i)
|
|
return line;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public TextRange GetLineRange(int lineIndex)
|
|
{
|
|
var start = _lines.Take(lineIndex).Sum(x => x.Length);
|
|
var end = start + _lines[lineIndex].Length;
|
|
return new TextRange(start, end);
|
|
}
|
|
|
|
public IReadOnlyList<TextRange> GetSelection()
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public string GetText(TextRange range)
|
|
{
|
|
var text = Text;
|
|
var end = Math.Min(range.End, text.Length);
|
|
return text.Substring(range.Start, end - range.Start);
|
|
}
|
|
|
|
public TextRange RangeFromPoint(Point p)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public void ScrollIntoView(TextRange range)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public void Select(TextRange range)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
}
|
|
|
|
private class Newlines : IEnumerable<object[]>
|
|
{
|
|
public IEnumerator<object[]> GetEnumerator()
|
|
{
|
|
yield return new object[] { "\r" };
|
|
yield return new object[] { "\n" };
|
|
yield return new object[] { "\r\n" };
|
|
}
|
|
|
|
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
}
|
|
}
|
|
|