From 939f6c910bd86dc275480584cf29d4fd7ef1f6fd Mon Sep 17 00:00:00 2001 From: GabrielTofvesson Date: Fri, 18 May 2018 20:45:43 +0200 Subject: [PATCH] Fixed som graphics routines Added support for on-the-fly textview contents updates Added iterative view removal to ConsoleController Added dumy layouts to Common Added support for account removal Fixed command management (now supports leading, padding and trailing spaces) Various smaller changes --- Client/BankNetInteractor.cs | 32 +++++++++++------ Client/Client.csproj | 2 ++ Client/ConsoleForms/ConsoleController.cs | 6 ++++ Client/ConsoleForms/Graphics/DialogView.cs | 22 +++++++++--- Client/ConsoleForms/Graphics/TextView.cs | 12 ++++--- Client/ConsoleForms/Graphics/View.cs | 11 +++--- Client/Context/SessionContext.cs | 41 +++++++++++++++++++--- Client/Resources/Layout/Common.xml | 19 ++++++++++ Client/Resources/Strings/en_GB/strings.xml | 5 +++ Client/Resources/Strings/en_US/strings.xml | 5 +++ Client/Resources/Strings/sv_SE/strings.xml | 8 ++++- Server/Command.cs | 1 + Server/CommandHandler.cs | 26 ++++++++++++++ Server/Program.cs | 38 ++++++++++++++------ Server/SessionManager.cs | 12 +++++-- 15 files changed, 197 insertions(+), 43 deletions(-) diff --git a/Client/BankNetInteractor.cs b/Client/BankNetInteractor.cs index 81297b4..da42d77 100644 --- a/Client/BankNetInteractor.cs +++ b/Client/BankNetInteractor.cs @@ -132,8 +132,9 @@ namespace Client bool b = !p.Value.StartsWith("ERROR"); if (b) // Set proper state before notifying listener { - RefreshTimeout(); sessionID = p.Value; + RefreshTimeout(); + SetAutoRefresh(true); } PostPromise(p.handler, b); return false; @@ -203,7 +204,7 @@ namespace Client RefreshTimeout(); return RegisterEventPromise(pID, p => { - p.handler.Value = p.Value.StartsWith("ERROR").ToString(); + PostPromise(p.handler, !p.Value.StartsWith("ERROR")); return false; }); } @@ -350,8 +351,7 @@ namespace Client protected void SetAutoRefresh(bool doAR) { - if (RefreshSessions == doAR) return; - if (RefreshSessions = doAR) + if (RefreshSessions = doAR && (sessionChecker==null || sessionChecker.Status!=TaskStatus.Running)) { triggerRefreshCancel = false; sessionChecker = new Task(DoRefresh); @@ -367,7 +367,14 @@ namespace Client return; } // Refresher calls refresh 1500ms before expiry (or asap if less time is available) - Task.Delay((int)((Math.Min(0, loginTimeout - DateTime.Now.Ticks - 1500)) / TimeSpan.TicksPerMillisecond)); + try + { + Task.Delay((int)Math.Max(1, ((loginTimeout - DateTime.Now.Ticks) / TimeSpan.TicksPerMillisecond) - 1500)).Wait(); + } + catch + { + System.Diagnostics.Debug.WriteLine("OOF"); + } if (triggerRefreshCancel) { triggerRefreshCancel = false; @@ -377,18 +384,21 @@ namespace Client { try { - Refresh(); + Promise p = Promise.AwaitPromise(Refresh()); + p.Subscribe = refreshResult => + { + if (RefreshSessions && bool.Parse(refreshResult.Value)) + { + sessionChecker = new Task(DoRefresh); + sessionChecker.Start(); + } + }; } catch { // Session probably died return; } - if (RefreshSessions) - { - sessionChecker = new Task(DoRefresh); - sessionChecker.Start(); - } } } diff --git a/Client/Client.csproj b/Client/Client.csproj index 5cd02fb..ec01505 100644 --- a/Client/Client.csproj +++ b/Client/Client.csproj @@ -11,6 +11,8 @@ v4.6.1 512 true + + AnyCPU diff --git a/Client/ConsoleForms/ConsoleController.cs b/Client/ConsoleForms/ConsoleController.cs index 3fe2a8c..fa91ba1 100644 --- a/Client/ConsoleForms/ConsoleController.cs +++ b/Client/ConsoleForms/ConsoleController.cs @@ -78,6 +78,12 @@ namespace Client.ConsoleForms Draw(false); } + public void CloseIf(Predicate p) + { + for(int i = renderQueue.Count - 1; i>=0; --i) + if (p(renderQueue[i].Item1)) + CloseView(i); + } public void CloseTop() => CloseView(renderQueue[renderQueue.Count - 1].Item1); public void CloseView(int idx) => CloseView(renderQueue[idx].Item1); public void CloseView(View v, bool redraw = true, int maxCloses = -1) diff --git a/Client/ConsoleForms/Graphics/DialogView.cs b/Client/ConsoleForms/Graphics/DialogView.cs index d752986..159ee8b 100644 --- a/Client/ConsoleForms/Graphics/DialogView.cs +++ b/Client/ConsoleForms/Graphics/DialogView.cs @@ -13,6 +13,7 @@ namespace Client.ConsoleForms.Graphics public delegate void SelectListener(DialogView view, int selectionIndex, string selection); protected readonly ViewData[] options; + protected readonly int optionsWidth; protected int select; protected SelectListener listener; @@ -21,6 +22,18 @@ namespace Client.ConsoleForms.Graphics get => select; set => select = value < 0 ? 0 : value >= options.Length ? options.Length - 1 : value; } + + public override string Text + { + get => base.Text; + set + { + base.Text = value; + // Since setting the text triggers a rendering recomputation for TextView, we have to recompute rendering for options too + if (optionsWidth > ContentWidth) ContentWidth = optionsWidth; + ContentHeight += 2; + } + } /* public override Region Occlusion => new Region( new Rectangle( @@ -47,27 +60,26 @@ namespace Client.ConsoleForms.Graphics ViewData optionsData = parameters.Get("Options"); this.options = optionsData.nestedData.Filter(p => p.Name.Equals("Option")).ToArray(); this.select = parameters.AttribueAsInt("select"); - ContentHeight += 2; + //ContentHeight += 2; select = select < 0 ? 0 : select >= options.Length ? 0 : select; SelectColor = (ConsoleColor)parameters.AttribueAsInt("select_color", (int)ConsoleColor.Gray); NotSelectColor = (ConsoleColor)parameters.AttribueAsInt("unselect_color", (int)ConsoleColor.White); + optionsWidth = ComputeLength(parameters.Get("Options")?.CollectSub("Option") ?? new Tuple[0]); + if (optionsWidth > ContentWidth) ContentWidth = optionsWidth; } protected override void _Draw(int left, ref int top) { - //DrawEmptyPadding(left, ref top, padding.Top()); base.DrawContent(left, ref top); DrawEmptyPadding(left, ref top, 1); DrawOptions(left, ref top); - //DrawEmptyPadding(left, ref top, padding.Bottom()); } protected virtual void DrawOptions(int left, ref int top) { - int pl = padding.Left(), pr = padding.Right(); Console.SetCursorPosition(left, top++); - int pad = ContentWidth - options.CollectiveLength() - options.Length;// + pl + pr; + int pad = ContentWidth - options.CollectiveLength() - options.Length; int lpad = (int)(pad / 2f); Console.BackgroundColor = BackgroundColor; Console.Write(Filler(' ', lpad)); diff --git a/Client/ConsoleForms/Graphics/TextView.cs b/Client/ConsoleForms/Graphics/TextView.cs index cd75c3c..79c2608 100644 --- a/Client/ConsoleForms/Graphics/TextView.cs +++ b/Client/ConsoleForms/Graphics/TextView.cs @@ -13,12 +13,13 @@ namespace Client.ConsoleForms.Graphics protected string[] text; protected string[] text_render; protected int maxWidth, maxHeight; + protected readonly bool enforceWidth; private string _text; - public string Text + public virtual string Text { get => _text; - protected set + set { _text = value; text = _text.Split(' '); @@ -27,7 +28,7 @@ namespace Client.ConsoleForms.Graphics text_render = ComputeTextDimensions(this.text); int actualWidth = 0; foreach (var t in text_render) if (actualWidth < t.Length) actualWidth = t.Length; - ContentWidth = maxWidth;// + padding.Left() + padding.Right(); + ContentWidth = enforceWidth ? maxWidth : actualWidth;// + padding.Left() + padding.Right(); ContentHeight = text_render.Length;// + padding.Top() + padding.Bottom(); Dirty = true; } @@ -42,6 +43,8 @@ namespace Client.ConsoleForms.Graphics ) ); + //TODO: Add contentocclusion + //public char Border { get; set; } //public ConsoleColor BorderColor { get; set; } @@ -52,7 +55,8 @@ namespace Client.ConsoleForms.Graphics foreach (var t in parameters.NestedText("Text").Split('\n')) if (t.Length > widest) widest = t.Length; - this.maxWidth = parameters.AttribueAsInt("width") < 1 ? widest : parameters.AttribueAsInt("width"); + enforceWidth = parameters.AttribueAsInt("width") > 0; + this.maxWidth = enforceWidth ? parameters.AttribueAsInt("width") : widest; this.maxHeight = parameters.AttribueAsInt("height", -1); this.text = (Text = parameters.NestedText("Text")).Split(' '); diff --git a/Client/ConsoleForms/Graphics/View.cs b/Client/ConsoleForms/Graphics/View.cs index 4cfd666..542ecb4 100644 --- a/Client/ConsoleForms/Graphics/View.cs +++ b/Client/ConsoleForms/Graphics/View.cs @@ -28,11 +28,12 @@ namespace Client.ConsoleForms.Graphics public ConsoleColor TextColor { get; set; } public int ContentWidth { get; protected set; } public int ContentHeight { get; protected set; } - public abstract Region Occlusion { get; } - public bool Dirty { get; set; } - public LangManager I18n { get; private set; } - public ViewEvent OnBackEvent { get; set; } - public ViewEvent OnClose { get; set; } + public abstract Region Occlusion { get; } // Reports dimensions of entire view + public Region ContentOcclusion { get => Occlusion; }// Reports dimensions of contents (SHOULD for most applications be smaller than or equal in size to Occlusion) + public bool Dirty { get; set; } // Flag for whether or not view requires re-rendering + public LangManager I18n { get; private set; } // Translation + public ViewEvent OnBackEvent { get; set; } // Callback for [ESC] key + public ViewEvent OnClose { get; set; } // Callback called immediately before controller removes view from render queue public View(ViewData parameters, LangManager lang) { diff --git a/Client/Context/SessionContext.cs b/Client/Context/SessionContext.cs index 5150074..feb5175 100644 --- a/Client/Context/SessionContext.cs +++ b/Client/Context/SessionContext.cs @@ -248,7 +248,7 @@ namespace Client accountsGetter.Unsubscribe(); Hide("data_fetch"); - Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), SubmitListener)); + Show(GenerateList(p.Value.Split('&').ForEach(Support.FromBase64String), ViewAccountListener)); }; }); @@ -518,7 +518,7 @@ namespace Client } } - private void SubmitListener(View listener) + private void ViewAccountListener(View listener) { ButtonView view = listener as ButtonView; @@ -536,7 +536,7 @@ namespace Client .SetAttribute("border", (int)ConsoleColor.DarkGreen) // Option buttons - .AddNested(new ViewData("Options").AddNestedSimple("Option", GetIntlString("GENERIC_dismiss"))) + .AddNested(new ViewData("Options").AddNestedSimple("Option", GetIntlString("GENERIC_dismiss")).AddNestedSimple("Option", GetIntlString("SE_account_delete"))) // Message .AddNestedSimple("Text", @@ -548,7 +548,40 @@ namespace Client // No translation (it's already handled) LangManager.NO_LANG); - show.RegisterSelectListener((_, s, l) => Hide(show)); + show.RegisterSelectListener((_, s, l) => + { + if(s==0) Hide(show); + else + { + var ynDialog = GetView("yn"); + ynDialog.Text = GetIntlString("SE_account_delete_warn"); + ynDialog.RegisterSelectListener((v, i, str) => + { + var stall = GetView("stall"); + stall.Text = GetIntlString("SE_account_delete_stall"); + Show(stall); + if (i == 1) + { + Promise p = Promise.AwaitPromise(interactor.CloseAccount(name)); + p.Subscribe = deleteAwait => + { + if (bool.Parse(deleteAwait.Value)) + { + accountChange = true; + controller.Popup(GetIntlString("SE_account_delete_success"), 1500, ConsoleColor.Green, () => { + bool closed = false; + controller.CloseIf(predV => closed = !closed && predV is ListView); + Hide(show); + }); + } + else controller.Popup(GetIntlString("SE_account_delete_fail"), 2000, ConsoleColor.Red); + Hide(stall); + }; + } + }); + Show(ynDialog); + } + }); Show(show); } diff --git a/Client/Resources/Layout/Common.xml b/Client/Resources/Layout/Common.xml index 2809cab..9cc3389 100644 --- a/Client/Resources/Layout/Common.xml +++ b/Client/Resources/Layout/Common.xml @@ -52,4 +52,23 @@ @string/NC_quit + + + + + + + + + + \ No newline at end of file diff --git a/Client/Resources/Strings/en_GB/strings.xml b/Client/Resources/Strings/en_GB/strings.xml index cc8171c..1ab6068 100644 --- a/Client/Resources/Strings/en_GB/strings.xml +++ b/Client/Resources/Strings/en_GB/strings.xml @@ -125,6 +125,11 @@ Balance: $2 SEK Creating account... Account "$0" already exists! Account successfully created! + Delete account + Are you sure you want to delete this account? + Account successfully deleted! + Account couldn't be deleted! + Deleting... Are you sure you would like log out and exit? Fetching data... diff --git a/Client/Resources/Strings/en_US/strings.xml b/Client/Resources/Strings/en_US/strings.xml index 32d3c65..f10b3af 100644 --- a/Client/Resources/Strings/en_US/strings.xml +++ b/Client/Resources/Strings/en_US/strings.xml @@ -124,6 +124,11 @@ Balance: $1 SEK Creating account... Account "$0" already exists! Account successfully created! + Delete account + Are you sure you want to delete this account? + Account successfully deleted! + Account couldn't be deleted! + Deleting... Are you sure you would like log out and exit? Fetching data... diff --git a/Client/Resources/Strings/sv_SE/strings.xml b/Client/Resources/Strings/sv_SE/strings.xml index b641d67..1137af3 100644 --- a/Client/Resources/Strings/sv_SE/strings.xml +++ b/Client/Resources/Strings/sv_SE/strings.xml @@ -1,4 +1,4 @@ - + Hej och välkommen till ConsoleForms bankprojektet! Om du är ovan vid ConsoleForms-gränssnittet och vill @@ -62,6 +62,7 @@ programmet, tryck [ESC]. Verifierar serverns identitet... Serveridentitet verifierad! Serveridentitet kunde inte verifieras! + Vill du avsluta programmet? Välkommen till Tofvessons banksystem! För att fortsätta, tryck [ENTER] @@ -129,6 +130,11 @@ Kontobalans: $1 SEK Skapar konto... Kontot "$0" finns redan! Konto skapat! + Radera konto + Är du säker på att du vill radera kontot? + Kontot raderat + Konto kunde inte raderas! + Raderar... Är du säker på att du vill logga ut och stänga? Hämtar data... diff --git a/Server/Command.cs b/Server/Command.cs index cfe989b..9660d84 100644 --- a/Server/Command.cs +++ b/Server/Command.cs @@ -74,6 +74,7 @@ namespace Server bool wasFlag = true; for (int i = 1; ipost) return false; + + // Trim leading and trailing spaces + cmd = cmd.Substring(pre, post - pre); + foreach (var command in commands) if (command.Item1.Invoke(cmd)) return true; diff --git a/Server/Program.cs b/Server/Program.cs index 975d596..afbcce1 100644 --- a/Server/Program.cs +++ b/Server/Program.cs @@ -471,21 +471,37 @@ Use command 'help' to get a list of available commands"; }), "Get or set verbosity level: DEBUG, INFO, FATAL (alternatively enter 0, 1 or 2 respectively)") .Append(new Command("sess") // Display active sessions .WithParameter("sessionID", 'r', Parameter.ParamType.STRING, true) + .WithParameter("username", 'u', 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) - foreach (var session in manager.Sessions) - builder - .Append(session.user.Name) - .Append(" : ") - .Append(session.sessionID) - .Append(" : ") - .Append((session.expiry-DateTime.Now.Ticks) /TimeSpan.TicksPerMillisecond) - .Append('\n'); - if (builder.Length == 0) builder.Append("There are no active sessions at the moment"); - else --builder.Insert(0, "Active sessions:\n").Length; - Output.Raw(builder); + bool r = l.HasFlag('r'), u = l.HasFlag('u'); + if(r && u) + { + Output.Error("Cannot refresh session by username AND SessionID!"); + return; + } + if((r||u) && !manager.HasSession(l[0].Item1, u)) + { + Output.Error("Session could not be found!"); + return; + } + if (r||u) Output.Raw($"Session refreshed: {manager.Refresh(l[0].Item1, u)}"); + else + { + foreach (var session in manager.Sessions) + builder + .Append(session.user.Name) + .Append(" : ") + .Append(session.sessionID) + .Append(" : ") + .Append((session.expiry - DateTime.Now.Ticks) / TimeSpan.TicksPerMillisecond) + .Append('\n'); + if (builder.Length == 0) builder.Append("There are no active sessions at the moment"); + else --builder.Insert(0, "Active sessions:\n").Length; + Output.Raw(builder); + } }), "List or refresh active client sessions") .Append(new Command("list").WithParameter(Parameter.Flag('a')).SetAction( // Display users (c, l) => { diff --git a/Server/SessionManager.cs b/Server/SessionManager.cs index 7d823d1..df4405b 100644 --- a/Server/SessionManager.cs +++ b/Server/SessionManager.cs @@ -19,6 +19,14 @@ namespace Server this.sidLength = sidLength < 10 ? 10 : sidLength; } + public bool HasSession(string sess, bool byUserName = false) + { + foreach (var session in Sessions) + if ((byUserName && session.user.Name.Equals(sess)) || (!byUserName && session.sessionID.Equals(sess))) + return true; + return false; + } + public string GetSession(Database.User user, string invalidSID) { Update(); @@ -70,11 +78,11 @@ namespace Server return; } - public bool Refresh(string sid) + public bool Refresh(string sid, bool asUser = false) { Update(); for (int i = sessions.Count - 1; i >= 0; --i) - if (sessions[i].sessionID.Equals(sid)) + if ((asUser && sessions[i].user.Name.Equals(sid)) || (!asUser && sessions[i].sessionID.Equals(sid))) { Session s = sessions[i]; s.expiry = DateTime.Now.Ticks + timeout;