sharee.bike-App/SharedBusinessLogic/ViewModel/Account/AccountPageViewModel.cs

489 lines
14 KiB
C#
Raw Permalink Normal View History

2021-05-13 20:03:07 +02:00
using System;
using System.ComponentModel;
using System.Threading.Tasks;
2022-08-30 15:42:25 +02:00
using Plugin.Connectivity;
using Serilog;
2024-04-09 12:53:23 +02:00
using ShareeBike.Model;
using ShareeBike.MultilingualResources;
using ShareeBike.Repository.Exception;
using ShareeBike.View;
using ShareeBike.ViewModel.Settings;
2021-05-13 20:03:07 +02:00
2024-04-09 12:53:23 +02:00
namespace ShareeBike.ViewModel.Account
2021-05-13 20:03:07 +02:00
{
2022-09-06 16:08:19 +02:00
public class AccountPageViewModel : INotifyPropertyChanged
{
/// <summary>
/// Count of requested/ booked bikes of logged in user.
/// </summary>
private int? m_iMyBikesCount;
/// <summary>
/// Reference on view service to show modal notifications and to perform navigation.
/// </summary>
private readonly IViewService m_oViewService;
/// <summary>
/// Holds the exception which occurred getting bikes occupied information.
/// </summary>
private Exception m_oException;
/// <summary>
/// Fired if a property changes.
/// </summary>
public event PropertyChangedEventHandler PropertyChanged;
/// <summary> Object to manage update of view model objects from Copri.</summary>
private IPollingUpdateTaskManager m_oViewUpdateManager;
/// <summary> Holds a reference to the external trigger service. </summary>
private Action<string> OpenUrlInExternalBrowser { get; }
2024-04-09 12:53:23 +02:00
/// <summary> Reference on the shareeBike app instance. </summary>
private IShareeBikeApp ShareeBikeApp { get; }
2022-09-06 16:08:19 +02:00
2023-07-19 10:10:36 +02:00
/// <summary> Used to block more than on copri requests at a given time.</summary>
private bool isIdle = true;
/// <summary>
/// True if any action can be performed (logout etc.)
/// </summary>
public virtual bool IsIdle
{
get => isIdle;
set
{
if (value == isIdle)
return;
Log.ForContext<AccountPageViewModel>().Debug($"Switch value of {nameof(IsIdle)} to {value}.");
isIdle = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(IsIdle)));
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(IsProcessWithRunningProcessView)));
}
}
public bool IsProcessWithRunningProcessView => !isIdle;
/// <summary> Holds info about current action. </summary>
private string statusInfoText;
/// <summary> Holds info about current action. </summary>
public string StatusInfoText
{
get => statusInfoText;
set
{
if (value == statusInfoText)
return;
Log.ForContext<LoginPageViewModel>().Debug($"Switch value of {nameof(StatusInfoText)} to {value}.");
statusInfoText = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(StatusInfoText)));
}
}
2022-09-06 16:08:19 +02:00
/// <summary> Constructs a settings page view model object.</summary>
2024-04-09 12:53:23 +02:00
/// <param name="shareeBikeApp"> Reference to shareeBike app model.</param>
2022-09-06 16:08:19 +02:00
/// <param name="p_oUser"></param>
/// <param name="p_oDevice"></param>
/// <param name="p_oFilterGroup">Filter to apply on stations and bikes.</param>
/// <param name="p_oUris">Available copri server host uris including uri to use for next start.</param>
2023-04-19 12:14:14 +02:00
/// <param name="p_oPolling"> Holds whether to poll or not and the period length is polling is on. </param>
/// <param name="p_oDefaultPollingPeriode">Default polling period length.</param>
2022-09-06 16:08:19 +02:00
/// <param name="p_oMinimumLogEventLevel">Controls logging level.</param>
/// <param name="viewService">Interface to view</param>
public AccountPageViewModel(
2024-04-09 12:53:23 +02:00
IShareeBikeApp shareeBikeApp,
2022-09-06 16:08:19 +02:00
Action<string> openUrlInExternalBrowser,
IViewService viewService)
{
2024-04-09 12:53:23 +02:00
ShareeBikeApp = shareeBikeApp
?? throw new ArgumentException("Can not instantiate settings page view model- object. No shareeBike app object available.");
2022-09-06 16:08:19 +02:00
OpenUrlInExternalBrowser = openUrlInExternalBrowser
?? throw new ArgumentException("Can not instantiate settings page view model- object. No user external browse service available.");
m_oViewService = viewService
?? throw new ArgumentException("Can not instantiate settings page view model- object. No user view service available.");
m_oViewUpdateManager = new IdlePollingUpdateTaskManager();
2024-04-09 12:53:23 +02:00
Polling = new PollingViewModel(ShareeBikeApp.Polling);
2022-09-06 16:08:19 +02:00
2024-04-09 12:53:23 +02:00
ShareeBikeApp.ActiveUser.StateChanged += OnStateChanged;
2022-09-06 16:08:19 +02:00
}
/// <summary>
/// Log in state of user changed.
/// </summary>
/// <param name="p_oSender"></param>
/// <param name="p_oEventArgs"></param>
private void OnStateChanged(object p_oSender, EventArgs p_oEventArgs)
{
var l_oPropertyChanged = PropertyChanged;
if (l_oPropertyChanged != null)
{
l_oPropertyChanged(this, new PropertyChangedEventArgs(nameof(LoggedInInfo)));
l_oPropertyChanged(this, new PropertyChangedEventArgs(nameof(IsBookingStateInfoVisible)));
l_oPropertyChanged(this, new PropertyChangedEventArgs(nameof(BookingStateInfo)));
}
}
/// <summary> Holds information whether app is connected to web or not. </summary>
private bool? isConnected = null;
/// <summary>Exposes the is connected state. </summary>
private bool IsConnected
{
get => isConnected ?? false;
set
{
var bookingStateInfo = BookingStateInfo;
isConnected = value;
if (bookingStateInfo == BookingStateInfo)
{
// Nothing to do.
return;
}
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(BookingStateInfo)));
}
}
/// <summary>
/// Gets the value of sessionn cookie (for debugging purposes).
/// </summary>
public string SessionCookie
{
2024-04-09 12:53:23 +02:00
get { return ShareeBikeApp.ActiveUser.SessionCookie; }
2022-09-06 16:08:19 +02:00
}
/// <summary>
/// Gets value whether a user is logged in or not.
/// </summary>
public bool IsLogoutPossible
{
get
{
return m_oException == null
2024-04-09 12:53:23 +02:00
&& ShareeBikeApp.ActiveUser.IsLoggedIn;
2022-09-06 16:08:19 +02:00
}
}
/// <summary>
/// Is true if user is logged in.
/// </summary>
public bool IsLoggedIn
{
get
{
2024-04-09 12:53:23 +02:00
return ShareeBikeApp.ActiveUser.IsLoggedIn;
2022-09-06 16:08:19 +02:00
}
}
/// <summary>
/// Shows info about logged in user if user is logged in. Text "Logged out" otherwise.
/// </summary>
public string LoggedInInfo
{
get
{
2024-04-09 12:53:23 +02:00
if (!ShareeBikeApp.ActiveUser.IsLoggedIn)
2022-09-06 16:08:19 +02:00
{
return AppResources.MarkingLoggedInStateInfoNotLoggedIn;
}
2024-04-09 12:53:23 +02:00
return string.Format(AppResources.MarkingLoggedInStateInfoLoggedIn, ShareeBikeApp.ActiveUser.Mail);
2022-09-06 16:08:19 +02:00
}
}
/// <summary>
2023-04-19 12:14:14 +02:00
/// Gets a value indicating whether booking state info is available or not.
2022-09-06 16:08:19 +02:00
/// </summary>
public bool IsBookingStateInfoVisible
{
get
{
return BookingStateInfo.Length > 0;
}
}
/// <summary>
/// Count of bikes reserved and/ or occupied.
/// </summary>
private int? BikesOccupiedCount
{
get
{
return m_iMyBikesCount;
}
set
{
var isBookingStateInfoVisible = IsBookingStateInfoVisible;
var bookingStateInfo = BookingStateInfo;
m_iMyBikesCount = value;
if (isBookingStateInfoVisible != IsBookingStateInfoVisible)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(IsBookingStateInfoVisible)));
}
if (bookingStateInfo != BookingStateInfo)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(BookingStateInfo)));
}
}
}
/// <summary>
/// Shows info about reserved/ booked bikes if there are.
/// </summary>
public string BookingStateInfo
{
get
{
2024-04-09 12:53:23 +02:00
if (!ShareeBikeApp.ActiveUser.IsLoggedIn || !m_iMyBikesCount.HasValue || m_iMyBikesCount.Value <= 0)
2022-09-06 16:08:19 +02:00
{
// Either
// - user is not logged in or
// - user has no bikes requested/ booked
return string.Empty;
}
2023-08-31 12:20:06 +02:00
var bookingStateInfo = string.Format(AppResources.MarkingAccountReservedBookedBikes, m_iMyBikesCount.Value);
2022-09-06 16:08:19 +02:00
if (!IsConnected)
{
// Append offline info
2023-08-31 12:20:06 +02:00
return $"{bookingStateInfo} {AppResources.MarkingAccountConnectionOffline}";
2022-09-06 16:08:19 +02:00
}
if (Exception != null)
{
// Append offline info
2023-08-31 12:20:06 +02:00
return $"{bookingStateInfo} {AppResources.MarkingAccountConnectionInterrupted} ";
2022-09-06 16:08:19 +02:00
}
return bookingStateInfo;
}
}
/// <summary> Command object to bind logout button to view model.</summary>
2024-04-09 12:53:23 +02:00
public System.Windows.Input.ICommand OnLogoutRequest => new Xamarin.Forms.Command(async () => await Logout(), () => IsLogoutPossible);
2022-09-06 16:08:19 +02:00
/// <summary> Processes request to view personal data.</summary>
public System.Windows.Input.ICommand OnManageAccount => new Xamarin.Forms.Command(async () =>
{
if (CrossConnectivity.Current.IsConnected)
{
await m_oViewService.PushAsync(ViewTypes.ManageAccountPage);
}
else
{
await m_oViewService.DisplayAlert(
2023-08-31 12:20:06 +02:00
AppResources.MessageHintTitle,
AppResources.ErrorNoWeb,
AppResources.MessageAnswerOk);
2022-09-06 16:08:19 +02:00
}
});
/// <summary>
/// Request to log out.
/// </summary>
public async Task Logout()
{
2023-07-19 10:10:36 +02:00
StatusInfoText = AppResources.ActivityTextOneMomentPlease;
IsIdle = false;
2022-09-06 16:08:19 +02:00
try
{
// Backup logout message before logout.
2024-04-09 12:53:23 +02:00
var l_oMessage = string.Format(AppResources.MessageLogoutGoodbye, ShareeBikeApp.ActiveUser.Mail);
2022-09-06 16:08:19 +02:00
// Stop polling before requesting bike.
2023-08-31 12:20:06 +02:00
await m_oViewUpdateManager.StopAsync();
2022-09-06 16:08:19 +02:00
try
{
// Log out user.
2024-04-09 12:53:23 +02:00
IsConnected = ShareeBikeApp.GetIsConnected();
await ShareeBikeApp.GetConnector(IsConnected).Command.DoLogout();
2022-09-06 16:08:19 +02:00
}
catch (UnsupportedCopriVersionDetectedException)
{
await m_oViewService.DisplayAlert(
2023-08-31 12:20:06 +02:00
AppResources.ErrorLogoutTitle,
2024-04-09 12:53:23 +02:00
string.Format(AppResources.MessageAppVersionIsOutdated, ShareeBikeApp.Flavor.GetDisplayName()),
2022-09-06 16:08:19 +02:00
AppResources.MessageAnswerOk);
// Restart polling again.
2023-08-31 12:20:06 +02:00
await m_oViewUpdateManager.StartAsync(Polling.ToImmutable());
2023-07-19 10:10:36 +02:00
IsIdle = true;
StatusInfoText = string.Empty;
2022-09-06 16:08:19 +02:00
return;
}
catch (Exception l_oException)
{
// Copri server is not reachable.
if (l_oException is WebConnectFailureException)
{
await m_oViewService.DisplayAlert(
2023-08-31 12:20:06 +02:00
AppResources.ErrorLogoutTitle,
AppResources.ErrorNoWeb,
AppResources.MessageAnswerOk);
2022-09-06 16:08:19 +02:00
}
else
{
2023-08-31 12:20:06 +02:00
await m_oViewService.DisplayAlert(
AppResources.ErrorLogoutTitle,
l_oException.Message,
AppResources.MessageAnswerOk);
2022-09-06 16:08:19 +02:00
}
// Restart polling again.
2023-08-31 12:20:06 +02:00
await m_oViewUpdateManager.StartAsync(Polling.ToImmutable());
2023-07-19 10:10:36 +02:00
IsIdle = true;
StatusInfoText = string.Empty;
2022-09-06 16:08:19 +02:00
return;
}
2024-04-09 12:53:23 +02:00
ShareeBikeApp.ActiveUser.Logout();
2022-09-06 16:08:19 +02:00
2023-08-31 12:20:06 +02:00
// Display information that log out was performed.
await m_oViewService.DisplayAlert(
AppResources.MessageLogoutGoodbyeTitle,
l_oMessage,
AppResources.MessageAnswerOk);
2022-09-06 16:08:19 +02:00
}
catch (Exception p_oException)
{
Log.Error("An unexpected error occurred displaying log out page. {@Exception}", p_oException);
2023-07-19 10:10:36 +02:00
IsIdle = true;
StatusInfoText = string.Empty;
2022-09-06 16:08:19 +02:00
return;
}
2023-07-19 10:10:36 +02:00
IsIdle = true;
StatusInfoText = string.Empty;
2022-09-06 16:08:19 +02:00
}
/// <summary>
/// Invoked when page is shown.
/// Starts update process.
/// </summary>
public async Task OnAppearing()
{
2024-04-09 12:53:23 +02:00
IsConnected = ShareeBikeApp.GetIsConnected();
BikesOccupiedCount = ShareeBikeApp.ActiveUser.IsLoggedIn
? (await ShareeBikeApp.GetConnector(IsConnected).Query.GetBikesOccupiedAsync()).Response.Count
2022-09-06 16:08:19 +02:00
: 0;
m_oViewUpdateManager = new PollingUpdateTaskManager(
() =>
{
2024-04-09 12:53:23 +02:00
ShareeBikeApp.PostAction(
unused => IsConnected = ShareeBikeApp.GetIsConnected(),
2022-09-06 16:08:19 +02:00
null);
int? l_iBikesCount = null;
2024-04-09 12:53:23 +02:00
var bikesOccupied = ShareeBikeApp.GetConnector(IsConnected).Query.GetBikesOccupiedAsync().Result;
l_iBikesCount = ShareeBikeApp.ActiveUser.IsLoggedIn
2022-09-06 16:08:19 +02:00
? bikesOccupied.Response.Count
: 0;
2024-04-09 12:53:23 +02:00
ShareeBikeApp.PostAction(
2022-09-06 16:08:19 +02:00
unused =>
{
BikesOccupiedCount = l_iBikesCount;
Exception = bikesOccupied.Exception;
},
null);
}
);
try
{
// Update bikes at station or my bikes depending on context.
2023-08-31 12:20:06 +02:00
await m_oViewUpdateManager.StartAsync(Polling.ToImmutable());
2022-09-06 16:08:19 +02:00
}
catch (Exception l_oExcetion)
{
Log.Error("Getting count of bikes on settings pags failed. {@l_oExcetion}.", l_oExcetion);
}
}
/// <summary>
/// Invoked when page is shutdown.
/// Currently invoked by code behind, would be nice if called by XAML in future versions.
/// </summary>
public async Task OnDisappearing()
{
try
{
Log.ForContext<AccountPageViewModel>().Information($"Entering {nameof(OnDisappearing)}...");
2023-08-31 12:20:06 +02:00
await m_oViewUpdateManager.StopAsync();
2022-09-06 16:08:19 +02:00
}
catch (Exception l_oException)
{
await m_oViewService.DisplayAlert(
2023-08-31 12:20:06 +02:00
AppResources.ErrorPageNotLoadedTitle,
$"{AppResources.ErrorPageNotLoaded}\r\n{l_oException.Message}",
AppResources.MessageAnswerOk);
2022-09-06 16:08:19 +02:00
}
}
/// <summary>
/// Exception which occurred getting bike information.
/// </summary>
protected Exception Exception
{
get
{
return m_oException;
}
set
{
var l_oException = m_oException;
var bookingStateInfo = BookingStateInfo;
m_oException = value;
if ((m_oException != null && l_oException == null)
|| (m_oException == null && l_oException != null))
{
// Error information is available and was not or the other way round.
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(IsLogoutPossible)));
}
if (bookingStateInfo != BookingStateInfo)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(BookingStateInfo)));
}
}
}
2023-04-19 12:14:14 +02:00
/// <summary>Polling period.</summary>
2022-09-06 16:08:19 +02:00
public PollingViewModel Polling { get; }
/// <summary> Opens login page.</summary>
public void RegisterRequest(string url)
{
try
{
OpenUrlInExternalBrowser(url);
}
catch (Exception p_oException)
{
Log.Error("Ein unerwarteter Fehler ist auf der Einstellungs- Seite beim Öffnen eines Browsers, Seite {url}, aufgetreten. {@Exception}", url, p_oException);
return;
}
}
}
2021-05-13 20:03:07 +02:00
}