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;