BankProject/Client/Context/WelcomeContext.cs
GabrielTofvesson 100f5a32be Major changes
Refactorings:
  * BinaryCollector -> BitWriter
  * BinaryDistributor -> BitReader

Additions:
  * Output class for making serverside output pretty and more readable
  * Better RSA keys (private keys withheld)

Changes:
  * Minor changes to all views and their rendering
  * Added corrective resizing to resize listener to prevent errant window sizes
  * Removed "default" language in favour of a purely priority-based system
  * NetContext now attempts to verify server identity before continuing to next context
  * Simplified common operations in Context
  * Minor updates to some layouts
  * Completed translations for english and swedish
  * Promise system now supports internal processing before notifying original caller
  * Bank interactor methods are now async
  * Added support for multiple accounts per user (separate repositories for money)
  * Removed test code from client program
  * Updated Database to support multiple accounts
  * Reimplemented RSA on the server side purely as an identity verification system on top of the networking layer (rather than part of the layer)
  * Added Account management endpoints
  * Added full support for System-sourced transactions
  * Added Account availability endpoint
  * Added verbose error responses
2018-04-26 00:24:58 +02:00

177 lines
6.4 KiB
C#

using Client.ConsoleForms;
using Client.ConsoleForms.Graphics;
using ConsoleForms;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Tofvesson.Collections;
namespace Client
{
public sealed class WelcomeContext : Context
{
private readonly BankNetInteractor interactor;
private long token;
private Promise promise;
private bool forceDestroy = true;
public WelcomeContext(ContextManager manager, BankNetInteractor connection) : base(manager, "Setup", "Common")
{
this.interactor = connection;
// Prepare events and stuff
// Just close when anything is selected and "submitted"
RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError");
GetView<InputView>("Login").SubmissionsListener = i =>
{
bool success = true;
foreach (var input in i.Inputs)
if (input.Text.Length == 0)
{
success = false;
input.SelectBackgroundColor = ConsoleColor.Red;
input.BackgroundColor = ConsoleColor.DarkRed;
}
if (success)
{
// Authenticate against server here
Show("AuthWait");
Task<Promise> prom = interactor.Authenticate(i.Inputs[0].Text, i.Inputs[1].Text);
if(!prom.IsCompleted) prom.RunSynchronously();
promise = prom.Result;
promise.Subscribe =
response =>
{
Hide("AuthWait");
if (response.Value.Equals("ERROR"))
Show("AuthError");
else
{
forceDestroy = false;
manager.LoadContext(new SessionContext(manager, interactor, response.Value));
}
};
}
else Show("EmptyFieldError");
};
// For a smooth effect
GetView<InputView>("Login").InputListener = (v, c, i) =>
{
c.BackgroundColor = v.DefaultBackgroundColor;
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
return true;
};
GetView<InputView>("Register").SubmissionsListener = i =>
{
bool success = true, mismatch = false;
foreach (var input in i.Inputs)
{
if (input.Text.Length == 0)
{
success = false;
input.SelectBackgroundColor = ConsoleColor.Red;
input.BackgroundColor = ConsoleColor.DarkRed;
}
}
mismatch = !i.Inputs[1].Text.Equals(i.Inputs[2].Text);
if (success && !mismatch)
{
void a()
{
Show("RegWait");
promise = Promise.AwaitPromise(interactor.Register(i.Inputs[0].Text, i.Inputs[1].Text));
promise.Subscribe =
response =>
{
Hide("RegWait");
if (response.Value.Equals("ERROR"))
Show("DuplicateAccountError");
else
{
forceDestroy = false;
manager.LoadContext(new SessionContext(manager, interactor, response.Value));
}
};
}
if (i.Inputs[1].Text.Length < 5 || i.Inputs[1].Text.StartsWith("asdfasdf") || i.Inputs[1].Text.StartsWith("asdf1234"))
{
var warning = GetView<DialogView>("WeakPasswordWarning");
warning.RegisterSelectListener((wrn, idx, sel) =>
{
Hide(warning);
if (idx == 0) a();
});
Show(warning);
}
else a();
}
else if (mismatch) Show("PasswordMismatchError");
else Show("EmptyFieldError");
};
((InputView)views.GetNamed("Register")).InputListener = (v, c, i) =>
{
c.BackgroundColor = v.DefaultBackgroundColor;
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
return true;
};
}
public override void OnCreate()
{
token = interactor.RegisterListener((c, s) =>
{
if(!s) controller.Popup("The connection to the server was severed! ", 4500, ConsoleColor.DarkRed, () => manager.LoadContext(new NetContext(manager)));
});
// Add the initial view
Show("WelcomeScreen");
}
public override void OnDestroy()
{
GetView<InputView>("Register").SelectedField = 0;
foreach (var v in GetView<InputView>("Register").Inputs)
{
v.Text = "";
v.SelectIndex = 0;
v.RenderStart = 0;
}
((InputView)views.GetNamed("Login")).SelectedField = 0;
foreach (var v in GetView<InputView>("Login").Inputs)
{
v.Text = "";
v.SelectIndex = 0;
v.RenderStart = 0;
}
// Close views
HideAll();
// Unsubscribe from events
if (promise != null && !promise.HasValue) promise.Subscribe = null;
// Stop listening
interactor.UnregisterListener(token);
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
if (forceDestroy) interactor.CancelAll();
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
}
}
}