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
}