This repository has been archived on 2025-01-01. You can view files and clone it, but cannot push or open issues or pull requests.
ShiftOS_TheReturn/ShiftOS.Frontend/Apps/Terminal.cs

452 lines
15 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
2017-07-04 09:52:49 -04:00
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Text;
2017-07-04 09:52:49 -04:00
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using Microsoft.Xna.Framework;
2017-07-04 09:52:49 -04:00
using Microsoft.Xna.Framework.Input;
using ShiftOS.Engine;
using ShiftOS.Frontend.GraphicsSubsystem;
using static ShiftOS.Engine.SkinEngine;
namespace ShiftOS.Frontend.Apps
{
[FileHandler("Shell script", ".trm", "fileicontrm")]
[Launcher("{TITLE_TERMINAL}", false, null, "{AL_UTILITIES}")]
[WinOpen("{WO_TERMINAL}")]
[DefaultTitle("{TITLE_TERMINAL}")]
[DefaultIcon("iconTerminal")]
public class Terminal : GUI.Control, IShiftOSWindow
{
private TerminalControl _terminal = null;
public Terminal()
{
Width = 493;
Height = 295;
}
public void OnLoad()
{
_terminal = new Apps.TerminalControl();
_terminal.Dock = GUI.DockStyle.Fill;
AddControl(_terminal);
AppearanceManager.ConsoleOut = _terminal;
AppearanceManager.StartConsoleOut();
TerminalBackend.PrintPrompt();
SaveSystem.GameReady += () =>
{
Console.WriteLine("[sessionmgr] Starting system UI...");
AppearanceManager.SetupWindow(new SystemStatus());
TerminalBackend.PrintPrompt();
};
}
protected override void OnLayout(GameTime gameTime)
{
if (ContainsFocusedControl || IsFocusedControl)
AppearanceManager.ConsoleOut = _terminal;
}
public void OnSkinLoad()
{
}
public bool OnUnload()
{
return true;
}
public void OnUpgrade()
{
}
}
public class TerminalControl : GUI.TextInput, ITerminalWidget
{
2017-07-04 09:52:49 -04:00
public TerminalControl()
{
Dock = GUI.DockStyle.Fill;
}
public string[] Lines
{
get
{
2017-07-04 09:52:49 -04:00
return Text.Split(new[] { "\r\n" }, StringSplitOptions.None);
}
}
public void Clear()
{
Text = "";
Index = 0;
_vertOffset = 0;
Invalidate();
}
public void SelectBottom()
{
Index = Text.Length - 1;
RecalculateLayout();
InvalidateTopLevel();
}
public void Write(string text)
{
Engine.Desktop.InvokeOnWorkerThread(() =>
{
Text += Localization.Parse(text);
SelectBottom();
Index += text.Length;
RecalculateLayout();
InvalidateTopLevel();
});
}
public void WriteLine(string text)
{
Write(text + Environment.NewLine);
}
public int GetCurrentLine()
{
int line = 0;
2017-07-19 13:06:39 -04:00
for(int i = 0; i < Index; i++)
{
if(Text[i]=='\n')
{
line++;
continue;
}
}
2017-07-19 13:06:39 -04:00
return line;
}
float _vertOffset = 0.0f;
protected void RecalculateLayout()
{
}
private bool blinkStatus = false;
private double blinkTime = 0.0;
protected override void OnLayout(GameTime gameTime)
2017-07-04 09:52:49 -04:00
{
blinkTime += gameTime.ElapsedGameTime.TotalMilliseconds;
if (blinkTime > 500.0)
blinkTime = 0;
bool prev = blinkStatus;
blinkStatus = blinkTime > 250.0;
if (prev != blinkStatus)
Invalidate();
2017-07-04 09:52:49 -04:00
}
public bool ReadOnly = false;
/// <summary>
/// Gets the X and Y coordinates (in pixels) of the caret.
/// </summary>
/// <param name="gfx">A <see cref="System.Drawing.Graphics"/> object used for font measurements</param>
/// <returns>An absolute fucking mess. Seriously, can someone fix this method so it uhh WORKS PROPERLY?</returns>
public System.Drawing.Point GetPointAtIndex(Graphics gfx)
{
2017-07-04 09:52:49 -04:00
int vertMeasure = 2;
int horizMeasure = 2;
if (string.IsNullOrEmpty(Text))
return new System.Drawing.Point(horizMeasure, vertMeasure);
int lineindex = 0;
int line = GetCurrentLine();
for (int l = 0; l < line; l++)
2017-07-04 09:52:49 -04:00
{
if (string.IsNullOrEmpty(Lines[l]))
{
vertMeasure += LoadedSkin.TerminalFont.Height;
continue;
}
lineindex += Lines[l].Length;
2017-07-05 08:43:35 -04:00
var stringMeasure = gfx.SmartMeasureString(Lines[l] == "\r" ? " " : Lines[l], LoadedSkin.TerminalFont, Width - 4);
vertMeasure += (int)stringMeasure.Height;
2017-07-04 09:52:49 -04:00
}
var lnMeasure = gfx.SmartMeasureString(Text.Substring(lineindex, Index - lineindex), LoadedSkin.TerminalFont);
int w = (int)Math.Floor(lnMeasure.Width);
2017-07-19 13:06:39 -04:00
while (w > Width - 4)
{
w = w - (Width - 4);
2017-07-19 13:06:39 -04:00
vertMeasure += (int)lnMeasure.Height;
}
horizMeasure += w;
return new System.Drawing.Point(horizMeasure, vertMeasure);
}
private PointF CaretPosition = new PointF(2, 2);
private Size CaretSize = new Size(2, 15);
2017-07-04 09:52:49 -04:00
protected override void OnKeyEvent(KeyEvent a)
{
if (a.Key == Keys.Enter && !ReadOnly)
{
if (!PerformTerminalBehaviours)
{
Text = Text.Insert(Index, Environment.NewLine);
Index+=2;
RecalculateLayout();
Invalidate();
return;
}
2017-07-04 09:52:49 -04:00
try
{
if (!TerminalBackend.PrefixEnabled)
{
string textraw = Lines[Lines.Length - 1];
TerminalBackend.SendText(textraw);
return;
}
var text = Lines;
var text2 = text[text.Length - 1];
var text3 = "";
var text4 = Regex.Replace(text2, @"\t|\n|\r", "");
WriteLine("");
2017-07-04 09:52:49 -04:00
if (TerminalBackend.PrefixEnabled)
{
text3 = text4.Remove(0, $"{SaveSystem.CurrentSave.Username}@{SaveSystem.CurrentSave.SystemName}:~$ ".Length);
2017-07-04 09:52:49 -04:00
}
if (!string.IsNullOrWhiteSpace(text3))
{
2017-07-04 09:52:49 -04:00
TerminalBackend.LastCommand = text3;
TerminalBackend.SendText(text4);
if (TerminalBackend.InStory == false)
{
{
var result = SkinEngine.LoadedSkin.CurrentParser.ParseCommand(text3);
if (result.Equals(default(KeyValuePair<string, Dictionary<string, string>>)))
{
Console.WriteLine("{ERR_SYNTAXERROR}");
}
else
{
TerminalBackend.InvokeCommand(result.Key, result.Value);
}
}
}
}
}
catch
{
}
finally
{
if (TerminalBackend.PrefixEnabled)
{
TerminalBackend.PrintPrompt();
}
AppearanceManager.CurrentPosition = 0;
}
2017-07-04 09:52:49 -04:00
}
else if (a.Key == Keys.Back && !ReadOnly)
2017-07-04 09:52:49 -04:00
{
try
{
if (PerformTerminalBehaviours)
{
var tostring3 = Lines[Lines.Length - 1];
var tostringlen = tostring3.Length + 1;
var workaround = $"{SaveSystem.CurrentSave.Username}@{SaveSystem.CurrentSave.SystemName}:~$ ";
var derp = workaround.Length + 1;
if (tostringlen != derp)
{
AppearanceManager.CurrentPosition--;
base.OnKeyEvent(a);
RecalculateLayout();
InvalidateTopLevel();
}
}
else
2017-07-04 09:52:49 -04:00
{
base.OnKeyEvent(a);
RecalculateLayout();
InvalidateTopLevel();
}
}
catch
{
Debug.WriteLine("Drunky alert in terminal.");
}
}
else if(a.Key == Keys.Right)
{
if(Index < Text.Length)
{
Index++;
AppearanceManager.CurrentPosition++;
RecalculateLayout();
InvalidateTopLevel();
}
}
2017-07-04 09:52:49 -04:00
else if (a.Key == Keys.Left)
{
if (SaveSystem.CurrentSave != null)
{
var getstring = Lines[Lines.Length - 1];
var stringlen = getstring.Length + 1;
var header = $"{SaveSystem.CurrentSave.Username}@{SaveSystem.CurrentSave.SystemName}:~$ ";
2017-07-04 09:52:49 -04:00
var headerlen = header.Length + 1;
var selstart = Index;
var remstrlen = Text.Length - stringlen;
var finalnum = selstart - remstrlen;
if (!PerformTerminalBehaviours)
headerlen = 0;
if (finalnum > headerlen)
2017-07-04 09:52:49 -04:00
{
AppearanceManager.CurrentPosition--;
base.OnKeyEvent(a);
}
}
}
else if (a.Key == Keys.Up && PerformTerminalBehaviours)
2017-07-04 09:52:49 -04:00
{
var tostring3 = Lines[Lines.Length - 1];
if (tostring3 == $"{SaveSystem.CurrentSave.Username}@{SaveSystem.CurrentSave.SystemName}:~$ ")
2017-07-04 09:52:49 -04:00
Console.Write(TerminalBackend.LastCommand);
ConsoleEx.OnFlush?.Invoke();
return;
}
else
{
if ((PerformTerminalBehaviours && TerminalBackend.InStory) || ReadOnly)
2017-07-04 09:52:49 -04:00
{
return;
}
if (a.KeyChar != '\0')
{
Text = Text.Insert(Index, a.KeyChar.ToString());
Index++;
2017-07-04 09:52:49 -04:00
AppearanceManager.CurrentPosition++;
// RecalculateLayout();
2017-07-04 09:52:49 -04:00
InvalidateTopLevel();
}
}
blinkStatus = true;
blinkTime = 250;
}
public bool PerformTerminalBehaviours = true;
protected override void OnPaint(GraphicsContext gfx)
2017-07-04 09:52:49 -04:00
{
gfx.Clear(LoadedSkin.TerminalBackColorCC.ToColor().ToMonoColor());
if (!string.IsNullOrEmpty(Text))
{
//Draw the caret.
if (blinkStatus == true)
{
PointF cursorPos;
using (var cgfx = System.Drawing.Graphics.FromImage(new System.Drawing.Bitmap(1, 1)))
{
cursorPos = GetPointAtIndex(cgfx);
}
var cursorSize = gfx.MeasureString("#", LoadedSkin.TerminalFont);
var lineMeasure = gfx.MeasureString(Lines[GetCurrentLine()], LoadedSkin.TerminalFont);
if (cursorPos.X > lineMeasure.X)
{
cursorPos.X = lineMeasure.X;
}
2017-07-19 13:06:39 -04:00
gfx.DrawRectangle((int)cursorPos.X, (int)cursorPos.Y - (int)_vertOffset, (int)cursorSize.X, (int)cursorSize.Y, LoadedSkin.TerminalForeColorCC.ToColor().ToMonoColor());
2017-07-19 13:06:39 -04:00
}
//Draw the text
gfx.DrawString(Text, 2, 2 - (int)Math.Floor(_vertOffset), LoadedSkin.TerminalForeColorCC.ToColor().ToMonoColor(), LoadedSkin.TerminalFont, Width - 4);
}
}
}
public static class ConsoleColorExtensions
{
public static System.Drawing.Color ToColor(this ConsoleColor cc)
{
switch (cc)
{
case ConsoleColor.Black:
return System.Drawing.Color.Black;
case ConsoleColor.Blue:
return System.Drawing.Color.Blue;
case ConsoleColor.Cyan:
return System.Drawing.Color.Cyan;
case ConsoleColor.DarkBlue:
return System.Drawing.Color.DarkBlue;
case ConsoleColor.DarkCyan:
return System.Drawing.Color.DarkCyan;
case ConsoleColor.DarkGray:
return System.Drawing.Color.DarkGray;
case ConsoleColor.DarkGreen:
return System.Drawing.Color.DarkGreen;
case ConsoleColor.DarkMagenta:
return System.Drawing.Color.DarkMagenta;
case ConsoleColor.DarkRed:
return System.Drawing.Color.DarkRed;
case ConsoleColor.DarkYellow:
return System.Drawing.Color.Orange;
case ConsoleColor.Gray:
return System.Drawing.Color.Gray;
case ConsoleColor.Green:
return System.Drawing.Color.Green;
case ConsoleColor.Magenta:
return System.Drawing.Color.Magenta;
case ConsoleColor.Red:
return System.Drawing.Color.Red;
case ConsoleColor.White:
return System.Drawing.Color.White;
case ConsoleColor.Yellow:
return System.Drawing.Color.Yellow;
}
return System.Drawing.Color.Empty;
}
}
public static class GraphicsExtensions
{
public static SizeF SmartMeasureString(this Graphics gfx, string s, Font font, int width)
{
var textformat = new StringFormat(StringFormat.GenericTypographic);
textformat.FormatFlags |= StringFormatFlags.MeasureTrailingSpaces;
//textformat.Trimming = StringTrimming.Character;
//textformat.FormatFlags |= StringFormatFlags.NoClip;
2017-07-05 08:43:35 -04:00
gfx.TextRenderingHint = System.Drawing.Text.TextRenderingHint.AntiAliasGridFit;
2017-07-04 09:52:49 -04:00
var measure = gfx.MeasureString(s, font, width, textformat);
if (string.IsNullOrEmpty(s))
measure.Width = 0;
return new SizeF((float)Math.Ceiling(Math.Max(1,measure.Width)), (float)Math.Ceiling(Math.Max(1,measure.Height)));
}
public static SizeF SmartMeasureString(this Graphics gfx, string s, Font font)
{
2017-07-05 08:43:35 -04:00
return SmartMeasureString(gfx, s, font, int.MaxValue);
}
}
}