using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using System;
using System.Text;
using System.Reflection;
using System.Linq;
namespace ConsoleUtility
{
public class Console : MonoBehaviour
{
[Header("Keys")]
public KeyCode ToggleKey = KeyCode.F12;
public KeyCode PreviousCommandKey = KeyCode.UpArrow;
public KeyCode NextCommandKey = KeyCode.DownArrow;
public KeyCode ScrollUpKey = KeyCode.PageUp;
public KeyCode ScrollDownKey = KeyCode.PageDown;
public KeyCode ValidateKey = KeyCode.Return;
[Header("Items")]
public Canvas Canvas;
public InputField InputField;
public Text LogText;
public Text ScrollInfo;
public GameObject AutoPanelRoot;
public Text AutoPanelText;
[Header("Settings")]
[Range(1.0f, 30.0f)]
public float ScrollSpeed = 5.0f;
private static ConsoleData s_ConsoleData;
private static Console s_Console;
private bool bVisible = false;
private int history = -1;
void OnEnable()
{
if(s_ConsoleData == null)
{
s_ConsoleData = new ConsoleData();
s_ConsoleData.AutoRegisterConsoleCommands();
}
s_ConsoleData.OnLogUpdated = UpdateLog;
s_Console = this;
Application.logMessageReceived += HandleUnityLog;
LogText.font.RequestCharactersInTexture("qwertyuiopasdfghjklzxcvbnmQWERYTUIOPASDFGHJKLZXCVBNM1234567890~`!@#$%^&*()_+{}[]:;\"'/.,?><");
Log("Console initialized successfully");
UpdateLog();
}
void OnDisable()
{
s_ConsoleData.OnLogUpdated = null;
s_Console = null;
Application.logMessageReceived -= HandleUnityLog;
}
public void UpdateAutoPanel()
{
AutoPanelRoot.SetActive(InputField.text != "");
string input = InputField.text;
string[] words = input.Split(' ');
if(s_ConsoleData.aliases.ContainsKey(words[0].ToUpper()))
{
StringBuilder sb = new StringBuilder();
sb.AppendFormat("Alias {0} => {1}:\n", words[0], s_ConsoleData.aliases[words[0].ToUpper()]);
var command = s_ConsoleData.commands[s_ConsoleData.aliases[words[0].ToUpper()].Split(' ').FirstOrDefault().ToUpper()];
sb.Append(command.help);
AutoPanelText.text = sb.ToString();
}
else if (s_ConsoleData.commands.ContainsKey(words[0].ToUpper()))
{
StringBuilder sb = new StringBuilder();
var command = s_ConsoleData.commands[words[0].ToUpper()];
sb.Append(command.help);
AutoPanelText.text = sb.ToString();
}
else
{
StringBuilder sb = new StringBuilder();
foreach (var word in s_ConsoleData.aliases.Keys.Concat(s_ConsoleData.commands.Keys).Where(o => o.Contains(words[0].ToUpper())).ToArray())
{
bool isAlias = s_ConsoleData.aliases.ContainsKey(word);
string append = isAlias ? string.Format("(alias for '{0}')",s_ConsoleData.aliases[word.ToUpper()]) : s_ConsoleData.commands[word].summary;
sb.AppendFormat("", isAlias ? "orange" : "yellow");
sb.Append(word.ToLower());
sb.Append(" ");
sb.Append(append);
sb.Append("\n");
}
AutoPanelText.text = sb.ToString();
}
}
void Update()
{
if (Input.GetKeyDown(ToggleKey))
ToggleVisibility();
if (!bVisible) return;
if (Input.GetKeyDown(PreviousCommandKey))
{
history++;
if (history > s_ConsoleData.commandHistory.Count - 1)
history = s_ConsoleData.commandHistory.Count - 1;
if (history >= 0)
InputField.text = s_ConsoleData.commandHistory[history];
InputField.MoveTextEnd(false);
}
else if (Input.GetKeyDown(NextCommandKey))
{
history--;
if (history <= -1)
{
InputField.text = "";
history = -1;
}
else
InputField.text = s_ConsoleData.commandHistory[history];
InputField.MoveTextEnd(false);
}
else if (Input.GetKeyDown(ValidateKey))
{
ValidateCommand();
}
else if (Input.GetKeyDown(ScrollUpKey))
{
ScrollUp();
}
else if (Input.GetKeyDown(ScrollDownKey))
{
ScrollDown();
}
}
public static void SetVisible(bool visible)
{
s_Console.SetVisibility(visible);
}
public static void CaptureScreenshot(string filename, int size)
{
s_Console.StartCoroutine(s_Console.Screenshot(filename, size));
}
public IEnumerator Screenshot(string filename, int size)
{
Console.SetVisible(false);
yield return new WaitForEndOfFrame();
ScreenCapture.CaptureScreenshot(filename, 1);
Console.SetVisible(true);
}
public void ToggleVisibility()
{
SetVisibility(!bVisible);
}
public void SetVisibility(bool visible)
{
if (bVisible == visible)
return;
bVisible = visible;
Canvas.gameObject.SetActive(bVisible);
if (bVisible)
{
InputField.text = "";
InputField.Select();
InputField.MoveTextStart(false);
InputField.ActivateInputField();
UpdateLog();
}
}
public static void ExecuteCommand(string command)
{
string[] words = command.Split(' ');
if (s_ConsoleData.commands.ContainsKey(words[0].ToUpper()))
{
s_ConsoleData.commands[words[0].ToUpper()].Execute(words.Skip(1).ToArray());
}
else if (s_ConsoleData.aliases.ContainsKey(words[0].ToUpper()))
{
string alias = words[0];
string aliascommand = command.Replace(alias, s_ConsoleData.aliases[alias.ToUpper()]);
string[] aliaswords = aliascommand.Split(' ');
s_ConsoleData.commands[aliaswords[0].ToUpper()].Execute(aliaswords.Skip(1).ToArray());
}
else
{
Log("Unknown Command: " + words[0]);
}
// Ensure no duplicates in history
if (s_ConsoleData.commandHistory.Count == 0 || command != s_ConsoleData.commandHistory[0])
s_ConsoleData.commandHistory.Insert(0, command);
}
public void ValidateCommand()
{
if (InputField.text == "")
return;
string command = InputField.text;
ExecuteCommand(command);
InputField.text = "";
InputField.Select();
InputField.MoveTextStart(false);
InputField.ActivateInputField();
history = -1;
UpdateLog();
}
public static void Log(string Message)
{
Log(string.Empty, Message, LogType.Log);
}
public static void Log(string Command, string Message, LogType type = LogType.Log)
{
string prepend = "";
if (Command != string.Empty)
{
string color = "white";
switch(type)
{
case LogType.Assert:
case LogType.Error:
case LogType.Exception:
color = "red";
break;
case LogType.Warning:
color = "orange";
break;
default:
case LogType.Log:
color = "lime";
break;
}
prepend = string.Format("[{0}] ", Command.ToUpper(), color);
}
string[] lines = Message.Split('\n');
foreach(string line in lines)
s_ConsoleData.lines.Add("[" + Time.unscaledTime.ToString("F3") + "] " + prepend +line);
if (s_ConsoleData.OnLogUpdated != null)
s_ConsoleData.OnLogUpdated.Invoke();
}
private static void HandleUnityLog(string logString, string stackTrace, LogType type)
{
Log("UNITY", string.Format("[{0}] : {1}", type, logString), type);
if(type == LogType.Error || type == LogType.Exception)
{
Log(stackTrace);
}
}
private int m_Scroll=-1;
private int GetCapacity()
{
return (int)(LogText.rectTransform.rect.height / (LogText.font.lineHeight + LogText.lineSpacing));
}
private int GetScrollPageSize()
{
float lineSize = LogText.lineSpacing + LogText.font.fontSize;
int count = Mathf.FloorToInt(LogText.rectTransform.rect.height / lineSize);
return count;
}
private void ScrollUp()
{
int pageSize = GetScrollPageSize();
if (s_ConsoleData.lines.Count < pageSize)
return;
if(m_Scroll == -1)
{
m_Scroll = s_ConsoleData.lines.Count - pageSize;
}
m_Scroll = Math.Max(0, m_Scroll - pageSize);
UpdateLog();
}
private void ScrollDown()
{
int pageSize = GetScrollPageSize();
if (s_ConsoleData.lines.Count < pageSize)
return;
if (m_Scroll == -1)
return;
m_Scroll += pageSize;
if (m_Scroll >= (s_ConsoleData.lines.Count - pageSize))
m_Scroll = -1; // Snap Again
UpdateLog();
}
private string GetScrolledText()
{
int pageSize = GetScrollPageSize();
int count = s_ConsoleData.lines.Count;
CharacterInfo info;
LogText.font.GetCharacterInfo('X', out info);
int maxCharsInLine = Mathf.FloorToInt(LogText.rectTransform.rect.width / info.advance);
int init = m_Scroll == -1 ? count - pageSize : m_Scroll;
StringBuilder sb = new StringBuilder();
for (int i = init; i < (init + pageSize); i++)
{
if (i < 0)
continue;
string line = s_ConsoleData.lines[i];
if (line.Length > maxCharsInLine)
line = line.Substring(0, maxCharsInLine - 4) + " ...";
sb.AppendLine(line);
}
return sb.ToString();
}
private void UpdateLog()
{
if (s_ConsoleData.lines.Count == 0 || !bVisible) return;
// Ensure m_Scroll is Consistent when clearing;
if (m_Scroll > s_ConsoleData.lines.Count)
m_Scroll = -1;
// Update Log Text
if (LogText != null)
{
LogText.text = GetScrolledText();
LogText.GraphicUpdateComplete();
}
// Update Scroll Info Text
if (ScrollInfo != null)
{
int pageSize = GetScrollPageSize();
int count = s_ConsoleData.lines.Count;
string text = string.Empty;
if (m_Scroll == -1)
{
ScrollInfo.text = $"*[{Mathf.FloorToInt(count / pageSize)}/{Mathf.CeilToInt(count / pageSize)}]";
}
else
{
ScrollInfo.text = $"[{Mathf.FloorToInt(m_Scroll / pageSize)}/{Mathf.CeilToInt(count / pageSize)}]";
}
ScrollInfo.GraphicUpdateComplete();
}
}
public static void Clear()
{
s_ConsoleData.lines.Clear();
Log("Console","Cleared Output", LogType.Log);
if (s_ConsoleData.OnLogUpdated != null)
s_ConsoleData.OnLogUpdated.Invoke();
}
public static IConsoleCommand[] listAllCommands()
{
return s_ConsoleData.commands.Values.ToArray();
}
public static bool HasCommand(string name)
{
return s_ConsoleData.commands.ContainsKey(name);
}
public static void AddCommand(IConsoleCommand command)
{
// Add Command
s_ConsoleData.commands.Add(command.name.ToUpper(), command);
// Add Aliases
var aliases = command.aliases;
if(aliases != null)
foreach (var alias in aliases)
{
if (!alias.AliasString.Contains(" "))
s_ConsoleData.aliases.Add(alias.AliasString.ToUpper(), alias.Command);
else
Debug.LogError(string.Format("Cannot add alias for command {0} : alias '{1}' contains spaces", command.name, alias.AliasString));
}
}
public struct Alias
{
public string AliasString;
public string Command;
public static Alias Get(string alias, string command)
{
return new Alias { AliasString = alias, Command = command };
}
}
public class ConsoleData
{
public List lines;
public Dictionary commands;
public Dictionary aliases;
public System.Action OnLogUpdated;
public List commandHistory;
public ConsoleData()
{
lines = new List();
commands = new Dictionary();
aliases = new Dictionary();
commandHistory = new List();
}
public void AutoRegisterConsoleCommands()
{
// Use reflection to add automatically console commands
var autoCommandTypes = new List();
foreach (Assembly assembly in AppDomain.CurrentDomain.GetAssemblies())
{
foreach (Type type in assembly.GetTypes())
{
if (type.GetInterfaces().Contains(typeof(IConsoleCommand)) && type.GetCustomAttributes(typeof(AutoRegisterConsoleCommandAttribute), true).Length > 0)
{
autoCommandTypes.Add(type);
}
}
}
foreach (var type in autoCommandTypes)
{
AddCommand(Activator.CreateInstance(type) as IConsoleCommand);
}
}
}
[AutoRegisterConsoleCommand]
public class HelpCommand : IConsoleCommand
{
public void Execute(string[] args)
{
if (args.Length > 0)
{
var commands = Console.listAllCommands();
var command = commands.FirstOrDefault((o) => o.name == args[0]);
if (command != null)
{
Console.Log(name, "Help for command : " + command.name);
Console.Log("Summary : " + command.summary);
string[] helpText = command.help.Replace("\r", "").Split('\n');
foreach (var line in helpText)
{
Console.Log(" " + line);
}
}
else
Console.Log(name, "Could not find help for command " + args[0]);
}
else
{
Console.Log(name, "Available Commands:");
foreach (var command in Console.listAllCommands())
{
Console.Log(command.name + " : " + command.summary);
}
}
}
public string name => "help";
public string summary =>"Gets a summary of all commands or shows help for a specific command.";
public string help => @"Usage: HELP command
Shows help for specific command or list any available command.
Additional arguments are ignored";
public IEnumerable aliases => null;
}
}
[AutoRegisterConsoleCommand]
public class Clear : IConsoleCommand
{
public void Execute(string[] args)
{
Console.Clear();
}
public string name => "clear";
public string summary => "Clears the console output";
public string help => @"Clear
Usage: clear";
public IEnumerable aliases
{
get
{
yield return Console.Alias.Get("cls", "clear");
}
}
}
public interface IConsoleCommand
{
void Execute(string[] args);
string name { get; }
string summary { get; }
string help { get; }
IEnumerable aliases { get; }
}
public class AutoRegisterConsoleCommandAttribute : System.Attribute { }
}