using Xamarin.Forms; using Xamarin.Forms.Xaml; using TINK.Model.Bike.BluetoothLock; namespace TINK.View.BikesAtStation { using System; using System.Linq; using System.Threading; using System.Threading.Tasks; using TINK.Model.Device; #if USEFLYOUT using TINK.View.MasterDetail; #endif using TINK.ViewModel; using TINK.Model; using TINK.Services.BluetoothLock.Tdo; using System.Collections.Generic; using Serilog; using TINK.Services.BluetoothLock; using Plugin.BLE; using TINK.ViewModel.BikesAtStation; using TINK.ViewModel.Bikes; using Xamarin.CommunityToolkit.Extensions; [XamlCompilation(XamlCompilationOptions.Compile)] #if USEFLYOUT public partial class BikesAtStationPage : ContentPage, IViewService, IDetailPage #else public partial class BikesAtStationPage : ContentPage, IViewService #endif { private BikesAtStationPageViewModel m_oViewModel; #if TRYNOTBACKSTYLE public BikesAtStationPage() { InitializeComponent(); var l_oModel = App.ModelRoot; var l_oViewModel = new BikesAtStationPageViewModel( l_oModel.BikesAtStation, l_oModel.ActiveUser, l_oModel.SelectedStation, this); BindingContext = l_oViewModel; BikesAtStationListView.ItemsSource = l_oViewModel; } #else public BikesAtStationPage() { } #endif /// /// Invoked when page is shown. /// Starts update process. /// protected async override void OnAppearing() { if (m_oViewModel != null) { #if BACKSTYLE // Hide master- detail menu to force user to navigate using back button. m_oNavigation.IsGestureEnabled = false; #endif // No need to create view model, set binding context an items source if already done. // If done twice tap events are fired multiple times (when hiding page using home button). await m_oViewModel.OnAppearing(); return; } try { var model = App.ModelRoot; // Backup synchronization context when called from GUI-thread. var synchronizationContext = SynchronizationContext.Current; m_oViewModel = new BikesAtStationPageViewModel( model.ActiveUser, App.PermissionsService, App.BluetoothService, Device.RuntimePlatform, model.SelectedStation, () => model.GetIsConnected(), (isConnected) => model.GetConnector(isConnected), App.GeolocationServicesContainer.Active, model.LocksServices.Active, model.Polling, (url) => DependencyService.Get().OpenUrl(url), (d, obj) => synchronizationContext.Post(d, obj), model.SmartDevice, this) { IsReportLevelVerbose = model.IsReportLevelVerbose }; } catch (Exception exception) { Log.ForContext().Error("Displaying bikes at station page failed. {Exception}", exception); await DisplayAlert("Fehler", $"Seite Räder an Station kann nicht angezeigt werden. ${exception.Message}", "OK"); return; } InitializeComponent(); #if BACKSTYLE // Hide master- detail menu to force user to navigate using back button. m_oNavigation.IsGestureEnabled = false; #endif BindingContext = m_oViewModel; BikesAtStationListView.ItemsSource = m_oViewModel; await m_oViewModel.OnAppearing(); } /// /// Invoked when pages is closed/ hidden. /// Stops update process. /// protected async override void OnDisappearing() { if (m_oViewModel != null) { // View model might be null. await m_oViewModel?.OnDisappearing(); } #if BACKSTYLE if (m_oNavigation!= null) m_oNavigation.IsGestureEnabled = true; // Enables master- detail menu navigation again when page is unloaded. #endif } /// Displays alert message. /// Title of message. /// Message to display. /// Type of buttons. public new async Task DisplayAlert(string title, string message, string cancel) => await App.Current.MainPage.DisplayAlert(title, message, cancel); /// Displays alert message. /// Title of message. /// Message to display. /// Detailed error description. /// Type of buttons. public async Task DisplayAdvancedAlert( string title, string message, string details, string cancel) => await App.Current.MainPage.DisplayAlert(title, $"{message}\r\nDetails:\r\n{details}", cancel); /// /// Displays alert message. /// /// Title of message. /// Message to display. /// Text of accept button. /// Text of button. /// True if user pressed accept. public new async Task DisplayAlert(string title, string message, string accept, string cancel) => await App.Current.MainPage.DisplayAlert(title, message, accept, cancel); /// Displays detailed alert message. /// Title of message. /// Message to display. /// Detailed error description. /// Text of accept button. /// Text of cancel button. /// True if user pressed accept. public async Task DisplayAdvancedAlert(string title, string message, string details, string accept, string cancel) => await App.Current.MainPage.DisplayAlert(title, $"{message}\r\nDetails:\r\n{details}", accept, cancel); #if USEFLYOUT /// Creates and a page an shows it. /// When user is not logged in navigation to Login page is supported. /// Type of page to show. public void ShowPage(ViewTypes p_oType, string title = null) => m_oNavigation.ShowPage(p_oType.GetViewType(), title); #else /// Shows a page. /// Route of the page to show. public async Task ShowPage(string route) => await Shell.Current.GoToAsync(route); #endif /// Pushes a page onto the modal stack. /// Page to display. public async Task PushModalAsync(ViewTypes typeOfPage) => await Navigation.PushModalAsync((Page)Activator.CreateInstance(typeOfPage.GetViewType())); /// Pops a page from the modal stack. public Task PopModalAsync() { throw new NotSupportedException(); } public Task PushAsync(ViewTypes p_oTypeOfPage) { throw new NotImplementedException(); } #if USEFLYOUT /// /// Delegate to perform navigation. /// private INavigationMasterDetail m_oNavigation; /// /// Delegate to perform navigation. /// public INavigationMasterDetail NavigationMasterDetail { set { m_oNavigation = value; } } #endif #if USCSHARP9 public async Task DisplayUserFeedbackPopup() => await Navigation.ShowPopupAsync(new FeedbackPopup()); #else public async Task DisplayUserFeedbackPopup() => await Navigation.ShowPopupAsync(new FeedbackPopup()); #endif } }