DateFieldTests.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. using System.Globalization;
  2. using System.Runtime.InteropServices;
  3. namespace Terminal.Gui.ViewsTests;
  4. public class DateFieldTests
  5. {
  6. [Fact]
  7. [TestDate]
  8. public void Constructors_Defaults ()
  9. {
  10. var df = new DateField ();
  11. df.Layout ();
  12. Assert.Equal (DateTime.MinValue, df.Date);
  13. Assert.Equal (1, df.CursorPosition);
  14. Assert.Equal (new Rectangle (0, 0, 12, 1), df.Frame);
  15. Assert.Equal (" 01/01/0001", df.Text);
  16. DateTime date = DateTime.Now;
  17. df = new DateField (date);
  18. df.Layout ();
  19. Assert.Equal (date, df.Date);
  20. Assert.Equal (1, df.CursorPosition);
  21. Assert.Equal (new Rectangle (0, 0, 12, 1), df.Frame);
  22. Assert.Equal ($" {date.ToString (CultureInfo.InvariantCulture.DateTimeFormat.ShortDatePattern)}", df.Text);
  23. df = new DateField (date) { X = 1, Y = 2 };
  24. df.Layout ();
  25. Assert.Equal (date, df.Date);
  26. Assert.Equal (1, df.CursorPosition);
  27. Assert.Equal (new Rectangle (1, 2, 12, 1), df.Frame);
  28. Assert.Equal ($" {date.ToString (CultureInfo.InvariantCulture.DateTimeFormat.ShortDatePattern)}", df.Text);
  29. }
  30. [Fact]
  31. [TestDate]
  32. [SetupFakeDriver]
  33. public void Copy_Paste ()
  34. {
  35. var df1 = new DateField (DateTime.Parse ("12/12/1971"));
  36. var df2 = new DateField (DateTime.Parse ("12/31/2023"));
  37. // Select all text
  38. Assert.True (df2.NewKeyDownEvent (Key.End.WithShift));
  39. Assert.Equal (1, df2.SelectedStart);
  40. Assert.Equal (10, df2.SelectedLength);
  41. Assert.Equal (11, df2.CursorPosition);
  42. // Copy from df2
  43. Assert.True (df2.NewKeyDownEvent (Key.C.WithCtrl));
  44. // Paste into df1
  45. Assert.True (df1.NewKeyDownEvent (Key.V.WithCtrl));
  46. Assert.Equal (" 12/31/2023", df1.Text);
  47. Assert.Equal (11, df1.CursorPosition);
  48. }
  49. [Fact]
  50. [TestDate]
  51. public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format ()
  52. {
  53. var df = new DateField ();
  54. Assert.Equal (1, df.CursorPosition);
  55. df.CursorPosition = 0;
  56. Assert.Equal (1, df.CursorPosition);
  57. df.CursorPosition = 11;
  58. Assert.Equal (10, df.CursorPosition);
  59. }
  60. [Fact]
  61. [TestDate]
  62. public void CursorPosition_Min_Is_Always_One_Max_Is_Always_Max_Format_After_Selection ()
  63. {
  64. var df = new DateField ();
  65. // Start selection
  66. Assert.True (df.NewKeyDownEvent (Key.CursorLeft.WithShift));
  67. Assert.Equal (1, df.SelectedStart);
  68. Assert.Equal (1, df.SelectedLength);
  69. Assert.Equal (0, df.CursorPosition);
  70. // Without selection
  71. Assert.True (df.NewKeyDownEvent (Key.CursorLeft));
  72. Assert.Equal (-1, df.SelectedStart);
  73. Assert.Equal (0, df.SelectedLength);
  74. Assert.Equal (1, df.CursorPosition);
  75. df.CursorPosition = 10;
  76. Assert.True (df.NewKeyDownEvent (Key.CursorRight.WithShift));
  77. Assert.Equal (10, df.SelectedStart);
  78. Assert.Equal (1, df.SelectedLength);
  79. Assert.Equal (11, df.CursorPosition);
  80. Assert.True (df.NewKeyDownEvent (Key.CursorRight));
  81. Assert.Equal (-1, df.SelectedStart);
  82. Assert.Equal (0, df.SelectedLength);
  83. Assert.Equal (10, df.CursorPosition);
  84. }
  85. [Fact]
  86. [TestDate]
  87. public void Date_Start_From_01_01_0001_And_End_At_12_31_9999 ()
  88. {
  89. var df = new DateField (DateTime.Parse ("01/01/0001"));
  90. Assert.Equal (" 01/01/0001", df.Text);
  91. df.Date = DateTime.Parse ("12/31/9999");
  92. Assert.Equal (" 12/31/9999", df.Text);
  93. }
  94. [Fact]
  95. [TestDate]
  96. public void KeyBindings_Command ()
  97. {
  98. var df = new DateField (DateTime.Parse ("12/12/1971")) { ReadOnly = true };
  99. Assert.True (df.NewKeyDownEvent (Key.Delete));
  100. Assert.Equal (" 12/12/1971", df.Text);
  101. df.ReadOnly = false;
  102. Assert.True (df.NewKeyDownEvent (Key.D.WithCtrl));
  103. Assert.Equal (" 02/12/1971", df.Text);
  104. df.CursorPosition = 4;
  105. df.ReadOnly = true;
  106. Assert.True (df.NewKeyDownEvent (Key.Delete));
  107. Assert.Equal (" 02/12/1971", df.Text);
  108. df.ReadOnly = false;
  109. Assert.True (df.NewKeyDownEvent (Key.Backspace));
  110. Assert.Equal (" 02/02/1971", df.Text);
  111. Assert.True (df.NewKeyDownEvent (Key.Home));
  112. Assert.Equal (1, df.CursorPosition);
  113. Assert.True (df.NewKeyDownEvent (Key.End));
  114. Assert.Equal (10, df.CursorPosition);
  115. Assert.True (df.NewKeyDownEvent (Key.A.WithCtrl));
  116. Assert.Equal (1, df.CursorPosition);
  117. Assert.True (df.NewKeyDownEvent (Key.E.WithCtrl));
  118. Assert.Equal (10, df.CursorPosition);
  119. Assert.True (df.NewKeyDownEvent (Key.CursorLeft));
  120. Assert.Equal (9, df.CursorPosition);
  121. Assert.True (df.NewKeyDownEvent (Key.CursorRight));
  122. Assert.Equal (10, df.CursorPosition);
  123. // Non-numerics are ignored
  124. Assert.False (df.NewKeyDownEvent (Key.A));
  125. df.ReadOnly = true;
  126. df.CursorPosition = 1;
  127. Assert.True (df.NewKeyDownEvent (Key.D1));
  128. Assert.Equal (" 02/02/1971", df.Text);
  129. df.ReadOnly = false;
  130. Assert.True (df.NewKeyDownEvent (Key.D1));
  131. Assert.Equal (" 12/02/1971", df.Text);
  132. Assert.Equal (2, df.CursorPosition);
  133. #if UNIX_KEY_BINDINGS
  134. Assert.True (df.NewKeyDownEvent (Key.D.WithAlt));
  135. Assert.Equal (" 10/02/1971", df.Text);
  136. #endif
  137. }
  138. [Fact]
  139. [TestDate]
  140. public void Typing_With_Selection_Normalize_Format ()
  141. {
  142. var df = new DateField (DateTime.Parse ("12/12/1971"))
  143. {
  144. // Start selection at before the first separator /
  145. CursorPosition = 2
  146. };
  147. // Now select the separator /
  148. Assert.True (df.NewKeyDownEvent (Key.CursorRight.WithShift));
  149. Assert.Equal (2, df.SelectedStart);
  150. Assert.Equal (1, df.SelectedLength);
  151. Assert.Equal (3, df.CursorPosition);
  152. // Type 3 over the separator
  153. Assert.True (df.NewKeyDownEvent (Key.D3));
  154. // The format was normalized and replaced again with /
  155. Assert.Equal (" 12/12/1971", df.Text);
  156. Assert.Equal (4, df.CursorPosition);
  157. }
  158. [Fact]
  159. public void Using_All_Culture_StandardizeDateFormat ()
  160. {
  161. // BUGBUG: This is a workaround for the issue with the date separator in macOS. See https://github.com/gui-cs/Terminal.Gui/issues/3592
  162. if (RuntimeInformation.IsOSPlatform (OSPlatform.OSX))
  163. {
  164. return;
  165. }
  166. CultureInfo cultureBackup = CultureInfo.CurrentCulture;
  167. DateTime date = DateTime.Parse ("1/1/1971");
  168. foreach (CultureInfo culture in CultureInfo.GetCultures (CultureTypes.AllCultures))
  169. {
  170. CultureInfo.CurrentCulture = culture;
  171. string separator = culture.DateTimeFormat.DateSeparator.Trim ();
  172. if (separator.Length > 1 && separator.Contains ('\u200f'))
  173. {
  174. separator = separator.Replace ("\u200f", "");
  175. }
  176. else if (culture.Name == "ar-SA" && RuntimeInformation.IsOSPlatform (OSPlatform.OSX))
  177. {
  178. separator = " ";
  179. }
  180. string format = culture.DateTimeFormat.ShortDatePattern;
  181. var df = new DateField (date);
  182. if ((!culture.TextInfo.IsRightToLeft || (culture.TextInfo.IsRightToLeft && !df.Text.Contains ('\u200f')))
  183. && (format.StartsWith ('d') || format.StartsWith ('M')))
  184. {
  185. switch (culture.Name)
  186. {
  187. case "ar-SA":
  188. Assert.Equal ($" 04{separator}11{separator}1390", df.Text);
  189. break;
  190. case "en-SA" when RuntimeInformation.IsOSPlatform (OSPlatform.OSX):
  191. Assert.Equal ($" 04{separator}11{separator}1390", df.Text);
  192. break;
  193. case "en-TH" when RuntimeInformation.IsOSPlatform (OSPlatform.OSX):
  194. Assert.Equal ($" 01{separator}01{separator}2514", df.Text);
  195. break;
  196. case "th":
  197. case "th-TH":
  198. Assert.Equal ($" 01{separator}01{separator}2514", df.Text);
  199. break;
  200. default:
  201. Assert.Equal ($" 01{separator}01{separator}1971", df.Text);
  202. break;
  203. }
  204. }
  205. else if (culture.TextInfo.IsRightToLeft)
  206. {
  207. if (df.Text.Contains ('\u200f'))
  208. {
  209. // It's a Unicode Character (U+200F) - Right-to-Left Mark (RLM)
  210. Assert.True (df.Text.Contains ('\u200f'));
  211. switch (culture.Name)
  212. {
  213. case "ar-SA":
  214. Assert.Equal ($" 04‏{separator}11‏{separator}1390", df.Text);
  215. break;
  216. default:
  217. Assert.Equal ($" 01‏{separator}01‏{separator}1971", df.Text);
  218. break;
  219. }
  220. }
  221. else
  222. {
  223. switch (culture.Name)
  224. {
  225. case "ckb-IR":
  226. case "fa":
  227. case "fa-AF":
  228. case "fa-IR":
  229. case "lrc":
  230. case "lrc-IR":
  231. case "mzn":
  232. case "mzn-IR":
  233. case "ps":
  234. case "ps-AF":
  235. case "uz-Arab":
  236. case "uz-Arab-AF":
  237. Assert.Equal ($" 1349{separator}10{separator}11", df.Text);
  238. break;
  239. default:
  240. Assert.Equal ($" 1971{separator}01{separator}01", df.Text);
  241. break;
  242. }
  243. }
  244. }
  245. else
  246. {
  247. switch (culture.Name)
  248. {
  249. default:
  250. Assert.Equal ($" 1971{separator}01{separator}01", df.Text);
  251. break;
  252. }
  253. }
  254. }
  255. CultureInfo.CurrentCulture = cultureBackup;
  256. }
  257. [Fact]
  258. public void Using_Pt_Culture ()
  259. {
  260. CultureInfo cultureBackup = CultureInfo.CurrentCulture;
  261. CultureInfo.CurrentCulture = new CultureInfo ("pt-PT");
  262. var df = new DateField (DateTime.Parse ("12/12/1971"))
  263. {
  264. // Move to the first 2
  265. CursorPosition = 2
  266. };
  267. // Type 3 over the separator
  268. Assert.True (df.NewKeyDownEvent (Key.D3));
  269. // If InvariantCulture was used this will fail but not with PT culture
  270. Assert.Equal (" 13/12/1971", df.Text);
  271. Assert.Equal ("13/12/1971", df.Date.ToString (CultureInfo.CurrentCulture.DateTimeFormat.ShortDatePattern));
  272. Assert.Equal (4, df.CursorPosition);
  273. CultureInfo.CurrentCulture = cultureBackup;
  274. }
  275. }