mirror of
https://dev.azure.com/TeilRad/sharee.bike%20App/_git/Code
synced 2025-01-05 13:06:27 +01:00
Small refactoring
This commit is contained in:
parent
34590fbc7d
commit
ba3f987d2a
2 changed files with 141 additions and 114 deletions
|
@ -22,7 +22,7 @@ namespace TINK.View.Map
|
||||||
private MapPageViewModel MapPageViewModel { get; set; }
|
private MapPageViewModel MapPageViewModel { get; set; }
|
||||||
|
|
||||||
/// <summary> Initialization status to ensure initialization logic is not called multiple times. </summary>
|
/// <summary> Initialization status to ensure initialization logic is not called multiple times. </summary>
|
||||||
private bool viewInitialized = false;
|
private bool isInitializationStarted = false;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Constructs map page instance.
|
/// Constructs map page instance.
|
||||||
|
@ -136,12 +136,13 @@ namespace TINK.View.Map
|
||||||
protected async override void OnAppearing()
|
protected async override void OnAppearing()
|
||||||
{
|
{
|
||||||
// Don't repeat the initialization if it has been completed already.
|
// Don't repeat the initialization if it has been completed already.
|
||||||
if (viewInitialized) return;
|
if (isInitializationStarted) return;
|
||||||
|
isInitializationStarted = true;
|
||||||
|
|
||||||
// Pass reference to member Navigation to show bikes at station x dialog.
|
// Pass reference to member Navigation to show bikes at station x dialog.
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
MapPageViewModel = createMapPageViewModel();
|
MapPageViewModel = CreateMapPageViewModel();
|
||||||
}
|
}
|
||||||
catch (Exception exception)
|
catch (Exception exception)
|
||||||
{
|
{
|
||||||
|
@ -166,7 +167,7 @@ namespace TINK.View.Map
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
applyCustomiOSStyling();
|
ApplyCustomiOSStyling();
|
||||||
}
|
}
|
||||||
catch (Exception exception)
|
catch (Exception exception)
|
||||||
{
|
{
|
||||||
|
@ -188,7 +189,7 @@ namespace TINK.View.Map
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// Pre move and scanle maps to avoid initial display of map in Rome.
|
// Pre move and scanle maps to avoid initial display of map in Rome.
|
||||||
premoveAndScaleMap();
|
PremoveAndScaleMap();
|
||||||
}
|
}
|
||||||
catch(Exception exception)
|
catch(Exception exception)
|
||||||
{
|
{
|
||||||
|
@ -206,14 +207,13 @@ namespace TINK.View.Map
|
||||||
Log.ForContext<MapPage>().Error("Invoking OnAppearing on map page view model failed. {Exception}", exception);
|
Log.ForContext<MapPage>().Error("Invoking OnAppearing on map page view model failed. {Exception}", exception);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
viewInitialized = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Premoves the Map to a certain location.
|
/// Premoves the Map to a certain location.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private void premoveAndScaleMap()
|
private void PremoveAndScaleMap()
|
||||||
{
|
{
|
||||||
Log.ForContext<MapPage>().Verbose("Moving and scaling map.");
|
Log.ForContext<MapPage>().Verbose("Moving and scaling map.");
|
||||||
MapPageViewModel.MoveAndScale(
|
MapPageViewModel.MoveAndScale(
|
||||||
|
@ -226,7 +226,7 @@ namespace TINK.View.Map
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates the Map Page's view model.
|
/// Creates the Map Page's view model.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private MapPageViewModel createMapPageViewModel()
|
private MapPageViewModel CreateMapPageViewModel()
|
||||||
{
|
{
|
||||||
Log.ForContext<MapPage>().Verbose("Constructing map page view model.");
|
Log.ForContext<MapPage>().Verbose("Constructing map page view model.");
|
||||||
return new MapPageViewModel(
|
return new MapPageViewModel(
|
||||||
|
@ -243,7 +243,7 @@ namespace TINK.View.Map
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Applies iOS specific styling to branded Buttons.
|
/// Applies iOS specific styling to branded Buttons.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private void applyCustomiOSStyling()
|
private void ApplyCustomiOSStyling()
|
||||||
{
|
{
|
||||||
if (Device.RuntimePlatform == Device.iOS)
|
if (Device.RuntimePlatform == Device.iOS)
|
||||||
{
|
{
|
||||||
|
@ -267,7 +267,6 @@ namespace TINK.View.Map
|
||||||
// View model might be null.
|
// View model might be null.
|
||||||
await MapPageViewModel?.OnDisappearing();
|
await MapPageViewModel?.OnDisappearing();
|
||||||
}
|
}
|
||||||
viewInitialized = false;
|
|
||||||
base.OnDisappearing();
|
base.OnDisappearing();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -316,6 +316,170 @@ namespace TINK.ViewModel.Map
|
||||||
|
|
||||||
ActionText = AppResources.ActivityTextRequestingLocationPermissions;
|
ActionText = AppResources.ActivityTextRequestingLocationPermissions;
|
||||||
|
|
||||||
|
ActionText = AppResources.ActivityTextMapLoadingStationsAndBikes;
|
||||||
|
IsConnected = TinkApp.GetIsConnected();
|
||||||
|
|
||||||
|
Result<StationsAndBikesContainer> resultStationsAndBikes = await TinkApp.GetConnector(IsConnected).Query.GetBikesAndStationsAsync();
|
||||||
|
|
||||||
|
TinkApp.Stations = resultStationsAndBikes.Response.StationsAll;
|
||||||
|
|
||||||
|
await SetStationsOnMap(resultStationsAndBikes.Response.StationsAll);
|
||||||
|
await HandleAuthCookieNotDefinedException(resultStationsAndBikes.Exception);
|
||||||
|
|
||||||
|
|
||||||
|
// Update pin colors.
|
||||||
|
Log.ForContext<MapPageViewModel>().Verbose("Starting update pins color...");
|
||||||
|
|
||||||
|
var colors = GetStationColors(
|
||||||
|
Pins.Select(x => x.Tag.ToString()).ToList(),
|
||||||
|
resultStationsAndBikes.Response.Bikes);
|
||||||
|
|
||||||
|
// Update pins color form count of bikes located at station.
|
||||||
|
UpdatePinsColor(colors);
|
||||||
|
|
||||||
|
Log.ForContext<MapPageViewModel>().Verbose("Update pins color done.");
|
||||||
|
|
||||||
|
|
||||||
|
// Move and scale before getting stations and bikes which takes some time.
|
||||||
|
ActionText = AppResources.ActivityTextCenterMap;
|
||||||
|
m_oViewUpdateManager = CreateUpdateTask();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Update bikes at station or my bikes depending on context.
|
||||||
|
await m_oViewUpdateManager.StartUpdateAyncPeridically(Polling);
|
||||||
|
}
|
||||||
|
catch (Exception)
|
||||||
|
{
|
||||||
|
// Excpetions are handled insde update task;
|
||||||
|
}
|
||||||
|
|
||||||
|
Exception = resultStationsAndBikes.Exception;
|
||||||
|
ActionText = "";
|
||||||
|
IsRunning = false;
|
||||||
|
IsMapPageEnabled = true;
|
||||||
|
|
||||||
|
await MoveMapToCurrentPositionOfUser();
|
||||||
|
}
|
||||||
|
catch (Exception l_oException)
|
||||||
|
{
|
||||||
|
Log.ForContext<MapPageViewModel>().Error($"An error occurred switching view TINK/ Konrad.\r\n{l_oException.Message}");
|
||||||
|
|
||||||
|
IsRunning = false;
|
||||||
|
|
||||||
|
await ViewService.DisplayAlert(
|
||||||
|
"Fehler",
|
||||||
|
$"Beim Anzeigen der Fahrradstandorte- Seite ist ein Fehler aufgetreten.\r\n{l_oException.Message}",
|
||||||
|
"OK");
|
||||||
|
|
||||||
|
IsMapPageEnabled = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Invoked when the auth cookie is not defined.
|
||||||
|
/// </summary>
|
||||||
|
private async Task HandleAuthCookieNotDefinedException(Exception exception)
|
||||||
|
{
|
||||||
|
if (exception?.GetType() == typeof(AuthcookieNotDefinedException))
|
||||||
|
{
|
||||||
|
Log.ForContext<MapPageViewModel>().Error("Map page is shown (probable for the first time after startup of app) and COPRI auth cookie is not defined. {@l_oException}", exception);
|
||||||
|
|
||||||
|
// COPRI reports an auth cookie error.
|
||||||
|
await ViewService.DisplayAlert(
|
||||||
|
AppResources.MessageWaring,
|
||||||
|
AppResources.MessageMapPageErrorAuthcookieUndefined,
|
||||||
|
AppResources.MessageAnswerOk);
|
||||||
|
|
||||||
|
await TinkApp.GetConnector(IsConnected).Command.DoLogout();
|
||||||
|
TinkApp.ActiveUser.Logout();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets the available stations on the map.
|
||||||
|
/// </summary>
|
||||||
|
private async Task SetStationsOnMap(StationDictionary stations)
|
||||||
|
{
|
||||||
|
if (Pins.Count > 0 && Pins.Count != stations.Count)
|
||||||
|
{
|
||||||
|
// Either
|
||||||
|
// - user logged in/ logged out which might lead to more/ less stations beeing available
|
||||||
|
// - new stations were added/ existing ones remove
|
||||||
|
Pins.Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if there are alreay any pins to the map
|
||||||
|
// i.e detecte first call of member OnAppearing after construction
|
||||||
|
if (Pins.Count <= 0)
|
||||||
|
{
|
||||||
|
Log.ForContext<MapPageViewModel>().Debug($"{(ActiveFilterMap.GetGroup().Any() ? $"Active map filter is {string.Join(",", ActiveFilterMap.GetGroup())}." : "Map filter is off.")}");
|
||||||
|
|
||||||
|
// Map was not yet initialized.
|
||||||
|
// Get stations from Copri
|
||||||
|
Log.ForContext<MapPageViewModel>().Verbose("No pins detected on page.");
|
||||||
|
if (stations.CopriVersion < CopriCallsStatic.UnsupportedVersionLower)
|
||||||
|
{
|
||||||
|
await ViewService.DisplayAlert(
|
||||||
|
AppResources.MessageWaring,
|
||||||
|
string.Format(AppResources.MessageCopriVersionIsOutdated, ContactPageViewModel.GetAppName(TinkApp.Uris.ActiveUri)),
|
||||||
|
AppResources.MessageAnswerOk);
|
||||||
|
|
||||||
|
Log.ForContext<MapPageViewModel>().Error($"Outdated version of app detected. Version expected is {stations.CopriVersion}.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stations.CopriVersion >= CopriCallsStatic.UnsupportedVersionUpper)
|
||||||
|
{
|
||||||
|
await ViewService.DisplayAlert(
|
||||||
|
AppResources.MessageWaring,
|
||||||
|
string.Format(AppResources.MessageAppVersionIsOutdated, ContactPageViewModel.GetAppName(TinkApp.Uris.ActiveUri)),
|
||||||
|
AppResources.MessageAnswerOk);
|
||||||
|
|
||||||
|
Log.ForContext<MapPageViewModel>().Error($"Outdated version of app detected. Version expected is {stations.CopriVersion}.");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set pins to their positions on map.
|
||||||
|
InitializePins(stations);
|
||||||
|
|
||||||
|
Log.ForContext<MapPageViewModel>().Verbose("Update of pins done.");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Moves the map to the current position of the user.
|
||||||
|
/// If location permission hasn't been granted, the position is not adjusted.
|
||||||
|
/// </summary>
|
||||||
|
private async Task MoveMapToCurrentPositionOfUser()
|
||||||
|
{
|
||||||
|
|
||||||
|
Status status = await RequestLocationPermission();
|
||||||
|
if (status == Status.Granted)
|
||||||
|
{
|
||||||
|
Location currentLocation = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
currentLocation = TinkApp.CenterMapToCurrentLocation
|
||||||
|
? await GeolocationService.GetAsync()
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Log.ForContext<MapPageViewModel>().Error("Getting location failed. {Exception}", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
MoveAndScale(m_oMoveToRegionDelegate, TinkApp.Uris.ActiveUri, ActiveFilterMap, currentLocation);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Requests the location permission from the user.
|
||||||
|
/// If the user declines, a dialog prompot is shown, telling the user to toggle the permission in the device settings.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The permission status.</returns>
|
||||||
|
private async Task<Status> RequestLocationPermission()
|
||||||
|
{
|
||||||
// Check location permission
|
// Check location permission
|
||||||
var status = await PermissionsService.CheckStatusAsync();
|
var status = await PermissionsService.CheckStatusAsync();
|
||||||
if (TinkApp.CenterMapToCurrentLocation
|
if (TinkApp.CenterMapToCurrentLocation
|
||||||
|
@ -339,146 +503,10 @@ namespace TINK.ViewModel.Map
|
||||||
ActionText = "";
|
ActionText = "";
|
||||||
IsRunning = false;
|
IsRunning = false;
|
||||||
IsMapPageEnabled = true;
|
IsMapPageEnabled = true;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return status;
|
||||||
ActionText = AppResources.ActivityTextMapLoadingStationsAndBikes;
|
|
||||||
IsConnected = TinkApp.GetIsConnected();
|
|
||||||
var resultStationsAndBikes = await TinkApp.GetConnector(IsConnected).Query.GetBikesAndStationsAsync();
|
|
||||||
|
|
||||||
TinkApp.Stations = resultStationsAndBikes.Response.StationsAll;
|
|
||||||
|
|
||||||
if (Pins.Count > 0 && Pins.Count != resultStationsAndBikes.Response.StationsAll.Count)
|
|
||||||
{
|
|
||||||
// Either
|
|
||||||
// - user logged in/ logged out which might lead to more/ less stations beeing available
|
|
||||||
// - new stations were added/ existing ones remove
|
|
||||||
Pins.Clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if there are alreay any pins to the map
|
|
||||||
// i.e detecte first call of member OnAppearing after construction
|
|
||||||
if (Pins.Count <= 0)
|
|
||||||
{
|
|
||||||
Log.ForContext<MapPageViewModel>().Debug($"{(ActiveFilterMap.GetGroup().Any() ? $"Active map filter is {string.Join(",", ActiveFilterMap.GetGroup())}." : "Map filter is off.")}");
|
|
||||||
|
|
||||||
// Map was not yet initialized.
|
|
||||||
// Get stations from Copri
|
|
||||||
Log.ForContext<MapPageViewModel>().Verbose("No pins detected on page.");
|
|
||||||
if (resultStationsAndBikes.Response.StationsAll.CopriVersion < CopriCallsStatic.UnsupportedVersionLower)
|
|
||||||
{
|
|
||||||
await ViewService.DisplayAlert(
|
|
||||||
AppResources.MessageWaring,
|
|
||||||
string.Format(AppResources.MessageCopriVersionIsOutdated, ContactPageViewModel.GetAppName(TinkApp.Uris.ActiveUri)),
|
|
||||||
AppResources.MessageAnswerOk);
|
|
||||||
|
|
||||||
Log.ForContext<MapPageViewModel>().Error($"Outdated version of app detected. Version expected is {resultStationsAndBikes.Response.StationsAll.CopriVersion}.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (resultStationsAndBikes.Response.StationsAll.CopriVersion >= CopriCallsStatic.UnsupportedVersionUpper)
|
|
||||||
{
|
|
||||||
await ViewService.DisplayAlert(
|
|
||||||
AppResources.MessageWaring,
|
|
||||||
string.Format(AppResources.MessageAppVersionIsOutdated, ContactPageViewModel.GetAppName(TinkApp.Uris.ActiveUri)),
|
|
||||||
AppResources.MessageAnswerOk);
|
|
||||||
|
|
||||||
Log.ForContext<MapPageViewModel>().Error($"Outdated version of app detected. Version expected is {resultStationsAndBikes.Response.StationsAll.CopriVersion}.");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set pins to their positions on map.
|
|
||||||
InitializePins(resultStationsAndBikes.Response.StationsAll);
|
|
||||||
|
|
||||||
Log.ForContext<MapPageViewModel>().Verbose("Update of pins done.");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (resultStationsAndBikes.Exception?.GetType() == typeof(AuthcookieNotDefinedException))
|
|
||||||
{
|
|
||||||
Log.ForContext<MapPageViewModel>().Error("Map page is shown (probable for the first time after startup of app) and COPRI auth cookie is not defined. {@l_oException}", resultStationsAndBikes.Exception);
|
|
||||||
|
|
||||||
// COPRI reports an auth cookie error.
|
|
||||||
await ViewService.DisplayAlert(
|
|
||||||
AppResources.MessageWaring,
|
|
||||||
AppResources.MessageMapPageErrorAuthcookieUndefined,
|
|
||||||
AppResources.MessageAnswerOk);
|
|
||||||
|
|
||||||
await TinkApp.GetConnector(IsConnected).Command.DoLogout();
|
|
||||||
TinkApp.ActiveUser.Logout();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update pin colors.
|
|
||||||
Log.ForContext<MapPageViewModel>().Verbose("Starting update pins color...");
|
|
||||||
|
|
||||||
var colors = GetStationColors(
|
|
||||||
Pins.Select(x => x.Tag.ToString()).ToList(),
|
|
||||||
resultStationsAndBikes.Response.Bikes);
|
|
||||||
|
|
||||||
// Update pins color form count of bikes located at station.
|
|
||||||
UpdatePinsColor(colors);
|
|
||||||
|
|
||||||
// Move and scale before getting stations and bikes which takes some time.
|
|
||||||
ActionText = AppResources.ActivityTextCenterMap;
|
|
||||||
|
|
||||||
moveMapToCurrentPositionOfUser();
|
|
||||||
|
|
||||||
m_oViewUpdateManager = CreateUpdateTask();
|
|
||||||
|
|
||||||
Log.ForContext<MapPageViewModel>().Verbose("Update pins color done.");
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// Update bikes at station or my bikes depending on context.
|
|
||||||
await m_oViewUpdateManager.StartUpdateAyncPeridically(Polling);
|
|
||||||
}
|
|
||||||
catch (Exception)
|
|
||||||
{
|
|
||||||
// Excpetions are handled insde update task;
|
|
||||||
}
|
|
||||||
|
|
||||||
Exception = resultStationsAndBikes.Exception;
|
|
||||||
ActionText = "";
|
|
||||||
IsRunning = false;
|
|
||||||
IsMapPageEnabled = true;
|
|
||||||
}
|
|
||||||
catch (Exception l_oException)
|
|
||||||
{
|
|
||||||
Log.ForContext<MapPageViewModel>().Error($"An error occurred switching view TINK/ Konrad.\r\n{l_oException.Message}");
|
|
||||||
|
|
||||||
IsRunning = false;
|
|
||||||
|
|
||||||
await ViewService.DisplayAlert(
|
|
||||||
"Fehler",
|
|
||||||
$"Beim Anzeigen der Fahrradstandorte- Seite ist ein Fehler aufgetreten.\r\n{l_oException.Message}",
|
|
||||||
"OK");
|
|
||||||
|
|
||||||
IsMapPageEnabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Moves the map to the current position of the user.
|
|
||||||
/// If location permission hasn't been granted, the position is not adjusted.
|
|
||||||
/// </summary>
|
|
||||||
private async void moveMapToCurrentPositionOfUser()
|
|
||||||
{
|
|
||||||
var status = await PermissionsService.CheckStatusAsync();
|
|
||||||
if (status != Status.Granted) return;
|
|
||||||
Location currentLocation = null;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
currentLocation = TinkApp.CenterMapToCurrentLocation
|
|
||||||
? await GeolocationService.GetAsync()
|
|
||||||
: null;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log.ForContext<MapPageViewModel>().Error("Getting location failed. {Exception}", ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
MoveAndScale(m_oMoveToRegionDelegate, TinkApp.Uris.ActiveUri, ActiveFilterMap, currentLocation);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary> Moves map and scales visible region depending on active filter. </summary>
|
/// <summary> Moves map and scales visible region depending on active filter. </summary>
|
||||||
|
|
Loading…
Reference in a new issue