TabControlTests.cs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.ObjectModel;
  4. using System.Linq;
  5. using Avalonia.Controls.Presenters;
  6. using Avalonia.Controls.Primitives;
  7. using Avalonia.Controls.Selection;
  8. using Avalonia.Controls.Templates;
  9. using Avalonia.Controls.Utils;
  10. using Avalonia.Input;
  11. using Avalonia.LogicalTree;
  12. using Avalonia.Markup.Xaml;
  13. using Avalonia.Styling;
  14. using Avalonia.UnitTests;
  15. using Xunit;
  16. namespace Avalonia.Controls.UnitTests
  17. {
  18. public class TabControlTests
  19. {
  20. [Fact]
  21. public void First_Tab_Should_Be_Selected_By_Default()
  22. {
  23. TabItem selected;
  24. var target = new TabControl
  25. {
  26. Template = TabControlTemplate(),
  27. Items =
  28. {
  29. (selected = new TabItem
  30. {
  31. Name = "first",
  32. Content = "foo",
  33. }),
  34. new TabItem
  35. {
  36. Name = "second",
  37. Content = "bar",
  38. },
  39. }
  40. };
  41. target.ApplyTemplate();
  42. Assert.Equal(0, target.SelectedIndex);
  43. Assert.Equal(selected, target.SelectedItem);
  44. }
  45. [Fact]
  46. public void Pre_Selecting_TabItem_Should_Set_SelectedContent_After_It_Was_Added()
  47. {
  48. const string secondContent = "Second";
  49. var target = new TabControl
  50. {
  51. Template = TabControlTemplate(),
  52. Items =
  53. {
  54. new TabItem { Header = "First"},
  55. new TabItem { Header = "Second", Content = secondContent, IsSelected = true }
  56. },
  57. };
  58. ApplyTemplate(target);
  59. Assert.Equal(secondContent, target.SelectedContent);
  60. }
  61. [Fact]
  62. public void Logical_Children_Should_Be_TabItems()
  63. {
  64. var target = new TabControl
  65. {
  66. Template = TabControlTemplate(),
  67. Items =
  68. {
  69. new TabItem
  70. {
  71. Content = "foo"
  72. },
  73. new TabItem
  74. {
  75. Content = "bar"
  76. },
  77. }
  78. };
  79. Assert.Equal(target.Items, target.GetLogicalChildren().ToList());
  80. target.ApplyTemplate();
  81. Assert.Equal(target.Items, target.GetLogicalChildren().ToList());
  82. }
  83. [Fact]
  84. public void Removal_Should_Set_First_Tab()
  85. {
  86. var target = new TabControl
  87. {
  88. Template = TabControlTemplate(),
  89. Items =
  90. {
  91. new TabItem
  92. {
  93. Name = "first",
  94. Content = "foo",
  95. },
  96. new TabItem
  97. {
  98. Name = "second",
  99. Content = "bar",
  100. },
  101. new TabItem
  102. {
  103. Name = "3rd",
  104. Content = "barf",
  105. },
  106. }
  107. };
  108. Prepare(target);
  109. target.SelectedItem = target.Items[1];
  110. var item = Assert.IsType<TabItem>(target.Items[1]);
  111. Assert.Same(item, target.SelectedItem);
  112. Assert.Equal(item.Content, target.SelectedContent);
  113. target.Items.RemoveAt(1);
  114. item = Assert.IsType<TabItem>(target.Items[0]);
  115. Assert.Same(item, target.SelectedItem);
  116. Assert.Equal(item.Content, target.SelectedContent);
  117. }
  118. [Fact]
  119. public void Removal_Should_Set_New_Item0_When_Item0_Selected()
  120. {
  121. var target = new TabControl
  122. {
  123. Template = TabControlTemplate(),
  124. Items =
  125. {
  126. new TabItem
  127. {
  128. Name = "first",
  129. Content = "foo",
  130. },
  131. new TabItem
  132. {
  133. Name = "second",
  134. Content = "bar",
  135. },
  136. new TabItem
  137. {
  138. Name = "3rd",
  139. Content = "barf",
  140. },
  141. }
  142. };
  143. Prepare(target);
  144. target.SelectedItem = target.Items[0];
  145. var item = Assert.IsType<TabItem>(target.Items[0]);
  146. Assert.Same(item, target.SelectedItem);
  147. Assert.Equal(item.Content, target.SelectedContent);
  148. target.Items.RemoveAt(0);
  149. item = Assert.IsType<TabItem>(target.Items[0]);
  150. Assert.Same(item, target.SelectedItem);
  151. Assert.Equal(item.Content, target.SelectedContent);
  152. }
  153. [Fact]
  154. public void Removal_Should_Set_New_Item0_When_Item0_Selected_With_DataTemplate()
  155. {
  156. using var app = UnitTestApplication.Start(TestServices.StyledWindow);
  157. var collection = new ObservableCollection<Item>()
  158. {
  159. new Item("first"),
  160. new Item("second"),
  161. new Item("3rd"),
  162. };
  163. var target = new TabControl
  164. {
  165. Template = TabControlTemplate(),
  166. ItemsSource = collection,
  167. };
  168. Prepare(target);
  169. target.SelectedItem = collection[0];
  170. Assert.Same(collection[0], target.SelectedItem);
  171. Assert.Equal(collection[0], target.SelectedContent);
  172. collection.RemoveAt(0);
  173. Assert.Same(collection[0], target.SelectedItem);
  174. Assert.Equal(collection[0], target.SelectedContent);
  175. }
  176. [Fact]
  177. public void TabItem_Templates_Should_Be_Set_Before_TabItem_ApplyTemplate()
  178. {
  179. var template = new FuncControlTemplate<TabItem>((x, __) => new Decorator());
  180. TabControl target;
  181. var root = new TestRoot
  182. {
  183. Styles =
  184. {
  185. new Style(x => x.OfType<TabItem>())
  186. {
  187. Setters =
  188. {
  189. new Setter(TemplatedControl.TemplateProperty, template)
  190. }
  191. }
  192. },
  193. Child = (target = new TabControl
  194. {
  195. Template = TabControlTemplate(),
  196. Items =
  197. {
  198. new TabItem
  199. {
  200. Name = "first",
  201. Content = "foo",
  202. },
  203. new TabItem
  204. {
  205. Name = "second",
  206. Content = "bar",
  207. },
  208. new TabItem
  209. {
  210. Name = "3rd",
  211. Content = "barf",
  212. },
  213. },
  214. })
  215. };
  216. var collection = target.Items.Cast<TabItem>().ToList();
  217. Assert.Same(collection[0].Template, template);
  218. Assert.Same(collection[1].Template, template);
  219. Assert.Same(collection[2].Template, template);
  220. }
  221. [Fact]
  222. public void DataContexts_Should_Be_Correctly_Set()
  223. {
  224. var items = new object[]
  225. {
  226. "Foo",
  227. new Item("Bar"),
  228. new TextBlock { Text = "Baz" },
  229. new TabItem { Content = "Qux" },
  230. new TabItem { Content = new TextBlock { Text = "Bob" } }
  231. };
  232. var target = new TabControl
  233. {
  234. Template = TabControlTemplate(),
  235. DataContext = "Base",
  236. DataTemplates =
  237. {
  238. new FuncDataTemplate<Item>((x, __) => new Button { Content = x })
  239. },
  240. ItemsSource = items,
  241. };
  242. ApplyTemplate(target);
  243. target.ContentPart.UpdateChild();
  244. var dataContext = ((TextBlock)target.ContentPart.Child).DataContext;
  245. Assert.Equal(items[0], dataContext);
  246. target.SelectedIndex = 1;
  247. target.ContentPart.UpdateChild();
  248. dataContext = ((Button)target.ContentPart.Child).DataContext;
  249. Assert.Equal(items[1], dataContext);
  250. target.SelectedIndex = 2;
  251. target.ContentPart.UpdateChild();
  252. dataContext = ((TextBlock)target.ContentPart.Child).DataContext;
  253. Assert.Equal("Base", dataContext);
  254. target.SelectedIndex = 3;
  255. target.ContentPart.UpdateChild();
  256. dataContext = ((TextBlock)target.ContentPart.Child).DataContext;
  257. Assert.Equal("Qux", dataContext);
  258. target.SelectedIndex = 4;
  259. target.ContentPart.UpdateChild();
  260. dataContext = target.ContentPart.DataContext;
  261. Assert.Equal("Base", dataContext);
  262. }
  263. /// <summary>
  264. /// Non-headered control items should result in TabItems with empty header.
  265. /// </summary>
  266. /// <remarks>
  267. /// If a TabControl is created with non IHeadered controls as its items, don't try to
  268. /// display the control in the header: if the control is part of the header then
  269. /// *that* control would also end up in the content region, resulting in dual-parentage
  270. /// breakage.
  271. /// </remarks>
  272. [Fact]
  273. public void Non_IHeadered_Control_Items_Should_Be_Ignored()
  274. {
  275. var target = new TabControl
  276. {
  277. Template = TabControlTemplate(),
  278. Items =
  279. {
  280. new TextBlock { Text = "foo" },
  281. new TextBlock { Text = "bar" },
  282. },
  283. };
  284. ApplyTemplate(target);
  285. var logicalChildren = target.GetLogicalChildren();
  286. var result = logicalChildren
  287. .OfType<TabItem>()
  288. .Select(x => x.Header)
  289. .ToList();
  290. Assert.Equal(new object[] { null, null }, result);
  291. }
  292. [Fact]
  293. public void Should_Handle_Changing_To_TabItem_With_Null_Content()
  294. {
  295. TabControl target = new TabControl
  296. {
  297. Template = TabControlTemplate(),
  298. Items =
  299. {
  300. new TabItem { Header = "Foo" },
  301. new TabItem { Header = "Foo", Content = new Decorator() },
  302. new TabItem { Header = "Baz" },
  303. },
  304. };
  305. ApplyTemplate(target);
  306. target.SelectedIndex = 2;
  307. var page = (TabItem)target.SelectedItem;
  308. Assert.Null(page.Content);
  309. }
  310. [Fact]
  311. public void DataTemplate_Created_Content_Should_Be_Logical_Child_After_ApplyTemplate()
  312. {
  313. TabControl target = new TabControl
  314. {
  315. Template = TabControlTemplate(),
  316. ContentTemplate = new FuncDataTemplate<string>((x, _) =>
  317. new TextBlock { Tag = "bar", Text = x }),
  318. ItemsSource = new[] { "Foo" },
  319. };
  320. var root = new TestRoot(target);
  321. ApplyTemplate(target);
  322. target.ContentPart.UpdateChild();
  323. var content = Assert.IsType<TextBlock>(target.ContentPart.Child);
  324. Assert.Equal("bar", content.Tag);
  325. Assert.Same(target, content.GetLogicalParent());
  326. Assert.Single(target.GetLogicalChildren(), content);
  327. }
  328. [Fact]
  329. public void SelectedContentTemplate_Updates_After_New_ContentTemplate()
  330. {
  331. TabControl target = new TabControl
  332. {
  333. Template = TabControlTemplate(),
  334. ItemsSource = new[] { "Foo" },
  335. };
  336. var root = new TestRoot(target);
  337. ApplyTemplate(target);
  338. ((ContentPresenter)target.ContentPart).UpdateChild();
  339. Assert.Equal(null, Assert.IsType<TextBlock>(target.ContentPart.Child).Tag);
  340. target.ContentTemplate = new FuncDataTemplate<string>((x, _) =>
  341. new TextBlock { Tag = "bar", Text = x });
  342. Assert.Equal("bar", Assert.IsType<TextBlock>(target.ContentPart.Child).Tag);
  343. }
  344. [Fact]
  345. public void Should_Not_Propagate_DataContext_To_TabItem_Content()
  346. {
  347. var dataContext = "DataContext";
  348. var tabItem = new TabItem();
  349. var target = new TabControl
  350. {
  351. Template = TabControlTemplate(),
  352. DataContext = dataContext,
  353. Items = { tabItem }
  354. };
  355. ApplyTemplate(target);
  356. Assert.NotEqual(dataContext, tabItem.Content);
  357. }
  358. [Fact]
  359. public void Can_Have_Empty_Tab_Control()
  360. {
  361. using (UnitTestApplication.Start(TestServices.StyledWindow))
  362. {
  363. var xaml = @"
  364. <Window xmlns='https://github.com/avaloniaui'
  365. xmlns:x='http://schemas.microsoft.com/winfx/2006/xaml'
  366. xmlns:local='clr-namespace:Avalonia.Markup.Xaml.UnitTests.Xaml;assembly=Avalonia.Markup.Xaml.UnitTests'>
  367. <TabControl Name='tabs' ItemsSource='{Binding Tabs}'/>
  368. </Window>";
  369. var window = (Window)AvaloniaRuntimeXamlLoader.Load(xaml);
  370. var tabControl = window.FindControl<TabControl>("tabs");
  371. tabControl.DataContext = new { Tabs = new List<string>() };
  372. window.ApplyTemplate();
  373. Assert.Equal(0, tabControl.ItemsSource.Count());
  374. }
  375. }
  376. [Fact]
  377. public void Tab_Navigation_Should_Move_To_First_TabItem_When_No_Anchor_Element_Selected()
  378. {
  379. var services = TestServices.StyledWindow.With(
  380. focusManager: new FocusManager(),
  381. keyboardDevice: () => new KeyboardDevice());
  382. using var app = UnitTestApplication.Start(services);
  383. var target = new TabControl
  384. {
  385. Template = TabControlTemplate(),
  386. Items =
  387. {
  388. new TabItem { Header = "foo" },
  389. new TabItem { Header = "bar" },
  390. new TabItem { Header = "baz" },
  391. }
  392. };
  393. var button = new Button
  394. {
  395. Content = "Button",
  396. [DockPanel.DockProperty] = Dock.Top,
  397. };
  398. var root = new TestRoot
  399. {
  400. Child = new DockPanel
  401. {
  402. Children =
  403. {
  404. button,
  405. target,
  406. }
  407. }
  408. };
  409. var navigation = new KeyboardNavigationHandler();
  410. navigation.SetOwner(root);
  411. root.LayoutManager.ExecuteInitialLayoutPass();
  412. button.Focus();
  413. RaiseKeyEvent(button, Key.Tab);
  414. var item = target.ContainerFromIndex(0);
  415. Assert.Same(item, root.FocusManager.GetFocusedElement());
  416. }
  417. [Fact]
  418. public void Tab_Navigation_Should_Move_To_Anchor_TabItem()
  419. {
  420. var services = TestServices.StyledWindow.With(
  421. focusManager: new FocusManager(),
  422. keyboardDevice: () => new KeyboardDevice());
  423. using var app = UnitTestApplication.Start(services);
  424. var target = new TestTabControl
  425. {
  426. Template = TabControlTemplate(),
  427. Items =
  428. {
  429. new TabItem { Header = "foo" },
  430. new TabItem { Header = "bar" },
  431. new TabItem { Header = "baz" },
  432. }
  433. };
  434. var button = new Button
  435. {
  436. Content = "Button",
  437. [DockPanel.DockProperty] = Dock.Top,
  438. };
  439. var root = new TestRoot
  440. {
  441. Width = 1000,
  442. Height = 1000,
  443. Child = new DockPanel
  444. {
  445. Children =
  446. {
  447. button,
  448. target,
  449. }
  450. }
  451. };
  452. var navigation = new KeyboardNavigationHandler();
  453. navigation.SetOwner(root);
  454. root.LayoutManager.ExecuteInitialLayoutPass();
  455. button.Focus();
  456. target.Selection.AnchorIndex = 1;
  457. RaiseKeyEvent(button, Key.Tab);
  458. var item = target.ContainerFromIndex(1);
  459. Assert.Same(item, root.FocusManager.GetFocusedElement());
  460. RaiseKeyEvent(item, Key.Tab);
  461. Assert.Same(button, root.FocusManager.GetFocusedElement());
  462. target.Selection.AnchorIndex = 2;
  463. RaiseKeyEvent(button, Key.Tab);
  464. item = target.ContainerFromIndex(2);
  465. Assert.Same(item, root.FocusManager.GetFocusedElement());
  466. }
  467. [Fact]
  468. public void TabItem_Header_Should_Be_Settable_By_Style_When_DataContext_Is_Set()
  469. {
  470. var tabItem = new TabItem
  471. {
  472. DataContext = "Some DataContext"
  473. };
  474. _ = new TestRoot
  475. {
  476. Styles =
  477. {
  478. new Style(x => x.OfType<TabItem>())
  479. {
  480. Setters =
  481. {
  482. new Setter(HeaderedContentControl.HeaderProperty, "Header from style")
  483. }
  484. }
  485. },
  486. Child = tabItem
  487. };
  488. Assert.Equal("Header from style", tabItem.Header);
  489. }
  490. [Fact]
  491. public void TabItem_TabStripPlacement_Should_Be_Correctly_Set()
  492. {
  493. var items = new object[]
  494. {
  495. "Foo",
  496. new TabItem { Content = new TextBlock { Text = "Baz" } }
  497. };
  498. var target = new TabControl
  499. {
  500. Template = TabControlTemplate(),
  501. DataContext = "Base",
  502. ItemsSource = items
  503. };
  504. ApplyTemplate(target);
  505. var result = target.GetLogicalChildren()
  506. .OfType<TabItem>()
  507. .ToList();
  508. Assert.Collection(
  509. result,
  510. x => Assert.Equal(Dock.Top, x.TabStripPlacement),
  511. x => Assert.Equal(Dock.Top, x.TabStripPlacement)
  512. );
  513. target.TabStripPlacement = Dock.Right;
  514. result = target.GetLogicalChildren()
  515. .OfType<TabItem>()
  516. .ToList();
  517. Assert.Collection(
  518. result,
  519. x => Assert.Equal(Dock.Right, x.TabStripPlacement),
  520. x => Assert.Equal(Dock.Right, x.TabStripPlacement)
  521. );
  522. }
  523. [Fact]
  524. public void TabItem_TabStripPlacement_Should_Be_Correctly_Set_For_New_Items()
  525. {
  526. var items = new object[]
  527. {
  528. "Foo",
  529. new TabItem { Content = new TextBlock { Text = "Baz" } }
  530. };
  531. var target = new TabControl
  532. {
  533. Template = TabControlTemplate(),
  534. DataContext = "Base"
  535. };
  536. ApplyTemplate(target);
  537. target.ItemsSource = items;
  538. var result = target.GetLogicalChildren()
  539. .OfType<TabItem>()
  540. .ToList();
  541. Assert.Collection(
  542. result,
  543. x => Assert.Equal(Dock.Top, x.TabStripPlacement),
  544. x => Assert.Equal(Dock.Top, x.TabStripPlacement)
  545. );
  546. target.TabStripPlacement = Dock.Right;
  547. result = target.GetLogicalChildren()
  548. .OfType<TabItem>()
  549. .ToList();
  550. Assert.Collection(
  551. result,
  552. x => Assert.Equal(Dock.Right, x.TabStripPlacement),
  553. x => Assert.Equal(Dock.Right, x.TabStripPlacement)
  554. );
  555. }
  556. private static IControlTemplate TabControlTemplate()
  557. {
  558. return new FuncControlTemplate<TabControl>((parent, scope) =>
  559. new StackPanel
  560. {
  561. Children =
  562. {
  563. new ItemsPresenter
  564. {
  565. Name = "PART_ItemsPresenter",
  566. }.RegisterInNameScope(scope),
  567. new ContentPresenter
  568. {
  569. Name = "PART_SelectedContentHost",
  570. [!ContentPresenter.ContentProperty] = parent[!TabControl.SelectedContentProperty],
  571. [!ContentPresenter.ContentTemplateProperty] = parent[!TabControl.SelectedContentTemplateProperty],
  572. }.RegisterInNameScope(scope)
  573. }
  574. });
  575. }
  576. private static IControlTemplate TabItemTemplate()
  577. {
  578. return new FuncControlTemplate<TabItem>((parent, scope) =>
  579. new ContentPresenter
  580. {
  581. Name = "PART_ContentPresenter",
  582. [!ContentPresenter.ContentProperty] = parent[!TabItem.HeaderProperty],
  583. [!ContentPresenter.ContentTemplateProperty] = parent[!TabItem.HeaderTemplateProperty]
  584. }.RegisterInNameScope(scope));
  585. }
  586. private static void Prepare(TabControl target)
  587. {
  588. ApplyTemplate(target);
  589. target.Measure(Size.Infinity);
  590. target.Arrange(new Rect(target.DesiredSize));
  591. }
  592. private static void RaiseKeyEvent(Control target, Key key, KeyModifiers inputModifiers = 0)
  593. {
  594. target.RaiseEvent(new KeyEventArgs
  595. {
  596. RoutedEvent = InputElement.KeyDownEvent,
  597. KeyModifiers = inputModifiers,
  598. Key = key
  599. });
  600. }
  601. private static void ApplyTemplate(TabControl target)
  602. {
  603. target.ApplyTemplate();
  604. target.Presenter.ApplyTemplate();
  605. foreach (var tabItem in target.GetLogicalChildren().OfType<TabItem>())
  606. {
  607. tabItem.Template = TabItemTemplate();
  608. tabItem.ApplyTemplate();
  609. tabItem.Presenter.UpdateChild();
  610. }
  611. target.ContentPart.ApplyTemplate();
  612. }
  613. private class Item
  614. {
  615. public Item(string value)
  616. {
  617. Value = value;
  618. }
  619. public string Value { get; }
  620. }
  621. private class TestTabControl : TabControl
  622. {
  623. protected override Type StyleKeyOverride => typeof(TabControl);
  624. public new ISelectionModel Selection => base.Selection;
  625. }
  626. }
  627. }