Fixes #3157. DateField is validating wrong the date on different culture than us-US. (#3165)

* Remove the old short two digits year and done some cleanup.

* Fixes #3160. TextField doesn't update correctly the CursorPosition on Paste.

* Using TextChanging instead of TextChanged event.

* Removes old IsShortFormat.

* Removing unnecessary private fields and done code cleanup.

* Removes unnecessary GetShortDatePattern method,

* Fix AdjCursorPosition method.

* Create TestDateAttribute.

* Reduces indentation and removes unused using.

* Remove location from constructors parameters.
This commit is contained in:
BDisp
2024-01-14 22:54:07 +00:00
committed by GitHub
parent 7fe95cb9c7
commit 43b889d1b1
8 changed files with 274 additions and 335 deletions

View File

@@ -20,16 +20,9 @@ namespace Terminal.Gui;
/// </remarks>
public class DateField : TextField {
DateTime _date;
bool _isShort;
int _longFieldLen = 10;
int _shortFieldLen = 8;
int _fieldLen = 10;
string _sepChar;
string _longFormat;
string _shortFormat;
int _fieldLen => _isShort ? _shortFieldLen : _longFieldLen;
string _format => _isShort ? _shortFormat : _longFormat;
string _format;
/// <summary>
/// DateChanged event, raised when the <see cref="Date"/> property has changed.
@@ -42,15 +35,6 @@ public class DateField : TextField {
/// </remarks>
public event EventHandler<DateTimeEventArgs<DateTime>> DateChanged;
/// <summary>
/// Initializes a new instance of <see cref="DateField"/> using <see cref="LayoutStyle.Absolute"/> layout.
/// </summary>
/// <param name="x">The x coordinate.</param>
/// <param name="y">The y coordinate.</param>
/// <param name="date">Initial date contents.</param>
/// <param name="isShort">If true, shows only two digits for the year.</param>
public DateField (int x, int y, DateTime date, bool isShort = false) : base (x, y, isShort ? 10 : 12, "") => SetInitialProperties (date, isShort);
/// <summary>
/// Initializes a new instance of <see cref="DateField"/> using <see cref="LayoutStyle.Computed"/> layout.
/// </summary>
@@ -66,16 +50,14 @@ public class DateField : TextField {
SetInitialProperties (date);
}
void SetInitialProperties (DateTime date, bool isShort = false)
void SetInitialProperties (DateTime date)
{
var cultureInfo = CultureInfo.CurrentCulture;
_sepChar = cultureInfo.DateTimeFormat.DateSeparator;
_longFormat = GetLongFormat (cultureInfo.DateTimeFormat.ShortDatePattern);
_shortFormat = GetShortFormat (_longFormat);
this._isShort = isShort;
_format = $" {cultureInfo.DateTimeFormat.ShortDatePattern}";
Date = date;
CursorPosition = 1;
TextChanged += DateField_Changed;
TextChanging += DateField_Changing;
// Things this view knows how to do
AddCommand (Command.DeleteCharRight, () => {
@@ -109,7 +91,6 @@ public class DateField : TextField {
KeyBindings.Add (Key.CursorRight, Command.Right);
KeyBindings.Add (Key.F.WithCtrl, Command.Right);
}
/// <inheritdoc />
@@ -127,45 +108,33 @@ public class DateField : TextField {
return false;
}
void DateField_Changed (object sender, TextChangedEventArgs e)
void DateField_Changing (object sender, TextChangingEventArgs e)
{
try {
var date = GetInvarianteDate (Text, _isShort);
if ($" {date}" != Text) {
Text = $" {date}";
var cultureInfo = CultureInfo.CurrentCulture;
DateTimeFormatInfo ccFmt = cultureInfo.DateTimeFormat;
int spaces = 0;
for (int i = 0; i < e.NewText.Length; i++) {
if (e.NewText [i] == ' ') {
spaces++;
} else {
break;
}
}
if (_isShort) {
date = GetInvarianteDate (Text, false);
}
if (!DateTime.TryParseExact (date, GetInvarianteFormat (), CultureInfo.CurrentCulture, DateTimeStyles.None, out var result)) {
Text = e.OldValue;
spaces += _fieldLen;
string trimedText = e.NewText [..spaces];
spaces -= _fieldLen;
trimedText = trimedText.Replace (new string (' ', spaces), " ");
var date = Convert.ToDateTime (trimedText, ccFmt).ToString (ccFmt.ShortDatePattern);
if ($" {date}" != e.NewText) {
e.NewText = $" {date}";
}
AdjCursorPosition (CursorPosition, true);
} catch (Exception) {
Text = e.OldValue;
e.Cancel = true;
}
}
string GetInvarianteFormat () => $"MM{_sepChar}dd{_sepChar}yyyy";
string GetLongFormat (string lf)
{
string [] frm = lf.Split (_sepChar);
for (int i = 0; i < frm.Length; i++) {
if (frm [i].Contains ("M") && frm [i].GetRuneCount () < 2) {
lf = lf.Replace ("M", "MM");
}
if (frm [i].Contains ("d") && frm [i].GetRuneCount () < 2) {
lf = lf.Replace ("d", "dd");
}
if (frm [i].Contains ("y") && frm [i].GetRuneCount () < 4) {
lf = lf.Replace ("yy", "yyyy");
}
}
return $" {lf}";
}
string GetShortFormat (string lf) => lf.Replace ("yyyy", "yy");
/// <summary>
/// Gets or sets the date of the <see cref="DateField"/>.
/// </summary>
@@ -188,28 +157,6 @@ public class DateField : TextField {
}
}
/// <summary>
/// Get or set the date format for the widget.
/// </summary>
public bool IsShortFormat {
get => _isShort;
set {
_isShort = value;
if (_isShort) {
Width = 10;
} else {
Width = 12;
}
bool ro = ReadOnly;
if (ro) {
ReadOnly = false;
}
SetText (Text);
ReadOnly = ro;
SetNeedsDisplay ();
}
}
/// <inheritdoc/>
public override int CursorPosition {
get => base.CursorPosition;
@@ -230,7 +177,7 @@ public class DateField : TextField {
var newText = text.GetRange (0, CursorPosition);
newText.Add (key);
if (CursorPosition < _fieldLen) {
newText = newText.Concat (text.GetRange (CursorPosition + 1, text.Count - (CursorPosition + 1))).ToList ();
newText = [.. newText, .. text.GetRange (CursorPosition + 1, text.Count - (CursorPosition + 1))];
}
return SetText (StringExtensions.ToString (newText));
}
@@ -310,18 +257,12 @@ public class DateField : TextField {
{
string date = " ";
for (int i = 0; i < fm.Length; i++) {
if (fm [i].Contains ("M")) {
if (fm [i].Contains ('M')) {
date += $"{month,2:00}";
} else if (fm [i].Contains ("d")) {
} else if (fm [i].Contains ('d')) {
date += $"{day,2:00}";
} else {
if (_isShort && year.ToString ().Length == 4) {
date += $"{year.ToString ().Substring (2, 2)}";
} else if (_isShort) {
date += $"{year,2:00}";
} else {
date += $"{year,4:0000}";
}
date += $"{year,4:0000}";
}
if (i < 2) {
date += $"{_sepChar}";
@@ -330,40 +271,7 @@ public class DateField : TextField {
return date;
}
string GetInvarianteDate (string text, bool isShort)
{
string [] vals = text.Split (_sepChar);
string [] frm = (isShort ? $"MM{_sepChar}dd{_sepChar}yy" : GetInvarianteFormat ()).Split (_sepChar);
string [] date = { null, null, null };
for (int i = 0; i < frm.Length; i++) {
if (frm [i].Contains ("M")) {
date [0] = vals [i].Trim ();
} else if (frm [i].Contains ("d")) {
date [1] = vals [i].Trim ();
} else {
string yearString;
if (isShort && vals [i].Length > 2) {
yearString = vals [i].Substring (0, 2);
} else if (!isShort && vals [i].Length > 4) {
yearString = vals [i].Substring (0, 4);
} else {
yearString = vals [i].Trim ();
}
var year = int.Parse (yearString);
if (isShort && year.ToString ().Length == 4) {
date [2] = year.ToString ().Substring (2, 2);
} else if (isShort) {
date [2] = year.ToString ();
} else {
date [2] = $"{year,4:0000}";
}
}
}
return $"{date [0]}{_sepChar}{date [1]}{_sepChar}{date [2]}";
}
int GetFormatIndex (string [] fm, string t)
static int GetFormatIndex (string [] fm, string t)
{
int idx = -1;
for (int i = 0; i < fm.Length; i++) {
@@ -381,9 +289,8 @@ public class DateField : TextField {
CursorPosition = _fieldLen;
return;
}
if (Text [++CursorPosition] == _sepChar.ToCharArray () [0]) {
CursorPosition++;
}
CursorPosition++;
AdjCursorPosition (CursorPosition);
}
void DecCursorPosition ()
@@ -392,15 +299,29 @@ public class DateField : TextField {
CursorPosition = 1;
return;
}
if (Text [--CursorPosition] == _sepChar.ToCharArray () [0]) {
CursorPosition--;
}
CursorPosition--;
AdjCursorPosition (CursorPosition, false);
}
void AdjCursorPosition ()
void AdjCursorPosition (int point, bool increment = true)
{
if (Text [CursorPosition] == _sepChar.ToCharArray () [0]) {
CursorPosition++;
var newPoint = point;
if (point > _fieldLen) {
newPoint = _fieldLen;
}
if (point < 1) {
newPoint = 1;
}
if (newPoint != point) {
CursorPosition = newPoint;
}
while (Text [CursorPosition] == _sepChar [0]) {
if (increment) {
CursorPosition++;
} else {
CursorPosition--;
}
}
}
@@ -461,23 +382,13 @@ public class DateField : TextField {
/// <inheritdoc/>
public override bool MouseEvent (MouseEvent ev)
{
if (!ev.Flags.HasFlag (MouseFlags.Button1Clicked)) {
return false;
}
if (!HasFocus) {
SetFocus ();
}
var result = base.MouseEvent (ev);
int point = ev.X;
if (point > _fieldLen) {
point = _fieldLen;
if (result && SelectedLength == 0 && ev.Flags.HasFlag (MouseFlags.Button1Pressed)) {
int point = ev.X;
AdjCursorPosition (point, true);
}
if (point < 1) {
point = 1;
}
CursorPosition = point;
AdjCursorPosition ();
return true;
return result;
}
/// <summary>

View File

@@ -77,8 +77,7 @@ public class DatePicker : View {
X = Pos.Right (_dateLabel),
Y = 0,
Width = Dim.Fill (1),
Height = 1,
IsShortFormat = false
Height = 1
};
_calendar = new TableView () {

View File

@@ -1375,11 +1375,11 @@ public class TextField : View {
cbTxt +
StringExtensions.ToString (_text.GetRange (selStart + SelectedLength, _text.Count - (selStart + SelectedLength)));
_cursorPosition = selStart + cbTxt.GetRuneCount ();
ClearAllSelection ();
SetNeedsDisplay ();
Adjust ();
}
_cursorPosition = Math.Min (selStart + cbTxt.GetRuneCount (), _text.Count);
ClearAllSelection ();
SetNeedsDisplay ();
Adjust ();
}
/// <summary>
/// Virtual method that invoke the <see cref="TextChanging"/> event if it's defined.

View File

@@ -161,13 +161,12 @@ public class Text : Scenario {
};
Win.Add (labelMirroringHexEditor);
var dateField = new DateField (DateTime.Now) {
X = 1,
Y = Pos.Bottom (hexEditor) + 1,
Width = 20,
IsShortFormat = false
};
Win.Add (dateField);
var dateField = new DateField (System.DateTime.Now) {
X = 1,
Y = Pos.Bottom (hexEditor) + 1,
Width = 20
};
Win.Add (dateField);
var labelMirroringDateField = new Label (dateField.Text) {
X = Pos.Right (dateField) + 1,

View File

@@ -35,7 +35,6 @@ namespace UICatalog.Scenarios {
var shortDate = new DateField (DateTime.Now) {
X = Pos.Center (),
Y = Pos.Bottom (shortTime) + 1,
IsShortFormat = true,
ReadOnly = true,
};
shortDate.DateChanged += DateChanged;
@@ -44,8 +43,7 @@ namespace UICatalog.Scenarios {
var longDate = new DateField (DateTime.Now) {
X = Pos.Center (),
Y = Pos.Bottom (shortDate) + 1,
IsShortFormat = false,
ReadOnly = true,
ReadOnly = false,
};
longDate.DateChanged += DateChanged;
Win.Add (longDate);
@@ -111,9 +109,6 @@ namespace UICatalog.Scenarios {
longDate.ReadOnly = !longDate.ReadOnly;
shortDate.ReadOnly = !shortDate.ReadOnly;
longDate.IsShortFormat = !longDate.IsShortFormat;
shortDate.IsShortFormat = !shortDate.IsShortFormat;
};
Win.Add (swapButton);
}

View File

@@ -147,6 +147,28 @@ public class SetupFakeDriverAttribute : BeforeAfterTestAttribute {
}
}
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class TestDateAttribute : Xunit.Sdk.BeforeAfterTestAttribute
{
CultureInfo _currentCulture = CultureInfo.CurrentCulture;
public TestDateAttribute()
{
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
}
public override void Before(MethodInfo methodUnderTest)
{
Assert.Equal(CultureInfo.CurrentCulture, CultureInfo.InvariantCulture);
}
public override void After(MethodInfo methodUnderTest)
{
CultureInfo.CurrentCulture = _currentCulture;
Assert.Equal(CultureInfo.CurrentCulture, _currentCulture);
}
}
partial class TestHelpers {
[GeneratedRegex ("\\s+$", RegexOptions.Multiline)]
private static partial Regex TrailingWhiteSpaceRegEx ();

View File

@@ -1,185 +1,173 @@
using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Xunit;
namespace Terminal.Gui.ViewsTests {
public class DateFieldTests {
[Fact]
public void Constructors_Defaults ()
{
var df = new DateField ();
Assert.False (df.IsShortFormat);
Assert.Equal (DateTime.MinValue, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (0, 0, 12, 1), df.Frame);
namespace Terminal.Gui.ViewsTests;
public class DateFieldTests {
[Fact, TestDate]
public void Constructors_Defaults ()
{
var df = new DateField ();
Assert.Equal (DateTime.MinValue, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (0, 0, 12, 1), df.Frame);
Assert.Equal (" 01/01/0001", df.Text);
var date = DateTime.Now;
df = new DateField (date);
Assert.False (df.IsShortFormat);
Assert.Equal (date, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (0, 0, 12, 1), df.Frame);
var date = DateTime.Now;
df = new DateField (date);
Assert.Equal (date, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (0, 0, 12, 1), df.Frame);
Assert.Equal ($" {date.ToString (CultureInfo.InvariantCulture.DateTimeFormat.ShortDatePattern)}", df.Text);
df = new DateField (1, 2, date);
Assert.False (df.IsShortFormat);
Assert.Equal (date, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (1, 2, 12, 1), df.Frame);
df = new DateField (date) { X = 1, Y = 2 };
Assert.Equal (date, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (1, 2, 12, 1), df.Frame);
Assert.Equal ($" {date.ToString (CultureInfo.InvariantCulture.DateTimeFormat.ShortDatePattern)}", df.Text);
}
df = new DateField (3, 4, date, true);
Assert.True (df.IsShortFormat);
Assert.Equal (date, df.Date);
Assert.Equal (1, df.CursorPosition);
Assert.Equal (new Rect (3, 4, 10, 1), df.Frame);
[Fact, TestDate]
public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format ()
{
var df = new DateField ();
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 0;
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 11;
Assert.Equal (10, df.CursorPosition);
}
df.IsShortFormat = false;
Assert.Equal (new Rect (3, 4, 12, 1), df.Frame);
Assert.Equal (12, df.Width);
}
[Fact, TestDate]
public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format_After_Selection ()
{
var df = new DateField ();
// Start selection
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft | KeyCode.ShiftMask)));
Assert.Equal (1, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (0, df.CursorPosition);
// Without selection
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft)));
Assert.Equal (-1, df.SelectedStart);
Assert.Equal (0, df.SelectedLength);
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 10;
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight | KeyCode.ShiftMask)));
Assert.Equal (10, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (11, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight)));
Assert.Equal (-1, df.SelectedStart);
Assert.Equal (0, df.SelectedLength);
Assert.Equal (10, df.CursorPosition);
}
[Fact]
public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format ()
{
var df = new DateField ();
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 0;
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 11;
Assert.Equal (10, df.CursorPosition);
df.IsShortFormat = true;
df.CursorPosition = 0;
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 9;
Assert.Equal (8, df.CursorPosition);
}
[Fact, TestDate]
public void KeyBindings_Command ()
{
DateField df = new DateField (DateTime.Parse ("12/12/1971")) {
ReadOnly = true
};
Assert.True (df.NewKeyDownEvent (new (KeyCode.Delete)));
Assert.Equal (" 12/12/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D | KeyCode.CtrlMask)));
Assert.Equal (" 02/12/1971", df.Text);
df.CursorPosition = 4;
df.ReadOnly = true;
Assert.True (df.NewKeyDownEvent (new (KeyCode.Delete)));
Assert.Equal (" 02/12/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.Backspace)));
Assert.Equal (" 02/02/1971", df.Text);
Assert.True (df.NewKeyDownEvent (new (KeyCode.Home)));
Assert.Equal (1, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.End)));
Assert.Equal (10, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.A | KeyCode.CtrlMask)));
Assert.Equal (1, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.E | KeyCode.CtrlMask)));
Assert.Equal (10, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft)));
Assert.Equal (9, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight)));
Assert.Equal (10, df.CursorPosition);
// Non-numerics are ignored
Assert.False (df.NewKeyDownEvent (new (KeyCode.A)));
df.ReadOnly = true;
df.CursorPosition = 1;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D1)));
Assert.Equal (" 02/02/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D1)));
Assert.Equal (" 12/02/1971", df.Text);
Assert.Equal (2, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.D | KeyCode.AltMask)));
Assert.Equal (" 10/02/1971", df.Text);
}
[Fact]
public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format_After_Selection ()
{
var df = new DateField ();
// Start selection
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft | KeyCode.ShiftMask)));
Assert.Equal (1, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (0, df.CursorPosition);
// Without selection
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft)));
Assert.Equal (-1, df.SelectedStart);
Assert.Equal (0, df.SelectedLength);
Assert.Equal (1, df.CursorPosition);
df.CursorPosition = 10;
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight | KeyCode.ShiftMask)));
Assert.Equal (10, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (11, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight)));
Assert.Equal (-1, df.SelectedStart);
Assert.Equal (0, df.SelectedLength);
Assert.Equal (10, df.CursorPosition);
}
[Fact]
public void KeyBindings_Command ()
{
CultureInfo cultureBackup = CultureInfo.CurrentCulture;
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
DateField df = new DateField (DateTime.Parse ("12/12/1971"));
df.ReadOnly = true;
Assert.True (df.NewKeyDownEvent (new (KeyCode.Delete)));
Assert.Equal (" 12/12/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D | KeyCode.CtrlMask)));
Assert.Equal (" 02/12/1971", df.Text);
df.CursorPosition = 4;
df.ReadOnly = true;
Assert.True (df.NewKeyDownEvent (new (KeyCode.Delete)));
Assert.Equal (" 02/12/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.Backspace)));
Assert.Equal (" 02/02/1971", df.Text);
Assert.True (df.NewKeyDownEvent (new (KeyCode.Home)));
Assert.Equal (1, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.End)));
Assert.Equal (10, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.A | KeyCode.CtrlMask)));
Assert.Equal (1, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.E | KeyCode.CtrlMask)));
Assert.Equal (10, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorLeft)));
Assert.Equal (9, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight)));
Assert.Equal (10, df.CursorPosition);
// Non-numerics are ignored
Assert.False (df.NewKeyDownEvent (new (KeyCode.A)));
df.ReadOnly = true;
df.CursorPosition = 1;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D1)));
Assert.Equal (" 02/02/1971", df.Text);
df.ReadOnly = false;
Assert.True (df.NewKeyDownEvent (new (KeyCode.D1)));
Assert.Equal (" 12/02/1971", df.Text);
Assert.Equal (2, df.CursorPosition);
Assert.True (df.NewKeyDownEvent (new (KeyCode.D | KeyCode.AltMask)));
Assert.Equal (" 10/02/1971", df.Text);
CultureInfo.CurrentCulture = cultureBackup;
}
[Fact]
public void Typing_With_Selection_Normalize_Format ()
{
CultureInfo cultureBackup = CultureInfo.CurrentCulture;
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
DateField df = new DateField (DateTime.Parse ("12/12/1971"));
[Fact, TestDate]
public void Typing_With_Selection_Normalize_Format ()
{
DateField df = new DateField (DateTime.Parse ("12/12/1971")) {
// Start selection at before the first separator /
df.CursorPosition = 2;
// Now select the separator /
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight | KeyCode.ShiftMask)));
Assert.Equal (2, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (3, df.CursorPosition);
// Type 3 over the separator
Assert.True (df.NewKeyDownEvent (new (KeyCode.D3)));
// The format was normalized and replaced again with /
Assert.Equal (" 12/12/1971", df.Text);
Assert.Equal (4, df.CursorPosition);
CultureInfo.CurrentCulture = cultureBackup;
}
CursorPosition = 2
};
// Now select the separator /
Assert.True (df.NewKeyDownEvent (new (KeyCode.CursorRight | KeyCode.ShiftMask)));
Assert.Equal (2, df.SelectedStart);
Assert.Equal (1, df.SelectedLength);
Assert.Equal (3, df.CursorPosition);
// Type 3 over the separator
Assert.True (df.NewKeyDownEvent (new (KeyCode.D3)));
// The format was normalized and replaced again with /
Assert.Equal (" 12/12/1971", df.Text);
Assert.Equal (4, df.CursorPosition);
}
[Fact, AutoInitShutdown]
public void Copy_Paste ()
{
CultureInfo cultureBackup = CultureInfo.CurrentCulture;
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
DateField df1 = new DateField (DateTime.Parse ("12/12/1971"));
DateField df2 = new DateField (DateTime.Parse ("12/31/2023"));
// Select all text
Assert.True (df2.NewKeyDownEvent (new (KeyCode.End | KeyCode.ShiftMask)));
Assert.Equal (1, df2.SelectedStart);
Assert.Equal (10, df2.SelectedLength);
Assert.Equal (11, df2.CursorPosition);
// Copy from df2
Assert.True (df2.NewKeyDownEvent (new (KeyCode.C | KeyCode.CtrlMask)));
// Paste into df1
Assert.True (df1.NewKeyDownEvent (new (KeyCode.V | KeyCode.CtrlMask)));
Assert.Equal (" 12/31/2023", df1.Text);
Assert.Equal (11, df1.CursorPosition);
CultureInfo.CurrentCulture = cultureBackup;
}
[Fact, TestDate, AutoInitShutdown]
public void Copy_Paste ()
{
DateField df1 = new DateField (DateTime.Parse ("12/12/1971"));
DateField df2 = new DateField (DateTime.Parse ("12/31/2023"));
// Select all text
Assert.True (df2.NewKeyDownEvent (new (KeyCode.End | KeyCode.ShiftMask)));
Assert.Equal (1, df2.SelectedStart);
Assert.Equal (10, df2.SelectedLength);
Assert.Equal (11, df2.CursorPosition);
// Copy from df2
Assert.True (df2.NewKeyDownEvent (new (KeyCode.C | KeyCode.CtrlMask)));
// Paste into df1
Assert.True (df1.NewKeyDownEvent (new (KeyCode.V | KeyCode.CtrlMask)));
Assert.Equal (" 12/31/2023", df1.Text);
Assert.Equal (11, df1.CursorPosition);
}
[Fact]
public void Date_Start_From_01_01_0001_And_End_At_12_31_9999 ()
{
CultureInfo cultureBackup = CultureInfo.CurrentCulture;
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
DateField df = new DateField (DateTime.Parse ("01/01/0001"));
Assert.Equal (" 01/01/0001", df.Text);
df.Date = DateTime.Parse ("12/31/9999");
Assert.Equal (" 12/31/9999", df.Text);
CultureInfo.CurrentCulture = cultureBackup;
}
[Fact, TestDate]
public void Date_Start_From_01_01_0001_And_End_At_12_31_9999 ()
{
DateField df = new DateField (DateTime.Parse ("01/01/0001"));
Assert.Equal (" 01/01/0001", df.Text);
df.Date = DateTime.Parse ("12/31/9999");
Assert.Equal (" 12/31/9999", df.Text);
}
[Fact]
public void Using_Pt_Culture ()
{
CultureInfo cultureBackup = CultureInfo.CurrentCulture;
CultureInfo.CurrentCulture = new CultureInfo ("pt-PT");
DateField df = new DateField (DateTime.Parse ("12/12/1971")) {
// Move to the first 2
CursorPosition = 2
};
// Type 3 over the separator
Assert.True (df.NewKeyDownEvent (new (KeyCode.D3)));
// If InvariantCulture was used this will fail but not with PT culture
Assert.Equal (" 13/12/1971", df.Text);
Assert.Equal ("13/12/1971", df.Date.ToString (CultureInfo.CurrentCulture.DateTimeFormat.ShortDatePattern));
Assert.Equal (4, df.CursorPosition);
CultureInfo.CurrentCulture = cultureBackup;
}
}

View File

@@ -1634,4 +1634,29 @@ Les Miśerables", output);
_textField.Paste ();
Assert.Equal ("TextField with some more test text. Unicode shouldn't 𝔹A𝔽!", _textField.Text);
}
[Fact, TextFieldTestsAutoInitShutdown]
public void Copy_Paste_Text_Changing_Updates_Cursor_Position ()
{
_textField.TextChanging += _textField_TextChanging;
void _textField_TextChanging (object sender, TextChangingEventArgs e)
{
if (e.NewText.GetRuneCount () > 11) {
e.NewText = e.NewText [..11];
}
}
Assert.Equal (32, _textField.CursorPosition);
_textField.SelectAll ();
_textField.Cut ();
Assert.Equal ("TAB to jump between text fields.", Application.Driver.Clipboard.GetClipboardData ());
Assert.Equal (string.Empty, _textField.Text);
Assert.Equal (0, _textField.CursorPosition);
_textField.Paste ();
Assert.Equal ("TAB to jump", _textField.Text);
Assert.Equal (11, _textField.CursorPosition);
_textField.TextChanging -= _textField_TextChanging;
}
}