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;