2017-01-08 15:17:07 +00:00
|
|
|
/*
|
|
|
|
* MIT License
|
|
|
|
*
|
|
|
|
* Copyright (c) 2017 Michael VanOverbeek and ShiftOS devs
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define MUD_RAPIDDEV
|
2017-01-08 14:57:10 +00:00
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Drawing;
|
|
|
|
using System.Linq;
|
2017-02-04 16:14:39 +00:00
|
|
|
using System.Reflection;
|
2017-01-08 14:57:10 +00:00
|
|
|
using System.Text;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using System.Windows.Forms;
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
using static ShiftOS.Engine.SaveSystem;
|
|
|
|
|
|
|
|
namespace ShiftOS.Engine
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides functionality for managing windows within ShiftOS.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static class AppearanceManager
|
|
|
|
{
|
|
|
|
[Obsolete("Please use Localization.GetAllLanguages().")]
|
|
|
|
public static string[] GetLanguages()
|
|
|
|
{
|
|
|
|
return Localization.GetAllLanguages();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Sets the title text of the specified window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="window">The window to modify</param>
|
|
|
|
/// <param name="title">The title text to use</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if the window is null.</exception>
|
2017-02-07 01:15:22 +00:00
|
|
|
public static void SetWindowTitle(IShiftOSWindow window, string title)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (window == null)
|
|
|
|
throw new ArgumentNullException("window", "The window cannot be null.");
|
2017-02-07 01:15:22 +00:00
|
|
|
winmgr.SetTitle(window, title);
|
|
|
|
}
|
|
|
|
|
2017-02-04 16:14:39 +00:00
|
|
|
public static IEnumerable<Type> GetAllWindowTypes()
|
|
|
|
{
|
|
|
|
List<Type> types = new List<Type>();
|
|
|
|
foreach(var file in System.IO.Directory.GetFiles(Environment.CurrentDirectory))
|
|
|
|
{
|
|
|
|
if(file.EndsWith(".exe") || file.EndsWith(".dll"))
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
var asm = Assembly.LoadFile(file);
|
|
|
|
foreach(var type in asm.GetTypes())
|
|
|
|
{
|
|
|
|
if (type.GetInterfaces().Contains(typeof(IShiftOSWindow)))
|
|
|
|
types.Add(type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return types;
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Returns the default window title for a specified <see cref="IShiftOSWindow"/>-inheriting type.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="winType">The type to scan</param>
|
|
|
|
/// <returns>The default title</returns>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="winType"/> is null.</exception>
|
2017-02-04 16:14:39 +00:00
|
|
|
public static string GetDefaultTitle(Type winType)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (winType == null)
|
|
|
|
throw new ArgumentNullException("winType");
|
2017-02-04 16:14:39 +00:00
|
|
|
foreach(var attrib in winType.GetCustomAttributes(false))
|
|
|
|
{
|
|
|
|
if(attrib is DefaultTitleAttribute)
|
|
|
|
{
|
|
|
|
return (attrib as DefaultTitleAttribute).Title;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return winType.Name;
|
|
|
|
}
|
2017-01-08 14:57:10 +00:00
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Current cursor position of the console
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static int CurrentPosition { get; set; }
|
2017-04-16 12:06:17 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// We don't know what this does. It may be gone if it does nothing.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static int LastLength { get; set; }
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Minimize a window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="form">The window border to minimize.</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="form"/> is null.</exception>
|
|
|
|
/// <exception cref="EngineModuleDisabledException">Thrown if this part of the engine hasn't been enabled.</exception>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void Minimize(IWindowBorder form)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (form == null)
|
|
|
|
throw new ArgumentNullException("form");
|
|
|
|
if (winmgr == null)
|
|
|
|
throw new EngineModuleDisabledException();
|
2017-01-08 14:57:10 +00:00
|
|
|
winmgr.Minimize(form);
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Maximizes a window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="form">The window border to maximize.</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="form"/> is null.</exception>
|
|
|
|
/// <exception cref="EngineModuleDisabledException">Thrown if this engine module hasn't been enabled.</exception>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void Maximize(IWindowBorder form)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (form == null)
|
|
|
|
throw new ArgumentNullException("form");
|
|
|
|
if (winmgr == null)
|
|
|
|
throw new EngineModuleDisabledException();
|
2017-01-08 14:57:10 +00:00
|
|
|
winmgr.Maximize(form);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides a list of all open ShiftOS windows.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static List<IWindowBorder> OpenForms = new List<IWindowBorder>();
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Decorates a window with a border, then shows the window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="form">The window to decorate and show.</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="form"/> is null. </exception>
|
|
|
|
/// <exception cref="EngineModuleDisabledException">Thrown if this engine module has not been initiated yet.</exception>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void SetupWindow(IShiftOSWindow form)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (form == null)
|
|
|
|
throw new ArgumentNullException("form");
|
|
|
|
if (winmgr == null)
|
|
|
|
throw new EngineModuleDisabledException();
|
2017-01-08 14:57:10 +00:00
|
|
|
winmgr.SetupWindow(form);
|
|
|
|
Desktop.ResetPanelButtons();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Closes the specified window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="win">The window to close.</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="win"/> is null. </exception>
|
|
|
|
/// <exception cref="EngineModuleDisabledException">Thrown if this engine module has not been initiated yet.</exception>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void Close(IShiftOSWindow win)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (win == null)
|
|
|
|
throw new ArgumentNullException("win");
|
|
|
|
if (winmgr == null)
|
|
|
|
throw new EngineModuleDisabledException();
|
2017-01-08 14:57:10 +00:00
|
|
|
winmgr.Close(win);
|
|
|
|
Desktop.ResetPanelButtons();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Decorates a window with a border, then shows the window, as a dialog box.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="form">The window to decorate and show.</param>
|
|
|
|
/// <exception cref="ArgumentNullException">Thrown if <paramref name="form"/> is null. </exception>
|
|
|
|
/// <exception cref="EngineModuleDisabledException">Thrown if this engine module has not been initiated yet.</exception>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void SetupDialog(IShiftOSWindow form)
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
if (form == null)
|
|
|
|
throw new ArgumentNullException("form");
|
|
|
|
if (winmgr == null)
|
|
|
|
throw new EngineModuleDisabledException();
|
2017-01-08 14:57:10 +00:00
|
|
|
winmgr.SetupDialog(form);
|
|
|
|
Desktop.ResetPanelButtons();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// The underlying window manager for this engine module
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
private static WindowManager winmgr = null;
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Initiate this engine module, and perform mandatory configuration.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="mgr">A working, configured <see cref="WindowManager"/> to use as a backend for this module </param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void Initiate(WindowManager mgr)
|
|
|
|
{
|
|
|
|
winmgr = mgr;
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Raised when the engine is entering its shutdown phase. Save your work!
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static event EmptyEventHandler OnExit;
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Starts the engine's exit routine, firing the OnExit event.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
internal static void Exit()
|
|
|
|
{
|
|
|
|
OnExit?.Invoke();
|
|
|
|
//disconnect from MUD
|
|
|
|
ServerManager.Disconnect();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// The current terminal body control.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static ITerminalWidget ConsoleOut { get; set; }
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Redirects the .NET <see cref="Console"/> to a new <see cref="TerminalTextWriter"/> instance.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void StartConsoleOut()
|
|
|
|
{
|
|
|
|
Console.SetOut(new TerminalTextWriter());
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Invokes an action on the window management thread.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="act">The action to invoke</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public static void Invoke(Action act)
|
|
|
|
{
|
|
|
|
winmgr.InvokeAction(act);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides the base functionality for a ShiftOS terminal.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public interface ITerminalWidget
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Write text to this Terminal.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="text">Text to write</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
void Write(string text);
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Write text to this Terminal, followed by a newline.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="text">Text to write.</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
void WriteLine(string text);
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Clear the contents of this Terminal.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
void Clear();
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Move the cursor to the last character in the Terminal.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
void SelectBottom();
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides the base functionality for a ShiftOS window manager.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract class WindowManager
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Minimizes a window
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="border">The window border to minimize</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void Minimize(IWindowBorder border);
|
2017-04-16 12:06:17 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Maximizes a window
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="border">The window border to maximize</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void Maximize(IWindowBorder border);
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Closes a window
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="win">The window to close</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void Close(IShiftOSWindow win);
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Decorates a window with a window border, then shows it to the user.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="win">The window to decorate.</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void SetupWindow(IShiftOSWindow win);
|
2017-04-16 12:06:17 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Decorates a window with a border, then shows it to the user as a dialog box.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="win">The window to decorate</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void SetupDialog(IShiftOSWindow win);
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Invokes an action on the window management thread.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="act">The action to invoke.</param>
|
2017-01-08 14:57:10 +00:00
|
|
|
public abstract void InvokeAction(Action act);
|
2017-02-07 01:15:22 +00:00
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Sets the title text of a window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="win">The window to modify.</param>
|
|
|
|
/// <param name="title">The new title text.</param>
|
2017-02-07 01:15:22 +00:00
|
|
|
public abstract void SetTitle(IShiftOSWindow win, string title);
|
2017-01-08 14:57:10 +00:00
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides the base functionality for a typical ShiftOS window border.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
public interface IWindowBorder
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Closes the border along with its window. Unload events should be invoked here.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
void Close();
|
2017-04-16 12:06:17 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the title text for the window border.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
string Text { get; set; }
|
2017-04-16 12:06:17 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the underlying <see cref="IShiftOSWindow"/> for this border.
|
|
|
|
/// </summary>
|
2017-01-08 14:57:10 +00:00
|
|
|
IShiftOSWindow ParentWindow { get; set; }
|
|
|
|
}
|
2017-02-04 16:14:39 +00:00
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Provides a way of setting default title text for <see cref="IShiftOSWindow"/> classes.
|
|
|
|
/// </summary>
|
2017-02-04 16:14:39 +00:00
|
|
|
public class DefaultTitleAttribute : Attribute
|
|
|
|
{
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Creates a new instance of the <see cref="DefaultTitleAttribute"/>.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="title">A default title to associate with this attribute.</param>
|
2017-02-04 16:14:39 +00:00
|
|
|
public DefaultTitleAttribute(string title)
|
|
|
|
{
|
|
|
|
Title = title;
|
|
|
|
}
|
|
|
|
|
|
|
|
public string Title { get; private set; }
|
|
|
|
}
|
|
|
|
|
2017-04-16 12:06:17 +00:00
|
|
|
/// <summary>
|
|
|
|
/// An exception that is thrown when mandatory configuration to run a specific method or module hasn't been done yet.
|
|
|
|
/// </summary>
|
|
|
|
public class EngineModuleDisabledException : Exception
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// Initializes a new instance of the <see cref="EngineModuleDisabledException"/>.
|
|
|
|
/// </summary>
|
|
|
|
public EngineModuleDisabledException() : base("This engine module has not yet been enabled.")
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2017-01-08 14:57:10 +00:00
|
|
|
}
|