I am building my own Flyout
control with list selection to use it on Windows Phone as well as Windows Desktop. Unlike ListPickerFlyout class, the Flyout class does not have an async method to show the flyout.
How can I call the ShowAt
method async and return the selected value back after the flyout was closed?
Solution:
The async behavior can be achieved with TaskCompletionSource<T>
(thanks to the AwaitableUI libary). What still bothers me is that I have to create the ListView manually within the constructor. It would be nice if I can use XAML instead and just assign a Template, but I did not find a working way.
public class ListPickerFlyout<T> : Flyout where T : class
{
private event EventHandler<object> ItemPicked;
public ListPickerFlyout(IEnumerable<T> items)
{
Placement = FlyoutPlacementMode.Full;
Opening += OnOpening;
Closed += OnClosed;
var listView = new ListView();
listView.SelectionMode = ListViewSelectionMode.None;
listView.IsItemClickEnabled = true;
listView.ItemClick += OnItemClick;
listView.DisplayMemberPath = "Name";
listView.SetBinding(ListView.ItemsSourceProperty, new Binding { Source = items });
Content = listView;
}
public async Task<T> ShowAsync()
{
this.ShowAt(Window.Current.Content as Frame);
var tcs = new TaskCompletionSource<T>();
EventHandler<object> eventHandler = null;
eventHandler = (s, e) =>
{
this.Closed -= eventHandler;
this.ItemPicked -= eventHandler;
tcs.SetResult(e as T);
};
this.Closed += eventHandler;
this.ItemPicked += eventHandler;
return await tcs.Task;
}
private void OnItemClick(object sender, ItemClickEventArgs e)
{
var selectedItem = e.ClickedItem as T;
var eventHandler = ItemPicked;
if (eventHandler != null)
eventHandler(this, selectedItem);
this.Hide();
}
private void OnOpening(object sender, object e)
{
var frame = Window.Current.Content as Frame;
var page = frame.Content as Page;
if (page != null)
page.BottomAppBar.Visibility = Visibility.Collapsed;
}
private void OnClosed(object sender, object e)
{
var frame = Window.Current.Content as Frame;
var page = frame.Content as Page;
if (page != null)
page.BottomAppBar.Visibility = Visibility.Visible;
}
}