diff --git a/Client/BankNetInteractor.cs b/Client/BankNetInteractor.cs index 6849bb3..752ea79 100644 --- a/Client/BankNetInteractor.cs +++ b/Client/BankNetInteractor.cs @@ -139,6 +139,17 @@ namespace Client }); } + public async virtual Task DeleteUser() + { + await StatusCheck(true); + client.Send(CreateCommandMessage("RmUsr", sessionID, out var pID)); + return RegisterEventPromise(pID, p => + { + PostPromise(p.handler, !p.Value.StartsWith("ERROR")); + return false; + }); + } + public async virtual Task UpdatePassword(string newPass) { await StatusCheck(true); diff --git a/Client/ConsoleForms/Graphics/DialogView.cs b/Client/ConsoleForms/Graphics/DialogView.cs index 7f3ab3e..d752986 100644 --- a/Client/ConsoleForms/Graphics/DialogView.cs +++ b/Client/ConsoleForms/Graphics/DialogView.cs @@ -67,7 +67,7 @@ namespace Client.ConsoleForms.Graphics int pl = padding.Left(), pr = padding.Right(); Console.SetCursorPosition(left, top++); - int pad = MaxWidth - options.CollectiveLength() - options.Length;// + pl + pr; + int pad = ContentWidth - options.CollectiveLength() - options.Length;// + pl + pr; int lpad = (int)(pad / 2f); Console.BackgroundColor = BackgroundColor; Console.Write(Filler(' ', lpad)); diff --git a/Client/ConsoleForms/Graphics/InputView.cs b/Client/ConsoleForms/Graphics/InputView.cs index 8df317f..5fe161d 100644 --- a/Client/ConsoleForms/Graphics/InputView.cs +++ b/Client/ConsoleForms/Graphics/InputView.cs @@ -32,6 +32,7 @@ namespace Client.ConsoleForms.Graphics } } private string[][] splitInputs; + protected ViewData data; public InputView(ViewData parameters, LangManager lang) : base(parameters, lang) @@ -47,6 +48,8 @@ namespace Client.ConsoleForms.Graphics DefaultSelectBackgroundColor = (ConsoleColor)sBC; DefaultSelectTextColor = (ConsoleColor)sTC; + this.data = parameters; + List fields = new List(); foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData) if (!data.Name.Equals("Field")) continue; @@ -63,14 +66,19 @@ namespace Client.ConsoleForms.Graphics Inputs = fields.ToArray(); + int max = ContentWidth; int computedSize = 0; splitInputs = new string[Inputs.Length][]; for (int i = 0; i < Inputs.Length; ++i) { splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' ')); + foreach (var input in splitInputs[i]) + if (input.Length > max) + max = input.Length; computedSize += splitInputs[i].Length; } ContentHeight += computedSize + Inputs.Length * 2; + if (ContentWidth < max) ContentWidth = max; } public int IndexOf(InputField field) @@ -98,7 +106,7 @@ namespace Client.ConsoleForms.Graphics { Console.SetCursorPosition(left, top++); Console.BackgroundColor = BackgroundColor; - Console.Write(splitInputs[j][i] + Filler(' ', MaxWidth - splitInputs[j][i].Length)); + Console.Write(splitInputs[j][i] + Filler(' ', ContentWidth - splitInputs[j][i].Length)); } Console.SetCursorPosition(left, top++); @@ -114,10 +122,10 @@ namespace Client.ConsoleForms.Graphics if (Inputs[j].SelectIndex < Inputs[j].Text.Length) Console.Write( Inputs[j].ShowText ? - Inputs[j].Text.Substring(Inputs[j].SelectIndex + 1, drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) : - Filler('*', drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) + Inputs[j].Text.Substring(Inputs[j].SelectIndex + 1, drawn = Math.Min(ContentWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) : + Filler('*', drawn = Math.Min(ContentWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) ); - Console.Write(Filler(' ', maxWidth - 1 - drawn - Inputs[j].SelectIndex + Inputs[j].RenderStart)); + Console.Write(Filler(' ', ContentWidth - 1 - drawn - Inputs[j].SelectIndex + Inputs[j].RenderStart)); Console.ForegroundColor = ConsoleColor.Black; } } @@ -140,7 +148,7 @@ namespace Client.ConsoleForms.Graphics case ConsoleKey.RightArrow: if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length) { - if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; + if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == ContentWidth) ++Inputs[selectedField].RenderStart; } else return changed; break; @@ -175,6 +183,7 @@ namespace Client.ConsoleForms.Graphics else return changed; break; case ConsoleKey.Enter: + ParseAction(data)(); SubmissionsListener?.Invoke(this); return changed; case ConsoleKey.Escape: @@ -184,7 +193,7 @@ namespace Client.ConsoleForms.Graphics { if (InputListener?.Invoke(this, Inputs[selectedField], info, triggered) == false) break; Inputs[selectedField].Text = Inputs[selectedField].Text.Substring(0, Inputs[selectedField].SelectIndex) + info.KeyChar + Inputs[selectedField].Text.Substring(Inputs[selectedField].SelectIndex); - if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; + if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == ContentWidth) ++Inputs[selectedField].RenderStart; } else return changed; break; diff --git a/Client/ConsoleForms/Graphics/TextView.cs b/Client/ConsoleForms/Graphics/TextView.cs index 22accf3..f2aa31f 100644 --- a/Client/ConsoleForms/Graphics/TextView.cs +++ b/Client/ConsoleForms/Graphics/TextView.cs @@ -32,29 +32,7 @@ namespace Client.ConsoleForms.Graphics Dirty = true; } } - - public int MaxWidth - { - get => maxWidth; - - set - { - maxWidth = value; - text_render = ComputeTextDimensions(text); - Dirty = true; - } - } - public int MaxHeight - { - get => maxHeight; - - set - { - maxHeight = value; - text_render = ComputeTextDimensions(text); - Dirty = true; - } - } + public override Region Occlusion => new Region( new Rectangle( -padding.Left() - (DrawBorder ? 2 : 0), // Left bound @@ -231,7 +209,7 @@ namespace Client.ConsoleForms.Graphics for (int i = 0; i < text_render.Length; ++i) { Console.SetCursorPosition(left, top++); - Console.Write(/*Filler(' ', pl) + */text_render[i] + Filler(' ', MaxWidth - text_render[i].Length)/* + Filler(' ', pr)*/); + Console.Write(/*Filler(' ', pl) + */text_render[i] + Filler(' ', ContentWidth - text_render[i].Length)/* + Filler(' ', pr)*/); } } @@ -243,7 +221,7 @@ namespace Client.ConsoleForms.Graphics { Console.SetCursorPosition(left, top++); Console.BackgroundColor = BackgroundColor; - Console.Write(Filler(' ', maxWidth/* + pl + pr*/)); + Console.Write(Filler(' ', ContentWidth/* + pl + pr*/)); } } } diff --git a/Client/ConsoleForms/Graphics/View.cs b/Client/ConsoleForms/Graphics/View.cs index beded5c..b3870f2 100644 --- a/Client/ConsoleForms/Graphics/View.cs +++ b/Client/ConsoleForms/Graphics/View.cs @@ -139,7 +139,7 @@ namespace Client.ConsoleForms.Graphics } protected EventAction ParseAction(ViewData data) { - bool.TryParse(data.GetAttribute("close"), out bool close); + bool.TryParse(data?.GetAttribute("close")??"", out bool close); return ParseAction(data.GetAttribute("event"), close); } protected EventAction ParseAction(string action, bool close) diff --git a/Client/Context/IntroContext.cs b/Client/Context/IntroContext.cs index 2d725b1..6934145 100644 --- a/Client/Context/IntroContext.cs +++ b/Client/Context/IntroContext.cs @@ -13,56 +13,11 @@ namespace Client public IntroContext(ContextManager manager, Action onComplete) : base(manager, "Intro", "Common") { GetView("welcome").RegisterSelectListener((v, i, s) => - { - if (i == 1) - { - Hide(v); - onComplete(); - } - else - { - Hide(v); - Show("describe1"); - } - }); - - GetView("describe1").RegisterSelectListener((v, i, s) => - { - if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false)); - else - { - Hide(v); - Show("describe2"); - } - }); - - GetView("describe2").RegisterSelectListener((v, i, s) => - { - if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false)); - else - { - Hide(v); - Show("describe3"); - } - }); - - GetView("describe3").SubmissionsListener = v => { Hide(v); - Show("describe4"); - }; - - GetView("describe4").SubmissionsListener = v => - { - Hide(v); - Show("describe4_1"); - }; - - GetView("describe4_1").SubmissionsListener = v => - { - Hide(v); - Show("describe5"); - }; + if (i == 1) onComplete(); + else Show("describe1"); + }); GetView("describe5").RegisterSelectListener((v, i, s) => { @@ -77,22 +32,7 @@ namespace Client }; } - public override void OnCreate() - { - Show("welcome"); - } - - public override void OnDestroy() - { - - } - - // Graphics update trigger - public override bool Update(ConsoleController.KeyEvent keypress, bool hasKeypress = true) - { - - // Return: whether or not to redraw graphics - return base.Update(keypress, hasKeypress); - } + public override void OnCreate() => Show("welcome"); + public override void OnDestroy() { } } } diff --git a/Client/Context/SessionContext.cs b/Client/Context/SessionContext.cs index 5e4e3df..56230bd 100644 --- a/Client/Context/SessionContext.cs +++ b/Client/Context/SessionContext.cs @@ -148,6 +148,26 @@ namespace Client }; }; + options.GetView("delete").SetEvent(v => Show("account_delete")); + + GetView("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("account_create"); input.SubmissionsListener = __ => diff --git a/Client/Context/WelcomeContext.cs b/Client/Context/WelcomeContext.cs index 19656cf..6d8595c 100644 --- a/Client/Context/WelcomeContext.cs +++ b/Client/Context/WelcomeContext.cs @@ -25,6 +25,8 @@ namespace Client // Just close when anything is selected and "submitted" RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError"); + // If Escape key is pressed, suggest to controller to terminate + GetView("WelcomeScreen").OnBackEvent = v => controller.ShouldExit = true; GetView("Login").SubmissionsListener = i => { diff --git a/Client/Resources/Layout/Intro.xml b/Client/Resources/Layout/Intro.xml index 91da77f..d2fb7a1 100644 --- a/Client/Resources/Layout/Intro.xml +++ b/Client/Resources/Layout/Intro.xml @@ -22,7 +22,7 @@ padding_bottom="1" back="Intro:welcome"> - + @string/WS_describe1 @@ -35,8 +35,8 @@ padding_bottom="1" back="Intro:describe1"> - - + + @string/WS_describe2 @@ -47,7 +47,9 @@ padding_right="2" padding_top="1" padding_bottom="1" - back="Intro:describe2"> + back="Intro:describe2" + event="Intro:describe4" + close="true"> @string/WS_input @string/WS_input @@ -62,7 +64,9 @@ padding_right="2" padding_top="1" padding_bottom="1" - back="Intro:describe3"> + back="Intro:describe3" + event="Intro:describe4_1" + close="true"> @string/WS_input @string/WS_input_integer @@ -77,7 +81,9 @@ padding_right="2" padding_top="1" padding_bottom="1" - back="Intro:describe4"> + back="Intro:describe4" + event="Intro:describe5" + close="true"> @string/WS_input_alphanum @string/WS_input_password diff --git a/Client/Resources/Layout/Session.xml b/Client/Resources/Layout/Session.xml index aa84c34..36fcb71 100644 --- a/Client/Resources/Layout/Session.xml +++ b/Client/Resources/Layout/Session.xml @@ -90,12 +90,37 @@ @string/SE_pwdu + + @string/SE_delete + + @string/SE_exit + + @string/SE_delete_stall + + + + + + + + @string/SE_delete_warn + + Transaction history Transfer funds Funds transferred! - Send to + Send to: Account From Account: To Account: @@ -98,6 +98,12 @@ Is this correct? Log out Open an account Close account + Delete user account + WARNING: This will delete the current user and all connected accounts! +Are you sure you would like to continue? + Deleting... + User deleted + User could not be deleted Show accounts Supplied balance is higher than available amount in source account! Available balance: $0 SEK diff --git a/Client/Resources/Strings/en_US/strings.xml b/Client/Resources/Strings/en_US/strings.xml index 9acc3f8..434899d 100644 --- a/Client/Resources/Strings/en_US/strings.xml +++ b/Client/Resources/Strings/en_US/strings.xml @@ -79,7 +79,7 @@ To go back, press [ESCAPE] Transaction history Transfer funds Funds transferred! - Send to + Send to: Account From Account: To Account: @@ -98,6 +98,12 @@ Is this correct? Log out Open an account Close account + Delete user account + WARNING: This will delete the current user and all connected accounts! +Are you sure you would like to continue? + Deleting... + User deleted + User could not be deleted Show accounts Supplied balance is higher than available amount in source account! Available balance: $0 SEK diff --git a/Client/Resources/Strings/sv_SE/strings.xml b/Client/Resources/Strings/sv_SE/strings.xml index 399efa5..df2fb61 100644 --- a/Client/Resources/Strings/sv_SE/strings.xml +++ b/Client/Resources/Strings/sv_SE/strings.xml @@ -84,7 +84,7 @@ För att backa, tryck [ESCAPE] Transaktionshistorik Överför pengar Belopp överfört! - Skicka till + Skicka till: Konto Från konto: Till konto: @@ -103,6 +103,13 @@ Till kontot: $2 Logga ut Öppna ett konto Stäng konto + Radera användare + VARNING: Detta kommer att radera den nuvarande användaren +och alla kopplade konton! +Vill du fortsätta? + Raderar... + Användare raderad + Användare kunde inte raderas Visa konton Angivet belopp är högre än det tillgängliga beloppet i ursprungskontot! Tillgängligt saldo: $0 SEK diff --git a/Server/Database.cs b/Server/Database.cs index 012f5cb..b85ed4a 100644 --- a/Server/Database.cs +++ b/Server/Database.cs @@ -73,17 +73,19 @@ namespace Server public void RemoveUser(User entry) => RemoveUser(entry, true); private void RemoveUser(User entry, bool withFlush) { - entry = ToEncoded(entry); + // Remove from loaded users collection for (int i = 0; i < loadedUsers.Count; ++i) - if (entry.Equals(loadedUsers[i])) + if (entry.Name.Equals(loadedUsers[i].Name)) loadedUsers.RemoveAt(i); + // Changes are retracted from change collectino for (int i = changeList.Count - 1; i >= 0; --i) - if (changeList[i].Equals(entry.Name)) + if (changeList[i].Name.Equals(entry.Name)) changeList.RemoveAt(i); + // Check if user already is scheduled for deletion for (int i = toRemove.Count - 1; i >= 0; --i) - if (toRemove[i].Equals(entry.Name)) + if (toRemove[i].Name.Equals(entry.Name)) return; toRemove.Add(entry); @@ -104,9 +106,10 @@ namespace Server using(var reader = XmlReader.Create(DatabaseName)) { int masterDepth = 0; - bool trigger = false, wn = false, recent = false; - while (wn || reader.Read()) + bool trigger = false, wn = false, recent = false, justTriggered = false; + while (wn || reader.Read() || (reader.NodeType==XmlNodeType.None && justTriggered)) { + justTriggered = false; wn = false; if (trigger) { @@ -114,7 +117,7 @@ namespace Server WriteUser(writer, user); bool wroteNode = false; - while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement) + while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement && reader.NodeType != XmlNodeType.None) { wroteNode = false; if (reader.Name.Equals("User")) @@ -154,6 +157,7 @@ namespace Server if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth])) { trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length; + justTriggered = true; reader.MoveToContent(); writer.WriteStartElement(MasterEntry[masterDepth - 1]); } @@ -241,10 +245,18 @@ namespace Server public User FirstUser(Predicate p) { if (p == null) return null; // Done to conveniently handle system insertions + + // Check if user is scheduled for removal + foreach (var entry in toRemove) + if (p(entry)) + return null; + + // Check loaded users foreach (var entry in loadedUsers) if (p(entry)) return entry; + // Check modified users foreach (var entry in changeList) if (p(entry)) { @@ -252,6 +264,7 @@ namespace Server return entry; } + // Read from database using (var reader = XmlReader.Create(DatabaseName)) { if (!Traverse(reader, MasterEntry)) return null; diff --git a/Server/Program.cs b/Server/Program.cs index 28b5ce2..8e2b8b6 100644 --- a/Server/Program.cs +++ b/Server/Program.cs @@ -147,11 +147,23 @@ Use command 'help' to get a list of available commands"; // Server endpoints switch (cmd[0]) { + case "RmUsr": + { + if (!GetUser(cmd[1], out var user)) + { + if(verbosity > 0) Output.Error($"Could not delete user from session as session isn't valid. (SessionID=\"{cmd[1]}\")"); + return ErrorResponse(id, "badsession"); + } + manager.Expire(user); + db.RemoveUser(user); + if (verbosity > 0) Output.Info($"Removed user \"{user.Name}\" (SessionID={cmd[1]})"); + return GenerateResponse(id, true); + } case "Auth": // Log in to a user account (get a session id) { if(!ParseDataPair(cmd[1], out string user, out string pass)) { - Output.Error($"Recieved problematic username or password! (User: \"{user}\")"); + if(verbosity > 0) Output.Error($"Recieved problematic username or password! (User: \"{user}\")"); return ErrorResponse(id); } Database.User usr = db.GetUser(user); @@ -447,7 +459,9 @@ Use command 'help' to get a list of available commands"; } Output.Raw($"Current verbosity level: {(verbosity<1?"FATAL":verbosity==1?"INFO":"DEBUG")}"); }), "Get or set verbosity level: DEBUG, INFO, FATAL (alternatively enter 0, 1 or 2 respectively)") - .Append(new Command("sess").WithParameter("sessionID", 'r', Parameter.ParamType.STRING, true).SetAction( // Display active sessions + .Append(new Command("sess") // Display active sessions + .WithParameter("sessionID", 'r', Parameter.ParamType.STRING, true) + .SetAction( (c, l) => { StringBuilder builder = new StringBuilder(); manager.Update(); // Ensure that we don't show expired sessions (artifacts exist until it is necessary to remove them) @@ -498,7 +512,8 @@ Use command 'help' to get a list of available commands"; db.AddUser(user); } else Output.Raw(user.IsAdministrator); - }), "Show or set admin status for a user"); + }), "Show or set admin status for a user") + .Append(new Command("flush").SetAction(() => { db.Flush(); Output.Raw("Database flushed"); }), "Flush database");// Flush database to database file // Set up a persistent terminal-esque input design Output.OnNewLine = () => Output.WriteOverwritable(">> ");