BankProject/Client/Context/SessionContext.cs
GabrielTofvesson 4531f6244f * Removed deprecated text-render computation
* Added accounts types (savings and checking)
  * FS flush optimized balance computation
  * Automatic daily rate growth computations
2018-05-16 18:37:02 +02:00

484 lines
22 KiB
C#

using Client.ConsoleForms;
using Client.ConsoleForms.Events;
using Client.ConsoleForms.Graphics;
using Client.ConsoleForms.Parameters;
using Client.Properties;
using ConsoleForms;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Tofvesson.Collections;
using Tofvesson.Common;
using Tofvesson.Crypto;
namespace Client
{
public sealed class SessionContext : Context
{
private readonly BankNetInteractor interactor;
private bool scheduleDestroy;
private Promise userDataGetter;
private Promise accountsGetter;
private List<string> accounts = null;
private string username;
private bool isAdministrator = false;
// Stores personal accounts
private readonly FixedQueue<Tuple<string, Account>> accountDataCache = new FixedQueue<Tuple<string, Account>>(64);
// Stores remote account data
private readonly FixedQueue<Tuple<string, string>> remoteUserCache = new FixedQueue<Tuple<string, string>>(8);
private bool accountChange = false;
public SessionContext(ContextManager manager, BankNetInteractor interactor) : base(manager, "Session", "Common")
{
this.interactor = interactor;
scheduleDestroy = !interactor.IsLoggedIn;
RegisterAutoHide("account_create", "account_info", "password_update", "exit_prompt", "account_show", "transfer");
GetView<DialogView>("Success").RegisterSelectListener((v, i, s) => HandleLogout());
// Menu option setup
ListView options = GetView<ListView>("menu_options");
options.GetView<ButtonView>("exit").SetEvent(v => Show("exit_prompt"));
void SubmitListener(View listener)
{
ButtonView view = listener as ButtonView;
void ShowAccountData(string name, decimal balance, Account.AccountType type)
{
// Build dialog view manually
var show = new DialogView(
new ViewData("DialogView")
// Layout parameters
.SetAttribute("padding_left", 2)
.SetAttribute("padding_right", 2)
.SetAttribute("padding_top", 1)
.SetAttribute("padding_bottom", 1)
.SetAttribute("border", (int)ConsoleColor.DarkGreen)
// Option buttons
.AddNested(new ViewData("Options").AddNestedSimple("Option", GetIntlString("GENERIC_dismiss")))
// Message
.AddNestedSimple("Text",
GetIntlString("SE_info")
.Replace("$0", name)
.Replace("$1", GetIntlString(type == Account.AccountType.Savings ? "SE_acc_saving" : "SE_acc_checking"))
.Replace("$2", balance.ToTruncatedString())),
// No translation (it's already handled)
LangManager.NO_LANG);
show.RegisterSelectListener((_, s, l) => Hide(show));
Show(show);
}
// TODO: Show account info
var account = AccountLookup(view.Text);
if (account == null)
{
// TODO: Get account data from server + cache data
Show("data_fetch");
Promise info_promise = Promise.AwaitPromise(interactor.AccountInfo(view.Text));
info_promise.Subscribe = evt =>
{
Hide("data_fetch");
if (evt.Value.StartsWith("ERROR") || !Account.TryParse(evt.Value, out var act))
controller.Popup(GetIntlString("GENERIC_error"), 3000, ConsoleColor.Red);
else
{
// Cache result (don't cache savings accounts because their value updates pretty frequently)
if (act.type!=Account.AccountType.Savings) accountDataCache.Enqueue(new Tuple<string, Account>(view.Text, act));
ShowAccountData(view.Text, act.balance, act.type);
}
};
}
else ShowAccountData(account.Item1, account.Item2.balance, account.Item2.type);
}
options.GetView<ButtonView>("view").SetEvent(v =>
{
if (accountChange) RefreshAccountList();
if (!accountsGetter.HasValue) Show("data_fetch");
accountsGetter.Subscribe = p =>
{
accountsGetter.Unsubscribe();
Hide("data_fetch");
Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), SubmitListener));
};
});
GetView<InputView>("password_update").SubmissionsListener = v =>
{
bool hasError = v.Inputs[0].Text.Length == 0;
if (hasError)
{
// Notify user, as well as mark the errant input field
v.Inputs[0].SelectBackgroundColor = ConsoleColor.Red;
v.Inputs[0].BackgroundColor = ConsoleColor.DarkRed;
controller.Popup(GetIntlString("ERR_empty"), 3000, ConsoleColor.Red);
}
if(v.Inputs[1].Text.Length == 0)
{
v.Inputs[1].SelectBackgroundColor = ConsoleColor.Red;
v.Inputs[1].BackgroundColor = ConsoleColor.DarkRed;
if(!hasError) controller.Popup(GetIntlString("ERR_empty"), 3000, ConsoleColor.Red);
return; // No need to continue, we have notified the user. There is no valid information to operate on past this point
}
if (!v.Inputs[0].Text.Equals(v.Inputs[1].Text))
{
controller.Popup(GetIntlString("SU_mismatch"), 3000, ConsoleColor.Red);
return;
}
Show("update_stall");
Task<Promise> t = interactor.UpdatePassword(v.Inputs[0].Text);
Promise.AwaitPromise(t).Subscribe = p =>
{
Hide("update_stall");
Hide("password_update");
v.Inputs[0].ClearText();
v.Inputs[1].ClearText();
v.SelectedField = 0;
};
};
options.GetView<ButtonView>("delete").SetEvent(v => Show("account_delete"));
GetView<DialogView>("account_delete").RegisterSelectListener((v, i, s) =>
{
Hide(v);
if (i == 1)
{
Show("delete_stall");
Promise deletion = Promise.AwaitPromise(interactor.DeleteUser());
deletion.Subscribe = p =>
{
Hide("delete_stall");
if (bool.Parse(p.Value))
controller.Popup(GetIntlString("SE_delete_success"), 2500, ConsoleColor.Green, () => manager.LoadContext(new NetContext(manager)));
else
controller.Popup(GetIntlString("SE_delete_failure"), 1500, ConsoleColor.Red);
};
}
});
// Actual "create account" input box thingy
var input = GetView<InputView>("account_create");
int accountType = -1;
ListView accountTypes = null;
accountTypes = GenerateList(
new string[] {
GetIntlString("SE_acc_checking"),
GetIntlString("SE_acc_saving")
}, v =>
{
accountType = accountTypes.SelectedView;
input.Inputs[1].Text = (v as ButtonView).Text;
CreateAccount();
}, true);
input.Inputs[1].Text = GetIntlString("SE_acc_sel");
input.SubmissionsListener = inputView =>
{
if (inputView.SelectedField == 1)
Show(accountTypes); // Show account type selection menu
else CreateAccount();
};
void CreateAccount()
{
bool hasError = false;
if (accountType == -1)
{
hasError = true;
input.Inputs[1].SelectBackgroundColor = ConsoleColor.Red;
input.Inputs[1].BackgroundColor = ConsoleColor.DarkRed;
controller.Popup(GetIntlString("SE_acc_nosel"), 2500, ConsoleColor.Red);
}
if (input.Inputs[0].Text.Length == 0)
{
input.Inputs[0].SelectBackgroundColor = ConsoleColor.Red;
input.Inputs[0].BackgroundColor = ConsoleColor.DarkRed;
if(!hasError) controller.Popup(GetIntlString("ERR_empty"), 3000, ConsoleColor.Red);
}
else
{
void AlreadyExists()
=> controller.Popup(GetIntlString("SE_account_exists").Replace("$0", input.Inputs[0].Text), 2500, ConsoleColor.Red, () => Hide(input));
var act = AccountLookup(input.Inputs[0].Text);
if (act != null) AlreadyExists();
else
{
Show("account_stall");
Promise accountPromise = Promise.AwaitPromise(interactor.CreateAccount(input.Inputs[0].Text, accountType==0));
accountPromise.Subscribe = p =>
{
if (bool.Parse(p.Value))
{
controller.Popup(GetIntlString("SE_account_success"), 750, ConsoleColor.Green, () => Hide(input));
accountChange = true;
}
else AlreadyExists();
Hide("account_stall");
};
}
}
}
input.InputListener = (v, c, i, t) =>
{
c.BackgroundColor = v.DefaultBackgroundColor;
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
if (v.IndexOf(c) == 1)
{
Show(accountTypes);
return false; // Don't process key event
}
return true;
};
options.GetView<ButtonView>("add").SetEvent(_ => Show(input));
// Set up a listener to reset color scheme
GetView<InputView>("password_update").InputListener = (v, c, i, t) =>
{
c.BackgroundColor = v.DefaultBackgroundColor;
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
return true;
};
// Update password
options.GetView<ButtonView>("update").SetEvent(v => Show("password_update"));
string acc1 = null, acc2 = null, user = null;
options.GetView<ButtonView>("tx").SetEvent(v =>
{
var txView = GetView<InputView>("transfer");
txView.Inputs[0].Text = GetIntlString("SE_account_select");
txView.Inputs[1].Text = GetIntlString("SE_user_select");
txView.Inputs[2].Text = GetIntlString("SE_account_select");
Show(txView);
});
GetView<InputView>("transfer").SubmissionsListener = v =>
{
switch (v.SelectedField)
{
case 0:
if (accountChange) accountsGetter = Promise.AwaitPromise(interactor.ListUserAccounts());
Show("data_fetch");
accountsGetter.Subscribe = p =>
{
accountsGetter.Unsubscribe();
Hide("data_fetch");
Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), sel => v.Inputs[0].Text = acc1 = (sel as ButtonView).Text, true));
};
break;
case 1:
Show("data_fetch");
Promise remoteUserGetter = Promise.AwaitPromise(interactor.ListUsers());
remoteUserGetter.Subscribe = p =>
{
remoteUserGetter.Unsubscribe();
Hide("data_fetch");
Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), sel => v.Inputs[1].Text = user = (sel as ButtonView).Text, true));
};
break;
case 2:
if (user == null)
controller.Popup(GetIntlString("SE_user_noselect"), 2000, ConsoleColor.Red);
else
{
Show("data_fetch");
Promise remoteAccountsGetter = Promise.AwaitPromise(interactor.ListAccounts(user));
remoteAccountsGetter.Subscribe = p =>
{
remoteAccountsGetter.Unsubscribe();
Hide("data_fetch");
Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), sel => v.Inputs[2].Text = acc2 = (sel as ButtonView).Text, true));
};
}
break;
case 3:
case 4:
Show("verify_stall");
bool error = false;
if (acc1==null)
{
controller.Popup(GetIntlString("SE_account_noselect"), 1500, ConsoleColor.Red);
error = true;
v.Inputs[0].BackgroundColor = ConsoleColor.Red;
v.Inputs[0].SelectBackgroundColor = ConsoleColor.DarkRed;
}
if (acc2 == null)
{
if(!error) controller.Popup(GetIntlString("SE_account_noselect"), 1500, ConsoleColor.Red);
error = true;
v.Inputs[2].BackgroundColor = ConsoleColor.Red;
v.Inputs[2].SelectBackgroundColor = ConsoleColor.DarkRed;
}
if(user == null)
{
if(!error) controller.Popup(GetIntlString("SE_account_nouser"), 1500, ConsoleColor.Red);
error = true;
v.Inputs[1].BackgroundColor = ConsoleColor.DarkRed;
v.Inputs[1].SelectBackgroundColor = ConsoleColor.Red;
}
userDataGetter = Promise.AwaitPromise(interactor.UserInfo());
userDataGetter.Subscribe = p =>
{
userDataGetter.Unsubscribe();
var account = AccountLookup("SE_balance_toohigh");
if (account == null) accountsGetter = Promise.AwaitPromise(interactor.AccountInfo(acc1));
accountsGetter.Subscribe = result =>
{
accountsGetter.Unsubscribe();
var resultData = p.Value.Split('&');
Hide("verify_stall");
decimal d;
if (result.Value.StartsWith("ERROR") || !Account.TryParse(result.Value, out var act))
controller.Popup(GetIntlString("GENERIC_error"), 1500, ConsoleColor.Red);
else if ((d = decimal.Parse(v.Inputs[3].Text)) > act.balance && (!bool.Parse(resultData[1]) || !acc1.Equals(acc2)))
controller.Popup(GetIntlString("SE_balance_toohigh").Replace("$0", act.balance.ToString()), 3000, ConsoleColor.Red);
else
{
Promise txPromise = Promise.AwaitPromise(interactor.CreateTransaction(acc1, user, acc2, d, v.Inputs[4].Text.Length == 0 ? null : v.Inputs[4].Text));
accountChange = true;
accountDataCache.Clear();
txPromise.Subscribe = txResult =>
{
if (txResult.Value.StartsWith("ERROR"))
controller.Popup(GetIntlString("GENERIC_error"), 1500, ConsoleColor.Red);
else controller.Popup(GetIntlString("SE_tx_success"), 2000, ConsoleColor.Green, () => Hide("transfer"));
};
}
};
};
break;
}
};
GetView<InputView>("transfer").InputListener = (v, i, s, t) =>
{
if (t) return false; // Don't handle artificial events
i.BackgroundColor = v.DefaultBackgroundColor;
i.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
if (v.IndexOf(i) < 3)
{
// Trigger a keypress event for key [ENTER]
v.TriggerKeyEvent(new ConsoleController.KeyEvent(new ConsoleKeyInfo('\n', ConsoleKey.Enter, false, false, false)));
return false; // Don't update input
}
return true;
};
options.OnBackEvent = v => Show("exit_prompt");
GetView<DialogView>("exit_prompt").RegisterSelectListener((v, i, s) =>
{
if (i == 0) Hide("exit_prompt");
else HandleLogout();
});
if (!scheduleDestroy)
{
// We have a valid context!
RefreshUserInfo(); // Get user info
RefreshAccountList(); // Get account list for user
}
}
private ListView GenerateList(string[] data, SubmissionEvent onclick, bool exitOnSubmit = false, bool hideOnBack = true)
{
//var list = GetView<ListView>("account_show");
var list = new ListView(new ViewData("ListView").SetAttribute("padding_left", 2).SetAttribute("padding_right", 2).SetAttribute("border", 8), LangManager.NO_LANG);
if (data.Length == 1 && data[0].Length == 0) return list;
bool b = data.Length == 1 && data[0].Length == 0;
Tuple<string, View>[] listData = new Tuple<string, View>[data.Length - (b ? 1 : 0)];
if (!b)
for (int i = 0; i < listData.Length; ++i)
{
ButtonView t = new ButtonView(new ViewData("ButtonView").AddNestedSimple("Text", data[i]), LangManager.NO_LANG); // Don't do translations
t.SetEvent(v =>
{
onclick?.Invoke(v);
if (exitOnSubmit) Hide(list);
});
listData[i] = new Tuple<string, View>(t.Text, t);
}
list.AddViews(0, listData); // Insert generated buttons before predefined "close" button
if (hideOnBack) list.OnBackEvent = v => Hide(v);
return list;
}
private void RefreshAccountList()
{
accountsGetter = Promise.AwaitPromise(interactor.ListUserAccounts()); // Get accounts associated with this user
accountsGetter.Subscribe = p =>
{
var data = p.Value.Split('&');
accounts = new List<string>();
accounts.AddRange(data);
};
}
private void RefreshUserInfo()
{
userDataGetter = Promise.AwaitPromise(interactor.UserInfo()); // Get basic user info
userDataGetter.Subscribe = p =>
{
var data = p.Value.Split('&');
username = data[0].FromBase64String();
isAdministrator = bool.Parse(data[1]);
};
}
private void HandleLogout(bool automatic = false)
{
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
interactor.Logout();
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
controller.Popup(GetIntlString($"SE_{(automatic ? "auto" : "")}lo"), 2500, ConsoleColor.DarkMagenta, () => manager.LoadContext(new NetContext(manager)));
}
private Tuple<string, Account> AccountLookup(string name)
{
foreach (var cacheEntry in accountDataCache)
if (cacheEntry.Item1.Equals(name))
return cacheEntry;
return null;
}
public override void OnCreate()
{
if (scheduleDestroy) manager.LoadContext(new WelcomeContext(manager, interactor));
else Show("menu_options");
}
public override void OnDestroy()
{
base.HideAll();
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
interactor.CancelAll();
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
}
}
}