* Added flush command to server to flush database
* Fixed how database manages user removal * Added serverside user removal * Added UI and netcode for user removal on clientside * Fixed text rendering for TextView and InputView * Removed redundant properties from TextView * Simplified event system in Intro
This commit is contained in:
parent
fdad3be98d
commit
9755f06120
@ -139,6 +139,17 @@ namespace Client
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async virtual Task<Promise> 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<Promise> UpdatePassword(string newPass)
|
public async virtual Task<Promise> UpdatePassword(string newPass)
|
||||||
{
|
{
|
||||||
await StatusCheck(true);
|
await StatusCheck(true);
|
||||||
|
@ -67,7 +67,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
int pl = padding.Left(), pr = padding.Right();
|
int pl = padding.Left(), pr = padding.Right();
|
||||||
Console.SetCursorPosition(left, top++);
|
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);
|
int lpad = (int)(pad / 2f);
|
||||||
Console.BackgroundColor = BackgroundColor;
|
Console.BackgroundColor = BackgroundColor;
|
||||||
Console.Write(Filler(' ', lpad));
|
Console.Write(Filler(' ', lpad));
|
||||||
|
@ -32,6 +32,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
private string[][] splitInputs;
|
private string[][] splitInputs;
|
||||||
|
protected ViewData data;
|
||||||
|
|
||||||
|
|
||||||
public InputView(ViewData parameters, LangManager lang) : base(parameters, lang)
|
public InputView(ViewData parameters, LangManager lang) : base(parameters, lang)
|
||||||
@ -47,6 +48,8 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
DefaultSelectBackgroundColor = (ConsoleColor)sBC;
|
DefaultSelectBackgroundColor = (ConsoleColor)sBC;
|
||||||
DefaultSelectTextColor = (ConsoleColor)sTC;
|
DefaultSelectTextColor = (ConsoleColor)sTC;
|
||||||
|
|
||||||
|
this.data = parameters;
|
||||||
|
|
||||||
List<InputField> fields = new List<InputField>();
|
List<InputField> fields = new List<InputField>();
|
||||||
foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData)
|
foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData)
|
||||||
if (!data.Name.Equals("Field")) continue;
|
if (!data.Name.Equals("Field")) continue;
|
||||||
@ -63,14 +66,19 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
|
|
||||||
Inputs = fields.ToArray();
|
Inputs = fields.ToArray();
|
||||||
|
|
||||||
|
int max = ContentWidth;
|
||||||
int computedSize = 0;
|
int computedSize = 0;
|
||||||
splitInputs = new string[Inputs.Length][];
|
splitInputs = new string[Inputs.Length][];
|
||||||
for (int i = 0; i < Inputs.Length; ++i)
|
for (int i = 0; i < Inputs.Length; ++i)
|
||||||
{
|
{
|
||||||
splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' '));
|
splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' '));
|
||||||
|
foreach (var input in splitInputs[i])
|
||||||
|
if (input.Length > max)
|
||||||
|
max = input.Length;
|
||||||
computedSize += splitInputs[i].Length;
|
computedSize += splitInputs[i].Length;
|
||||||
}
|
}
|
||||||
ContentHeight += computedSize + Inputs.Length * 2;
|
ContentHeight += computedSize + Inputs.Length * 2;
|
||||||
|
if (ContentWidth < max) ContentWidth = max;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int IndexOf(InputField field)
|
public int IndexOf(InputField field)
|
||||||
@ -98,7 +106,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
{
|
{
|
||||||
Console.SetCursorPosition(left, top++);
|
Console.SetCursorPosition(left, top++);
|
||||||
Console.BackgroundColor = BackgroundColor;
|
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++);
|
Console.SetCursorPosition(left, top++);
|
||||||
|
|
||||||
@ -114,10 +122,10 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
if (Inputs[j].SelectIndex < Inputs[j].Text.Length)
|
if (Inputs[j].SelectIndex < Inputs[j].Text.Length)
|
||||||
Console.Write(
|
Console.Write(
|
||||||
Inputs[j].ShowText ?
|
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)) :
|
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(maxWidth + 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;
|
Console.ForegroundColor = ConsoleColor.Black;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -140,7 +148,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
case ConsoleKey.RightArrow:
|
case ConsoleKey.RightArrow:
|
||||||
if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length)
|
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;
|
else return changed;
|
||||||
break;
|
break;
|
||||||
@ -175,6 +183,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
else return changed;
|
else return changed;
|
||||||
break;
|
break;
|
||||||
case ConsoleKey.Enter:
|
case ConsoleKey.Enter:
|
||||||
|
ParseAction(data)();
|
||||||
SubmissionsListener?.Invoke(this);
|
SubmissionsListener?.Invoke(this);
|
||||||
return changed;
|
return changed;
|
||||||
case ConsoleKey.Escape:
|
case ConsoleKey.Escape:
|
||||||
@ -184,7 +193,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
{
|
{
|
||||||
if (InputListener?.Invoke(this, Inputs[selectedField], info, triggered) == false) break;
|
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);
|
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;
|
else return changed;
|
||||||
break;
|
break;
|
||||||
|
@ -32,29 +32,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
Dirty = true;
|
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(
|
public override Region Occlusion => new Region(
|
||||||
new Rectangle(
|
new Rectangle(
|
||||||
-padding.Left() - (DrawBorder ? 2 : 0), // Left bound
|
-padding.Left() - (DrawBorder ? 2 : 0), // Left bound
|
||||||
@ -231,7 +209,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
for (int i = 0; i < text_render.Length; ++i)
|
for (int i = 0; i < text_render.Length; ++i)
|
||||||
{
|
{
|
||||||
Console.SetCursorPosition(left, top++);
|
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.SetCursorPosition(left, top++);
|
||||||
Console.BackgroundColor = BackgroundColor;
|
Console.BackgroundColor = BackgroundColor;
|
||||||
Console.Write(Filler(' ', maxWidth/* + pl + pr*/));
|
Console.Write(Filler(' ', ContentWidth/* + pl + pr*/));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -139,7 +139,7 @@ namespace Client.ConsoleForms.Graphics
|
|||||||
}
|
}
|
||||||
protected EventAction ParseAction(ViewData data)
|
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);
|
return ParseAction(data.GetAttribute("event"), close);
|
||||||
}
|
}
|
||||||
protected EventAction ParseAction(string action, bool close)
|
protected EventAction ParseAction(string action, bool close)
|
||||||
|
@ -13,56 +13,11 @@ namespace Client
|
|||||||
public IntroContext(ContextManager manager, Action onComplete) : base(manager, "Intro", "Common")
|
public IntroContext(ContextManager manager, Action onComplete) : base(manager, "Intro", "Common")
|
||||||
{
|
{
|
||||||
GetView<DialogView>("welcome").RegisterSelectListener((v, i, s) =>
|
GetView<DialogView>("welcome").RegisterSelectListener((v, i, s) =>
|
||||||
{
|
|
||||||
if (i == 1)
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
onComplete();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
Show("describe1");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
GetView<DialogView>("describe1").RegisterSelectListener((v, i, s) =>
|
|
||||||
{
|
|
||||||
if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false));
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
Show("describe2");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
GetView<DialogView>("describe2").RegisterSelectListener((v, i, s) =>
|
|
||||||
{
|
|
||||||
if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false));
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
Show("describe3");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
GetView<InputView>("describe3").SubmissionsListener = v =>
|
|
||||||
{
|
{
|
||||||
Hide(v);
|
Hide(v);
|
||||||
Show("describe4");
|
if (i == 1) onComplete();
|
||||||
};
|
else Show("describe1");
|
||||||
|
});
|
||||||
GetView<InputView>("describe4").SubmissionsListener = v =>
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
Show("describe4_1");
|
|
||||||
};
|
|
||||||
|
|
||||||
GetView<InputView>("describe4_1").SubmissionsListener = v =>
|
|
||||||
{
|
|
||||||
Hide(v);
|
|
||||||
Show("describe5");
|
|
||||||
};
|
|
||||||
|
|
||||||
GetView<DialogView>("describe5").RegisterSelectListener((v, i, s) =>
|
GetView<DialogView>("describe5").RegisterSelectListener((v, i, s) =>
|
||||||
{
|
{
|
||||||
@ -77,22 +32,7 @@ namespace Client
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void OnCreate()
|
public override void OnCreate() => Show("welcome");
|
||||||
{
|
public override void OnDestroy() { }
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -148,6 +148,26 @@ namespace Client
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
options.GetView<ButtonView>("delete").SetEvent(v => Show("account_delete"));
|
||||||
|
|
||||||
|
GetView<DialogView>("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
|
// Actual "create account" input box thingy
|
||||||
var input = GetView<InputView>("account_create");
|
var input = GetView<InputView>("account_create");
|
||||||
input.SubmissionsListener = __ =>
|
input.SubmissionsListener = __ =>
|
||||||
|
@ -25,6 +25,8 @@ namespace Client
|
|||||||
// Just close when anything is selected and "submitted"
|
// Just close when anything is selected and "submitted"
|
||||||
RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError");
|
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<InputView>("Login").SubmissionsListener = i =>
|
GetView<InputView>("Login").SubmissionsListener = i =>
|
||||||
{
|
{
|
||||||
|
@ -22,7 +22,7 @@
|
|||||||
padding_bottom="1"
|
padding_bottom="1"
|
||||||
back="Intro:welcome">
|
back="Intro:welcome">
|
||||||
<Options>
|
<Options>
|
||||||
<Option>@string/WS_continue</Option>
|
<Option event="Intro:describe2" close="true">@string/WS_continue</Option>
|
||||||
</Options>
|
</Options>
|
||||||
<Text>@string/WS_describe1</Text>
|
<Text>@string/WS_describe1</Text>
|
||||||
</DialogView>
|
</DialogView>
|
||||||
@ -35,8 +35,8 @@
|
|||||||
padding_bottom="1"
|
padding_bottom="1"
|
||||||
back="Intro:describe1">
|
back="Intro:describe1">
|
||||||
<Options>
|
<Options>
|
||||||
<Option>@string/WS_continue</Option>
|
<Option event="Intro:describe3" close="true">@string/WS_continue</Option>
|
||||||
<Option>@string/WS_back</Option>
|
<Option event="Intro:describe3" close="true">@string/WS_back</Option>
|
||||||
</Options>
|
</Options>
|
||||||
<Text>@string/WS_describe2</Text>
|
<Text>@string/WS_describe2</Text>
|
||||||
</DialogView>
|
</DialogView>
|
||||||
@ -47,7 +47,9 @@
|
|||||||
padding_right="2"
|
padding_right="2"
|
||||||
padding_top="1"
|
padding_top="1"
|
||||||
padding_bottom="1"
|
padding_bottom="1"
|
||||||
back="Intro:describe2">
|
back="Intro:describe2"
|
||||||
|
event="Intro:describe4"
|
||||||
|
close="true">
|
||||||
<Fields>
|
<Fields>
|
||||||
<Field>@string/WS_input</Field>
|
<Field>@string/WS_input</Field>
|
||||||
<Field>@string/WS_input</Field>
|
<Field>@string/WS_input</Field>
|
||||||
@ -62,7 +64,9 @@
|
|||||||
padding_right="2"
|
padding_right="2"
|
||||||
padding_top="1"
|
padding_top="1"
|
||||||
padding_bottom="1"
|
padding_bottom="1"
|
||||||
back="Intro:describe3">
|
back="Intro:describe3"
|
||||||
|
event="Intro:describe4_1"
|
||||||
|
close="true">
|
||||||
<Fields>
|
<Fields>
|
||||||
<Field>@string/WS_input</Field>
|
<Field>@string/WS_input</Field>
|
||||||
<Field input_type="integer">@string/WS_input_integer</Field>
|
<Field input_type="integer">@string/WS_input_integer</Field>
|
||||||
@ -77,7 +81,9 @@
|
|||||||
padding_right="2"
|
padding_right="2"
|
||||||
padding_top="1"
|
padding_top="1"
|
||||||
padding_bottom="1"
|
padding_bottom="1"
|
||||||
back="Intro:describe4">
|
back="Intro:describe4"
|
||||||
|
event="Intro:describe5"
|
||||||
|
close="true">
|
||||||
<Fields>
|
<Fields>
|
||||||
<Field input_type="alphanumeric">@string/WS_input_alphanum</Field>
|
<Field input_type="alphanumeric">@string/WS_input_alphanum</Field>
|
||||||
<Field hide="true">@string/WS_input_password</Field>
|
<Field hide="true">@string/WS_input_password</Field>
|
||||||
|
@ -90,12 +90,37 @@
|
|||||||
<Text>@string/SE_pwdu</Text>
|
<Text>@string/SE_pwdu</Text>
|
||||||
</ButtonView>
|
</ButtonView>
|
||||||
|
|
||||||
|
<ButtonView id="delete">
|
||||||
|
<Text>@string/SE_delete</Text>
|
||||||
|
</ButtonView>
|
||||||
|
|
||||||
<ButtonView id="exit">
|
<ButtonView id="exit">
|
||||||
<Text>@string/SE_exit</Text>
|
<Text>@string/SE_exit</Text>
|
||||||
</ButtonView>
|
</ButtonView>
|
||||||
</Views>
|
</Views>
|
||||||
</ListView>
|
</ListView>
|
||||||
|
|
||||||
|
<TextView id="delete_stall"
|
||||||
|
padding_left="2"
|
||||||
|
padding_right="2"
|
||||||
|
padding_top="1"
|
||||||
|
padding_bottom="1">
|
||||||
|
<Text>@string/SE_delete_stall</Text>
|
||||||
|
</TextView>
|
||||||
|
|
||||||
|
<DialogView id="account_delete"
|
||||||
|
padding_left="2"
|
||||||
|
padding_right="2"
|
||||||
|
padding_top="1"
|
||||||
|
padding_bottom="1"
|
||||||
|
border="4">
|
||||||
|
<Options>
|
||||||
|
<Option>@string/GENERIC_negative</Option>
|
||||||
|
<Option>@string/GENERIC_positive</Option>
|
||||||
|
</Options>
|
||||||
|
<Text>@string/SE_delete_warn</Text>
|
||||||
|
</DialogView>
|
||||||
|
|
||||||
<InputView id="account_create"
|
<InputView id="account_create"
|
||||||
padding_left="2"
|
padding_left="2"
|
||||||
padding_right="2"
|
padding_right="2"
|
||||||
|
@ -79,7 +79,7 @@ To go back, press [ESCAPE]</Entry>
|
|||||||
<Entry name="SE_hist">Transaction history</Entry>
|
<Entry name="SE_hist">Transaction history</Entry>
|
||||||
<Entry name="SE_tx">Transfer funds</Entry>
|
<Entry name="SE_tx">Transfer funds</Entry>
|
||||||
<Entry name="SE_tx_success">Funds transferred!</Entry>
|
<Entry name="SE_tx_success">Funds transferred!</Entry>
|
||||||
<Entry name="SE_who">Send to</Entry>
|
<Entry name="SE_who">Send to:</Entry>
|
||||||
<Entry name="SE_where">Account</Entry>
|
<Entry name="SE_where">Account</Entry>
|
||||||
<Entry name="SE_where_f">From Account:</Entry>
|
<Entry name="SE_where_f">From Account:</Entry>
|
||||||
<Entry name="SE_where_t">To Account:</Entry>
|
<Entry name="SE_where_t">To Account:</Entry>
|
||||||
@ -98,6 +98,12 @@ Is this correct?</Entry>
|
|||||||
<Entry name="SE_exit">Log out</Entry>
|
<Entry name="SE_exit">Log out</Entry>
|
||||||
<Entry name="SE_open">Open an account</Entry>
|
<Entry name="SE_open">Open an account</Entry>
|
||||||
<Entry name="SE_close">Close account</Entry>
|
<Entry name="SE_close">Close account</Entry>
|
||||||
|
<Entry name="SE_delete">Delete user account</Entry>
|
||||||
|
<Entry name="SE_delete_warn">WARNING: This will delete the current user and all connected accounts!
|
||||||
|
Are you sure you would like to continue?</Entry>
|
||||||
|
<Entry name="SE_delete_stall">Deleting...</Entry>
|
||||||
|
<Entry name="SE_delete_success">User deleted</Entry>
|
||||||
|
<Entry name="SE_delete_failure">User could not be deleted</Entry>
|
||||||
<Entry name="SE_accounts">Show accounts</Entry>
|
<Entry name="SE_accounts">Show accounts</Entry>
|
||||||
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
|
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
|
||||||
Available balance: $0 SEK</Entry>
|
Available balance: $0 SEK</Entry>
|
||||||
|
@ -79,7 +79,7 @@ To go back, press [ESCAPE]</Entry>
|
|||||||
<Entry name="SE_hist">Transaction history</Entry>
|
<Entry name="SE_hist">Transaction history</Entry>
|
||||||
<Entry name="SE_tx">Transfer funds</Entry>
|
<Entry name="SE_tx">Transfer funds</Entry>
|
||||||
<Entry name="SE_tx_success">Funds transferred!</Entry>
|
<Entry name="SE_tx_success">Funds transferred!</Entry>
|
||||||
<Entry name="SE_who">Send to</Entry>
|
<Entry name="SE_who">Send to:</Entry>
|
||||||
<Entry name="SE_where">Account</Entry>
|
<Entry name="SE_where">Account</Entry>
|
||||||
<Entry name="SE_where_f">From Account:</Entry>
|
<Entry name="SE_where_f">From Account:</Entry>
|
||||||
<Entry name="SE_where_t">To Account:</Entry>
|
<Entry name="SE_where_t">To Account:</Entry>
|
||||||
@ -98,6 +98,12 @@ Is this correct?</Entry>
|
|||||||
<Entry name="SE_exit">Log out</Entry>
|
<Entry name="SE_exit">Log out</Entry>
|
||||||
<Entry name="SE_open">Open an account</Entry>
|
<Entry name="SE_open">Open an account</Entry>
|
||||||
<Entry name="SE_close">Close account</Entry>
|
<Entry name="SE_close">Close account</Entry>
|
||||||
|
<Entry name="SE_delete">Delete user account</Entry>
|
||||||
|
<Entry name="SE_delete_warn">WARNING: This will delete the current user and all connected accounts!
|
||||||
|
Are you sure you would like to continue?</Entry>
|
||||||
|
<Entry name="SE_delete_stall">Deleting...</Entry>
|
||||||
|
<Entry name="SE_delete_success">User deleted</Entry>
|
||||||
|
<Entry name="SE_delete_failure">User could not be deleted</Entry>
|
||||||
<Entry name="SE_accounts">Show accounts</Entry>
|
<Entry name="SE_accounts">Show accounts</Entry>
|
||||||
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
|
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
|
||||||
Available balance: $0 SEK</Entry>
|
Available balance: $0 SEK</Entry>
|
||||||
|
@ -84,7 +84,7 @@ För att backa, tryck [ESCAPE]</Entry>
|
|||||||
<Entry name="SE_hist">Transaktionshistorik</Entry>
|
<Entry name="SE_hist">Transaktionshistorik</Entry>
|
||||||
<Entry name="SE_tx">Överför pengar</Entry>
|
<Entry name="SE_tx">Överför pengar</Entry>
|
||||||
<Entry name="SE_tx_success">Belopp överfört!</Entry>
|
<Entry name="SE_tx_success">Belopp överfört!</Entry>
|
||||||
<Entry name="SE_who">Skicka till</Entry>
|
<Entry name="SE_who">Skicka till:</Entry>
|
||||||
<Entry name="SE_where">Konto</Entry>
|
<Entry name="SE_where">Konto</Entry>
|
||||||
<Entry name="SE_where_f">Från konto:</Entry>
|
<Entry name="SE_where_f">Från konto:</Entry>
|
||||||
<Entry name="SE_where_t">Till konto:</Entry>
|
<Entry name="SE_where_t">Till konto:</Entry>
|
||||||
@ -103,6 +103,13 @@ Till kontot: $2
|
|||||||
<Entry name="SE_exit">Logga ut</Entry>
|
<Entry name="SE_exit">Logga ut</Entry>
|
||||||
<Entry name="SE_open">Öppna ett konto</Entry>
|
<Entry name="SE_open">Öppna ett konto</Entry>
|
||||||
<Entry name="SE_close">Stäng konto</Entry>
|
<Entry name="SE_close">Stäng konto</Entry>
|
||||||
|
<Entry name="SE_delete">Radera användare</Entry>
|
||||||
|
<Entry name="SE_delete_warn">VARNING: Detta kommer att radera den nuvarande användaren
|
||||||
|
och alla kopplade konton!
|
||||||
|
Vill du fortsätta?</Entry>
|
||||||
|
<Entry name="SE_delete_stall">Raderar...</Entry>
|
||||||
|
<Entry name="SE_delete_success">Användare raderad</Entry>
|
||||||
|
<Entry name="SE_delete_failure">Användare kunde inte raderas</Entry>
|
||||||
<Entry name="SE_accounts">Visa konton</Entry>
|
<Entry name="SE_accounts">Visa konton</Entry>
|
||||||
<Entry name="SE_balance_toohigh">Angivet belopp är högre än det tillgängliga beloppet i ursprungskontot!
|
<Entry name="SE_balance_toohigh">Angivet belopp är högre än det tillgängliga beloppet i ursprungskontot!
|
||||||
Tillgängligt saldo: $0 SEK</Entry>
|
Tillgängligt saldo: $0 SEK</Entry>
|
||||||
|
@ -73,17 +73,19 @@ namespace Server
|
|||||||
public void RemoveUser(User entry) => RemoveUser(entry, true);
|
public void RemoveUser(User entry) => RemoveUser(entry, true);
|
||||||
private void RemoveUser(User entry, bool withFlush)
|
private void RemoveUser(User entry, bool withFlush)
|
||||||
{
|
{
|
||||||
entry = ToEncoded(entry);
|
// Remove from loaded users collection
|
||||||
for (int i = 0; i < loadedUsers.Count; ++i)
|
for (int i = 0; i < loadedUsers.Count; ++i)
|
||||||
if (entry.Equals(loadedUsers[i]))
|
if (entry.Name.Equals(loadedUsers[i].Name))
|
||||||
loadedUsers.RemoveAt(i);
|
loadedUsers.RemoveAt(i);
|
||||||
|
|
||||||
|
// Changes are retracted from change collectino
|
||||||
for (int i = changeList.Count - 1; i >= 0; --i)
|
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);
|
changeList.RemoveAt(i);
|
||||||
|
|
||||||
|
// Check if user already is scheduled for deletion
|
||||||
for (int i = toRemove.Count - 1; i >= 0; --i)
|
for (int i = toRemove.Count - 1; i >= 0; --i)
|
||||||
if (toRemove[i].Equals(entry.Name))
|
if (toRemove[i].Name.Equals(entry.Name))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
toRemove.Add(entry);
|
toRemove.Add(entry);
|
||||||
@ -104,9 +106,10 @@ namespace Server
|
|||||||
using(var reader = XmlReader.Create(DatabaseName))
|
using(var reader = XmlReader.Create(DatabaseName))
|
||||||
{
|
{
|
||||||
int masterDepth = 0;
|
int masterDepth = 0;
|
||||||
bool trigger = false, wn = false, recent = false;
|
bool trigger = false, wn = false, recent = false, justTriggered = false;
|
||||||
while (wn || reader.Read())
|
while (wn || reader.Read() || (reader.NodeType==XmlNodeType.None && justTriggered))
|
||||||
{
|
{
|
||||||
|
justTriggered = false;
|
||||||
wn = false;
|
wn = false;
|
||||||
if (trigger)
|
if (trigger)
|
||||||
{
|
{
|
||||||
@ -114,7 +117,7 @@ namespace Server
|
|||||||
WriteUser(writer, user);
|
WriteUser(writer, user);
|
||||||
|
|
||||||
bool wroteNode = false;
|
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;
|
wroteNode = false;
|
||||||
if (reader.Name.Equals("User"))
|
if (reader.Name.Equals("User"))
|
||||||
@ -154,6 +157,7 @@ namespace Server
|
|||||||
if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth]))
|
if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth]))
|
||||||
{
|
{
|
||||||
trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length;
|
trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length;
|
||||||
|
justTriggered = true;
|
||||||
reader.MoveToContent();
|
reader.MoveToContent();
|
||||||
writer.WriteStartElement(MasterEntry[masterDepth - 1]);
|
writer.WriteStartElement(MasterEntry[masterDepth - 1]);
|
||||||
}
|
}
|
||||||
@ -241,10 +245,18 @@ namespace Server
|
|||||||
public User FirstUser(Predicate<User> p)
|
public User FirstUser(Predicate<User> p)
|
||||||
{
|
{
|
||||||
if (p == null) return null; // Done to conveniently handle system insertions
|
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)
|
foreach (var entry in loadedUsers)
|
||||||
if (p(entry))
|
if (p(entry))
|
||||||
return entry;
|
return entry;
|
||||||
|
|
||||||
|
// Check modified users
|
||||||
foreach (var entry in changeList)
|
foreach (var entry in changeList)
|
||||||
if (p(entry))
|
if (p(entry))
|
||||||
{
|
{
|
||||||
@ -252,6 +264,7 @@ namespace Server
|
|||||||
return entry;
|
return entry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Read from database
|
||||||
using (var reader = XmlReader.Create(DatabaseName))
|
using (var reader = XmlReader.Create(DatabaseName))
|
||||||
{
|
{
|
||||||
if (!Traverse(reader, MasterEntry)) return null;
|
if (!Traverse(reader, MasterEntry)) return null;
|
||||||
|
@ -147,11 +147,23 @@ Use command 'help' to get a list of available commands";
|
|||||||
// Server endpoints
|
// Server endpoints
|
||||||
switch (cmd[0])
|
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)
|
case "Auth": // Log in to a user account (get a session id)
|
||||||
{
|
{
|
||||||
if(!ParseDataPair(cmd[1], out string user, out string pass))
|
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);
|
return ErrorResponse(id);
|
||||||
}
|
}
|
||||||
Database.User usr = db.GetUser(user);
|
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")}");
|
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)")
|
}), "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) => {
|
(c, l) => {
|
||||||
StringBuilder builder = new StringBuilder();
|
StringBuilder builder = new StringBuilder();
|
||||||
manager.Update(); // Ensure that we don't show expired sessions (artifacts exist until it is necessary to remove them)
|
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);
|
db.AddUser(user);
|
||||||
}
|
}
|
||||||
else Output.Raw(user.IsAdministrator);
|
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
|
// Set up a persistent terminal-esque input design
|
||||||
Output.OnNewLine = () => Output.WriteOverwritable(">> ");
|
Output.OnNewLine = () => Output.WriteOverwritable(">> ");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user