DeckPresenter.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. // Copyright (c) The Perspex Project. All rights reserved.
  2. // Licensed under the MIT license. See licence.md file in the project root for full license information.
  3. using System;
  4. using System.Collections;
  5. using System.Linq;
  6. using System.Reactive.Linq;
  7. using System.Threading.Tasks;
  8. using Perspex.Animation;
  9. using Perspex.Controls.Generators;
  10. using Perspex.Controls.Primitives;
  11. using Perspex.Controls.Templates;
  12. using Perspex.Controls.Utils;
  13. namespace Perspex.Controls.Presenters
  14. {
  15. /// <summary>
  16. /// Displays pages inside an <see cref="ItemsControl"/>.
  17. /// </summary>
  18. public class DeckPresenter : Control, IItemsPresenter
  19. {
  20. /// <summary>
  21. /// Defines the <see cref="Items"/> property.
  22. /// </summary>
  23. public static readonly PerspexProperty<IEnumerable> ItemsProperty =
  24. ItemsControl.ItemsProperty.AddOwner<DeckPresenter>();
  25. /// <summary>
  26. /// Defines the <see cref="ItemsPanel"/> property.
  27. /// </summary>
  28. public static readonly PerspexProperty<ITemplate<IPanel>> ItemsPanelProperty =
  29. ItemsControl.ItemsPanelProperty.AddOwner<DeckPresenter>();
  30. /// <summary>
  31. /// Defines the <see cref="SelectedIndex"/> property.
  32. /// </summary>
  33. public static readonly PerspexProperty<int> SelectedIndexProperty =
  34. SelectingItemsControl.SelectedIndexProperty.AddOwner<DeckPresenter>();
  35. /// <summary>
  36. /// Defines the <see cref="Transition"/> property.
  37. /// </summary>
  38. public static readonly PerspexProperty<IPageTransition> TransitionProperty =
  39. Deck.TransitionProperty.AddOwner<DeckPresenter>();
  40. private bool _createdPanel;
  41. private IItemContainerGenerator _generator;
  42. /// <summary>
  43. /// Initializes static members of the <see cref="DeckPresenter"/> class.
  44. /// </summary>
  45. static DeckPresenter()
  46. {
  47. SelectedIndexProperty.Changed.AddClassHandler<DeckPresenter>(x => x.SelectedIndexChanged);
  48. }
  49. /// <summary>
  50. /// Gets the <see cref="IItemContainerGenerator"/> used to generate item container
  51. /// controls.
  52. /// </summary>
  53. public IItemContainerGenerator ItemContainerGenerator
  54. {
  55. get
  56. {
  57. if (_generator == null)
  58. {
  59. var i = TemplatedParent as ItemsControl;
  60. _generator = i?.ItemContainerGenerator ?? new ItemContainerGenerator(this);
  61. }
  62. return _generator;
  63. }
  64. set
  65. {
  66. if (_generator != null)
  67. {
  68. throw new InvalidOperationException("ItemContainerGenerator is already set.");
  69. }
  70. _generator = value;
  71. }
  72. }
  73. /// <summary>
  74. /// Gets or sets the items to display.
  75. /// </summary>
  76. public IEnumerable Items
  77. {
  78. get { return GetValue(ItemsProperty); }
  79. set { SetValue(ItemsProperty, value); }
  80. }
  81. /// <summary>
  82. /// Gets or sets the panel used to display the pages.
  83. /// </summary>
  84. public ITemplate<IPanel> ItemsPanel
  85. {
  86. get { return GetValue(ItemsPanelProperty); }
  87. set { SetValue(ItemsPanelProperty, value); }
  88. }
  89. /// <summary>
  90. /// Gets or sets the index of the selected page.
  91. /// </summary>
  92. public int SelectedIndex
  93. {
  94. get { return GetValue(SelectedIndexProperty); }
  95. set { SetValue(SelectedIndexProperty, value); }
  96. }
  97. /// <summary>
  98. /// Gets the panel used to display the pages.
  99. /// </summary>
  100. public IPanel Panel
  101. {
  102. get;
  103. private set;
  104. }
  105. /// <summary>
  106. /// Gets or sets a transition to use when switching pages.
  107. /// </summary>
  108. public IPageTransition Transition
  109. {
  110. get { return GetValue(TransitionProperty); }
  111. set { SetValue(TransitionProperty, value); }
  112. }
  113. /// <inheritdoc/>
  114. public override sealed void ApplyTemplate()
  115. {
  116. if (!_createdPanel)
  117. {
  118. CreatePanel();
  119. }
  120. }
  121. /// <summary>
  122. /// Creates the <see cref="Panel"/>.
  123. /// </summary>
  124. private void CreatePanel()
  125. {
  126. var logicalHost = this.FindReparentingHost();
  127. ClearVisualChildren();
  128. Panel = ItemsPanel.Build();
  129. Panel.SetValue(TemplatedParentProperty, TemplatedParent);
  130. AddVisualChild(Panel);
  131. if (logicalHost != null)
  132. {
  133. ((IReparentingControl)Panel).ReparentLogicalChildren(
  134. logicalHost,
  135. logicalHost.LogicalChildren);
  136. }
  137. _createdPanel = true;
  138. var task = MoveToPage(-1, SelectedIndex);
  139. }
  140. /// <summary>
  141. /// Moves to the selected page, animating if a <see cref="Transition"/> is set.
  142. /// </summary>
  143. /// <param name="fromIndex">The index of the old page.</param>
  144. /// <param name="toIndex">The index of the new page.</param>
  145. /// <returns>A task tracking the animation.</returns>
  146. private async Task MoveToPage(int fromIndex, int toIndex)
  147. {
  148. var generator = ItemContainerGenerator;
  149. IControl from = null;
  150. IControl to = null;
  151. if (fromIndex != -1)
  152. {
  153. from = generator.ContainerFromIndex(fromIndex);
  154. }
  155. if (toIndex != -1)
  156. {
  157. var item = Items.Cast<object>().ElementAt(toIndex);
  158. to = generator.CreateContainers(toIndex, new[] { item }, null).FirstOrDefault();
  159. if (to != null)
  160. {
  161. Panel.Children.Add(to);
  162. }
  163. }
  164. if (Transition != null)
  165. {
  166. await Transition.Start((Visual)from, (Visual)to, fromIndex < toIndex);
  167. }
  168. if (from != null)
  169. {
  170. Panel.Children.Remove(from);
  171. generator.RemoveContainers(fromIndex, new[] { from });
  172. }
  173. }
  174. /// <summary>
  175. /// Called when the <see cref="SelectedIndex"/> property changes.
  176. /// </summary>
  177. /// <param name="e">The event args.</param>
  178. private void SelectedIndexChanged(PerspexPropertyChangedEventArgs e)
  179. {
  180. if (Panel != null)
  181. {
  182. var task = MoveToPage((int)e.OldValue, (int)e.NewValue);
  183. }
  184. }
  185. }
  186. }