Rename UiChatView -> UiText (the retail UIElement_Text class, RegisterElementClass(0xc) @ acclient_2013_pseudo_c.txt:115655). Factory changes (DatWidgetFactory.cs): - Remove the Type-12 skip (was: no-media -> null, with-media -> UiDatElement). - Add Type 12 -> BuildText() -> UiText in the switch. - BuildText extracts the element's Direct/Normal sprite as BackgroundSprite so any dat-media the element carried keeps rendering under the text. UiText changes (renamed from UiChatView.cs): - BackgroundColor default: (0,0,0,0.35) -> (0,0,0,0) (transparent). An unbound UiText draws nothing; the controller opts in to the translucent bg. - New BackgroundSprite + SpriteResolve: optional dat state-sprite background drawn UNDER DrawFill+text (faithful UIElement_Text media support). ChatWindowController.cs (Task 5 Step 8): - Transcript property: UiChatView -> UiText. - Bind() now uses layout.FindElement(TranscriptId) as UiText (factory-built) instead of manually constructing + AddChild-ing a new UiChatView. - Sets BackgroundColor = (0,0,0,0.35) on the found widget (retail translucent bg). - Removes the tInfo null-check from the early guard (transcript is factory-built; iInfo lookup kept for the input widget which is still manually constructed). - BuildLines: UiChatView.Line -> UiText.Line throughout. Vitals frozen: the Type-12 vitals number elements are meter children and are never recursed by BuildWidget (the `if (w is not UiMeter)` gate), so they are not built as widgets and keep rendering via UiMeter.Label. Vitals fixture vitals_2100006C.json unchanged; LayoutConformanceTests + VitalsBindingTests green. Tests: - UiChatViewTests.cs -> UiTextTests.cs (class: UiTextTests, all UiChatView.* -> UiText.*) - UiChatViewDatFontTests.cs -> UiTextDatFontTests.cs (same) - DatWidgetFactoryTests: delete Type12_StylePrototype_ReturnsNull + DatWidgetFactory_Type12WithMedia_Renders; add Type12_Text_MakesUiText + DatWidgetFactory_Type12_AlwaysMakesUiText. - LayoutImporterTests: BuildFromInfos_Type12Child_IsSkipped_Type3Present updated to assert IsType<UiText> (element is now in tree, transparent, not skipped). Divergence register: AP-37 amended -- removed the "standalone Type-0 text elements skipped / dat-text widget is Plan 2" clause (now shipped as UiText); kept the meter-collapse clause and the vitals-numbers-via-UiMeter.Label clause. AP-38/AP-39/AD-28 file references updated UiChatView.cs -> UiText.cs. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
143 lines
4.9 KiB
C#
143 lines
4.9 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Numerics;
|
|
using AcDream.App.UI;
|
|
|
|
namespace AcDream.App.Tests.UI;
|
|
|
|
public class UiTextTests
|
|
{
|
|
[Fact]
|
|
public void ClampScroll_PinsToZero_WhenContentFitsView()
|
|
{
|
|
// 5 lines of content in a taller view → nothing to scroll, pinned at 0.
|
|
Assert.Equal(0f, UiText.ClampScroll(50f, contentHeight: 80f, viewHeight: 200f));
|
|
Assert.Equal(0f, UiText.ClampScroll(0f, contentHeight: 80f, viewHeight: 200f));
|
|
}
|
|
|
|
[Fact]
|
|
public void ClampScroll_CapsAtContentMinusView_WhenOverflowing()
|
|
{
|
|
// Content 500, view 200 → max scrollback is 300px (oldest line at top).
|
|
Assert.Equal(300f, UiText.ClampScroll(1000f, contentHeight: 500f, viewHeight: 200f));
|
|
Assert.Equal(120f, UiText.ClampScroll(120f, contentHeight: 500f, viewHeight: 200f));
|
|
}
|
|
|
|
[Fact]
|
|
public void ClampScroll_NeverNegative()
|
|
{
|
|
Assert.Equal(0f, UiText.ClampScroll(-50f, contentHeight: 500f, viewHeight: 200f));
|
|
}
|
|
|
|
// ── Char-index hit-testing (x → col) with a synthetic 10px monospace advance ──
|
|
|
|
private static readonly Func<char, float> Mono10 = static _ => 10f;
|
|
|
|
[Fact]
|
|
public void CharIndexAt_ZeroOrNegative_IsColumnZero()
|
|
{
|
|
Assert.Equal(0, UiText.CharIndexAt("hello", Mono10, 0f));
|
|
Assert.Equal(0, UiText.CharIndexAt("hello", Mono10, -5f));
|
|
}
|
|
|
|
[Fact]
|
|
public void CharIndexAt_SnapsToGlyphMidpoint()
|
|
{
|
|
// glyph[0] spans 0..10 (midpoint 5), glyph[1] 10..20 (midpoint 15), ...
|
|
Assert.Equal(0, UiText.CharIndexAt("hello", Mono10, 4f)); // before mid of glyph 0
|
|
Assert.Equal(1, UiText.CharIndexAt("hello", Mono10, 6f)); // past mid of glyph 0
|
|
Assert.Equal(1, UiText.CharIndexAt("hello", Mono10, 14f)); // before mid of glyph 1
|
|
Assert.Equal(2, UiText.CharIndexAt("hello", Mono10, 16f)); // past mid of glyph 1
|
|
}
|
|
|
|
[Fact]
|
|
public void CharIndexAt_PastEnd_IsLength()
|
|
{
|
|
Assert.Equal(5, UiText.CharIndexAt("hello", Mono10, 1000f));
|
|
}
|
|
|
|
[Fact]
|
|
public void CharIndexAt_EmptyString_IsZero()
|
|
{
|
|
Assert.Equal(0, UiText.CharIndexAt("", Mono10, 50f));
|
|
}
|
|
|
|
// ── SelectedText assembly ────────────────────────────────────────────
|
|
|
|
private static IReadOnlyList<UiText.Line> Lines(params string[] texts)
|
|
{
|
|
var list = new List<UiText.Line>(texts.Length);
|
|
foreach (var t in texts)
|
|
list.Add(new UiText.Line(t, new Vector4(1, 1, 1, 1)));
|
|
return list;
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_SingleLine_Substring()
|
|
{
|
|
var lines = Lines("hello world");
|
|
var s = UiText.SelectedText(lines, new UiText.Pos(0, 6), new UiText.Pos(0, 11));
|
|
Assert.Equal("world", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_SingleLine_ReversedAnchorCaret_IsNormalised()
|
|
{
|
|
var lines = Lines("hello world");
|
|
// caret BEFORE anchor — Order() must normalise.
|
|
var s = UiText.SelectedText(lines, new UiText.Pos(0, 11), new UiText.Pos(0, 6));
|
|
Assert.Equal("world", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_SamePosition_IsEmpty()
|
|
{
|
|
var lines = Lines("hello");
|
|
Assert.Equal("", UiText.SelectedText(lines, new UiText.Pos(0, 3), new UiText.Pos(0, 3)));
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_MultiLine_JoinsWithNewline()
|
|
{
|
|
var lines = Lines("first line", "second line", "third line");
|
|
// from col 6 of line 0 ("line") through col 5 of line 2 ("third")
|
|
var s = UiText.SelectedText(lines, new UiText.Pos(0, 6), new UiText.Pos(2, 5));
|
|
Assert.Equal("line\nsecond line\nthird", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_MultiLine_TwoLines_NoMiddle()
|
|
{
|
|
var lines = Lines("alpha", "bravo");
|
|
var s = UiText.SelectedText(lines, new UiText.Pos(0, 2), new UiText.Pos(1, 3));
|
|
Assert.Equal("pha\nbra", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_MultiLine_ReversedAnchorCaret_IsNormalised()
|
|
{
|
|
var lines = Lines("alpha", "bravo");
|
|
// end before start → Order() swaps them.
|
|
var s = UiText.SelectedText(lines, new UiText.Pos(1, 3), new UiText.Pos(0, 2));
|
|
Assert.Equal("pha\nbra", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void SelectedText_EmptyLineList_IsEmpty()
|
|
{
|
|
Assert.Equal("", UiText.SelectedText(Array.Empty<UiText.Line>(),
|
|
new UiText.Pos(0, 0), new UiText.Pos(0, 0)));
|
|
}
|
|
|
|
[Fact]
|
|
public void Order_SortsByLineThenColumn()
|
|
{
|
|
var (s1, e1) = UiText.Order(new UiText.Pos(2, 1), new UiText.Pos(0, 5));
|
|
Assert.Equal(new UiText.Pos(0, 5), s1);
|
|
Assert.Equal(new UiText.Pos(2, 1), e1);
|
|
|
|
var (s2, e2) = UiText.Order(new UiText.Pos(1, 8), new UiText.Pos(1, 2));
|
|
Assert.Equal(new UiText.Pos(1, 2), s2);
|
|
Assert.Equal(new UiText.Pos(1, 8), e2);
|
|
}
|
|
}
|