mirror of
https://github.com/seriocomedy/ShiftOS-C-.git
synced 2025-01-23 17:32:15 +00:00
d40fed5ce2
This'll be a lot easier to work on.
1123 lines
30 KiB
C#
1123 lines
30 KiB
C#
// 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.
|
|
//
|
|
// Copyright (c) 2004 - 2006 Novell, Inc.
|
|
//
|
|
// Authors:
|
|
// Peter Bartok pbartok@novell.com
|
|
// Daniel Nauck (dna(at)mono-project(dot)de)
|
|
//
|
|
|
|
// COMPLETE
|
|
|
|
#undef DebugRunLoop
|
|
|
|
using Microsoft.Win32;
|
|
using System;
|
|
using System.Drawing;
|
|
using System.ComponentModel;
|
|
using System.Collections;
|
|
using System.Diagnostics;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Reflection;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
using System.Text;
|
|
using ShiftUI.VisualStyles;
|
|
using ShiftUI.ShiftOS;
|
|
|
|
namespace ShiftUI
|
|
{
|
|
public sealed class Application
|
|
{
|
|
internal class MWFThread
|
|
{
|
|
#region Fields
|
|
|
|
private ApplicationContext context;
|
|
private bool messageloop_started;
|
|
private bool handling_exception;
|
|
private int thread_id;
|
|
|
|
private static readonly Hashtable threads = new Hashtable();
|
|
|
|
#endregion // Fields
|
|
|
|
#region Constructors
|
|
|
|
private MWFThread()
|
|
{
|
|
}
|
|
|
|
#endregion // Constructors
|
|
|
|
#region Properties
|
|
|
|
public ApplicationContext Context {
|
|
get { return context; }
|
|
set { context = value; }
|
|
}
|
|
|
|
public bool MessageLoop {
|
|
get { return messageloop_started; }
|
|
set { messageloop_started = value; }
|
|
}
|
|
|
|
public bool HandlingException {
|
|
get { return handling_exception; }
|
|
set { handling_exception = value; }
|
|
}
|
|
|
|
public static int LoopCount {
|
|
get {
|
|
lock (threads) {
|
|
int loops = 0;
|
|
|
|
foreach (MWFThread thread in threads.Values) {
|
|
if (thread.messageloop_started)
|
|
loops++;
|
|
}
|
|
|
|
return loops;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static MWFThread Current {
|
|
get {
|
|
MWFThread thread = null;
|
|
|
|
lock (threads) {
|
|
thread = (MWFThread) threads [Thread.CurrentThread.GetHashCode ()];
|
|
if (thread == null) {
|
|
thread = new MWFThread();
|
|
thread.thread_id = Thread.CurrentThread.GetHashCode ();
|
|
threads [thread.thread_id] = thread;
|
|
}
|
|
}
|
|
|
|
return thread;
|
|
}
|
|
}
|
|
|
|
#endregion // Properties
|
|
|
|
#region Methods
|
|
|
|
public void Exit ()
|
|
{
|
|
if (context != null)
|
|
context.ExitThread();
|
|
context = null;
|
|
|
|
if (Application.ThreadExit != null)
|
|
Application.ThreadExit(null, EventArgs.Empty);
|
|
|
|
if (LoopCount == 0) {
|
|
if (Application.ApplicationExit != null)
|
|
Application.ApplicationExit (null, EventArgs.Empty);
|
|
}
|
|
|
|
((MWFThread) threads [thread_id]).MessageLoop = false;
|
|
}
|
|
|
|
#endregion // Methods
|
|
}
|
|
|
|
public static void LoadSkin(Skin skin)
|
|
{
|
|
if (skin == null)
|
|
throw new Exception("Skin can't be null.");
|
|
_loadedSkin = skin;
|
|
}
|
|
|
|
private static ShiftOS.Skin _loadedSkin = null;
|
|
private static bool browser_embedded;
|
|
private static InputLanguage input_language = InputLanguage.CurrentInputLanguage;
|
|
private static string safe_caption_format = "{1} - {0} - {2}";
|
|
private static readonly ArrayList message_filters = new ArrayList();
|
|
private static readonly FormCollection forms = new FormCollection ();
|
|
|
|
private static bool use_wait_cursor;
|
|
private static ToolStrip keyboard_capture;
|
|
private static VisualStyleState visual_style_state = VisualStyleState.ClientAndNonClientAreasEnabled;
|
|
static bool visual_styles_enabled;
|
|
|
|
private Application ()
|
|
{
|
|
browser_embedded = false;
|
|
}
|
|
|
|
static Application ()
|
|
{
|
|
// Attempt to load UIA support for winforms
|
|
// UIA support requires .NET 2.0
|
|
InitializeUIAutomation ();
|
|
}
|
|
|
|
#region Private Methods
|
|
|
|
private static void InitializeUIAutomation ()
|
|
{
|
|
// Initialize the UIAutomationWinforms Global class,
|
|
// which create some listeners which subscribe to internal
|
|
// MWF events so that it can provide a11y support for MWF
|
|
const string UIA_WINFORMS_ASSEMBLY =
|
|
"UIAutomationWinforms, Version=1.0.0.0, Culture=neutral, PublicKeyToken=f4ceacb585d99812";
|
|
MethodInfo init_method;
|
|
Assembly mwf_providers = null;
|
|
try {
|
|
mwf_providers = Assembly.Load (UIA_WINFORMS_ASSEMBLY);
|
|
} catch { }
|
|
|
|
if (mwf_providers == null)
|
|
return;
|
|
|
|
const string UIA_WINFORMS_TYPE = "Mono.UIAutomation.Winforms.Global";
|
|
const string UIA_WINFORMS_METHOD = "Initialize";
|
|
try {
|
|
Type global_type = mwf_providers.GetType (UIA_WINFORMS_TYPE, false);
|
|
if (global_type != null) {
|
|
init_method = global_type.GetMethod (UIA_WINFORMS_METHOD,
|
|
BindingFlags.Static |
|
|
BindingFlags.Public);
|
|
if (init_method != null)
|
|
init_method.Invoke (null, new object [] {});
|
|
else
|
|
throw new Exception (String.Format ("Method {0} not found in type {1}.",
|
|
UIA_WINFORMS_METHOD, UIA_WINFORMS_TYPE));
|
|
}
|
|
else
|
|
throw new Exception (String.Format ("Type {0} not found in assembly {1}.",
|
|
UIA_WINFORMS_TYPE, UIA_WINFORMS_ASSEMBLY));
|
|
} catch (Exception ex) {
|
|
Console.Error.WriteLine ("Error setting up UIA: " + ex);
|
|
}
|
|
}
|
|
|
|
internal static void CloseForms (Thread thread)
|
|
{
|
|
#if DebugRunLoop
|
|
Console.WriteLine(" CloseForms({0}) called", thread);
|
|
#endif
|
|
|
|
ArrayList forms_to_close = new ArrayList ();
|
|
|
|
lock (forms) {
|
|
foreach (Form f in forms) {
|
|
if (thread == null || thread == f.creator_thread)
|
|
forms_to_close.Add (f);
|
|
}
|
|
|
|
foreach (Form f in forms_to_close) {
|
|
#if DebugRunLoop
|
|
Console.WriteLine(" Closing form {0}", f);
|
|
#endif
|
|
f.Dispose ();
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion // Private methods
|
|
|
|
#region Public Static Properties
|
|
|
|
public static bool AllowQuit {
|
|
get {
|
|
return !browser_embedded;
|
|
}
|
|
}
|
|
|
|
public static ShiftOS.Skin CurrentSkin
|
|
{
|
|
get
|
|
{
|
|
return _loadedSkin;
|
|
}
|
|
}
|
|
|
|
public static string CommonAppDataPath {
|
|
get {
|
|
return CreateDataPath (Environment.GetFolderPath (Environment.SpecialFolder.CommonApplicationData));
|
|
}
|
|
}
|
|
|
|
public static RegistryKey CommonAppDataRegistry {
|
|
get {
|
|
string key = string.Format ("Software\\{0}\\{1}\\{2}", CompanyName, ProductName, ProductVersion);
|
|
|
|
return Registry.LocalMachine.CreateSubKey (key);
|
|
}
|
|
}
|
|
|
|
public static string CompanyName {
|
|
get {
|
|
string company = string.Empty;
|
|
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
AssemblyCompanyAttribute[] attrs = (AssemblyCompanyAttribute[])
|
|
assembly.GetCustomAttributes (typeof(AssemblyCompanyAttribute), true);
|
|
if (attrs != null && attrs.Length > 0)
|
|
company = attrs [0].Company;
|
|
|
|
// If there is no [AssemblyCompany], return the outermost namespace
|
|
// on Main ()
|
|
if (company == null || company.Length == 0)
|
|
if (assembly.EntryPoint != null) {
|
|
company = assembly.EntryPoint.DeclaringType.Namespace;
|
|
|
|
if (company != null) {
|
|
int firstDot = company.IndexOf ('.');
|
|
if (firstDot >= 0)
|
|
company = company.Substring (0, firstDot);
|
|
}
|
|
}
|
|
|
|
// If that doesn't work, return the name of class containing Main ()
|
|
if (company == null || company.Length == 0)
|
|
if (assembly.EntryPoint != null)
|
|
company = assembly.EntryPoint.DeclaringType.FullName;
|
|
|
|
return company;
|
|
}
|
|
}
|
|
|
|
public static CultureInfo CurrentCulture {
|
|
get {
|
|
return Thread.CurrentThread.CurrentUICulture;
|
|
}
|
|
set {
|
|
Thread.CurrentThread.CurrentUICulture = value;
|
|
}
|
|
}
|
|
|
|
public static InputLanguage CurrentInputLanguage {
|
|
get {
|
|
return input_language;
|
|
}
|
|
set {
|
|
input_language=value;
|
|
}
|
|
}
|
|
|
|
public static string ExecutablePath {
|
|
get {
|
|
return Path.GetFullPath (Environment.GetCommandLineArgs ()[0]);
|
|
}
|
|
}
|
|
|
|
public static string LocalUserAppDataPath {
|
|
get {
|
|
return CreateDataPath (Environment.GetFolderPath (Environment.SpecialFolder.LocalApplicationData));
|
|
}
|
|
}
|
|
|
|
public static bool MessageLoop {
|
|
get {
|
|
return MWFThread.Current.MessageLoop;
|
|
}
|
|
}
|
|
|
|
public static string ProductName {
|
|
get {
|
|
string name = string.Empty;
|
|
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
AssemblyProductAttribute[] attrs = (AssemblyProductAttribute[])
|
|
assembly.GetCustomAttributes (typeof(AssemblyProductAttribute), true);
|
|
|
|
if (attrs != null && attrs.Length > 0)
|
|
name = attrs [0].Product;
|
|
|
|
// If there is no [AssemblyProduct], .NET returns the name of
|
|
// the innermost namespace and if that fails, resorts to the
|
|
// name of the class containing Main ()
|
|
if (name == null || name.Length == 0)
|
|
if (assembly.EntryPoint != null) {
|
|
name = assembly.EntryPoint.DeclaringType.Namespace;
|
|
|
|
if (name != null) {
|
|
int lastDot = name.LastIndexOf ('.');
|
|
if (lastDot >= 0 && lastDot < name.Length - 1)
|
|
name = name.Substring (lastDot + 1);
|
|
}
|
|
|
|
if (name == null || name.Length == 0)
|
|
name = assembly.EntryPoint.DeclaringType.FullName;
|
|
}
|
|
|
|
return name;
|
|
}
|
|
}
|
|
|
|
public static string ProductVersion {
|
|
get {
|
|
String version = string.Empty;
|
|
|
|
Assembly assembly = Assembly.GetEntryAssembly ();
|
|
|
|
if (assembly == null)
|
|
assembly = Assembly.GetCallingAssembly ();
|
|
|
|
AssemblyInformationalVersionAttribute infoVersion =
|
|
Attribute.GetCustomAttribute (assembly,
|
|
typeof (AssemblyInformationalVersionAttribute))
|
|
as AssemblyInformationalVersionAttribute;
|
|
|
|
if (infoVersion != null)
|
|
version = infoVersion.InformationalVersion;
|
|
|
|
// If [AssemblyFileVersion] is present it is used
|
|
// before resorting to assembly version
|
|
if (version == null || version.Length == 0) {
|
|
AssemblyFileVersionAttribute fileVersion =
|
|
Attribute.GetCustomAttribute (assembly,
|
|
typeof (AssemblyFileVersionAttribute))
|
|
as AssemblyFileVersionAttribute;
|
|
if (fileVersion != null)
|
|
version = fileVersion.Version;
|
|
}
|
|
|
|
// If neither [AssemblyInformationalVersionAttribute]
|
|
// nor [AssemblyFileVersion] are present, then use
|
|
// the assembly version
|
|
if (version == null || version.Length == 0)
|
|
version = assembly.GetName ().Version.ToString ();
|
|
|
|
return version;
|
|
}
|
|
}
|
|
|
|
public static string SafeTopLevelCaptionFormat {
|
|
get {
|
|
return safe_caption_format;
|
|
}
|
|
set {
|
|
safe_caption_format = value;
|
|
}
|
|
}
|
|
|
|
public static string StartupPath {
|
|
get {
|
|
return Path.GetDirectoryName (Application.ExecutablePath);
|
|
}
|
|
}
|
|
|
|
public static string UserAppDataPath {
|
|
get {
|
|
return CreateDataPath (Environment.GetFolderPath (Environment.SpecialFolder.ApplicationData));
|
|
}
|
|
}
|
|
|
|
public static RegistryKey UserAppDataRegistry {
|
|
get {
|
|
string key = string.Format ("Software\\{0}\\{1}\\{2}", CompanyName, ProductName, ProductVersion);
|
|
|
|
return Registry.CurrentUser.CreateSubKey (key);
|
|
}
|
|
}
|
|
|
|
public static bool UseWaitCursor {
|
|
get {
|
|
return use_wait_cursor;
|
|
}
|
|
set {
|
|
use_wait_cursor = value;
|
|
if (use_wait_cursor) {
|
|
foreach (Form form in OpenForms) {
|
|
form.Cursor = Cursors.WaitCursor;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public static bool RenderWithVisualStyles {
|
|
get {
|
|
if (VisualStyleInformation.IsSupportedByOS) {
|
|
if (!VisualStyleInformation.IsEnabledByUser)
|
|
return false;
|
|
if (!XplatUI.ThemesEnabled)
|
|
return false;
|
|
if (Application.VisualStyleState == VisualStyleState.ClientAndNonClientAreasEnabled)
|
|
return true;
|
|
if (Application.VisualStyleState == VisualStyleState.ClientAreaEnabled)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public static VisualStyleState VisualStyleState {
|
|
get { return Application.visual_style_state; }
|
|
set { Application.visual_style_state = value; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Public Static Methods
|
|
|
|
public static void AddMessageFilter (IMessageFilter value)
|
|
{
|
|
lock (message_filters) {
|
|
message_filters.Add (value);
|
|
}
|
|
}
|
|
|
|
internal static void AddKeyFilter (IKeyFilter value)
|
|
{
|
|
XplatUI.AddKeyFilter (value);
|
|
}
|
|
|
|
public static void DoEvents ()
|
|
{
|
|
XplatUI.DoEvents ();
|
|
}
|
|
|
|
public static void EnableVisualStyles ()
|
|
{
|
|
try {
|
|
visual_styles_enabled = true;
|
|
XplatUI.EnableThemes ();
|
|
}
|
|
catch {
|
|
visual_styles_enabled = false;
|
|
}
|
|
}
|
|
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static bool FilterMessage (ref Message message)
|
|
{
|
|
lock (message_filters) {
|
|
for (int i = 0; i < message_filters.Count; i++) {
|
|
IMessageFilter filter = (IMessageFilter) message_filters[i];
|
|
if (filter.PreFilterMessage (ref message))
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//
|
|
// If true, it uses GDI+, performance reasons were quoted
|
|
//
|
|
static internal bool use_compatible_text_rendering = true;
|
|
|
|
public static void SetCompatibleTextRenderingDefault (bool defaultValue)
|
|
{
|
|
use_compatible_text_rendering = defaultValue;
|
|
}
|
|
|
|
public static FormCollection OpenForms {
|
|
get {
|
|
return forms;
|
|
}
|
|
}
|
|
|
|
[MonoNotSupported ("Only applies when Winforms is being hosted by an unmanaged app.")]
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static void RegisterMessageLoop (MessageLoopCallback callback)
|
|
{
|
|
}
|
|
|
|
[MonoNotSupported ("Empty stub.")]
|
|
public static bool SetSuspendState (PowerState state, bool force, bool disableWakeEvent)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
[MonoNotSupported ("Empty stub.")]
|
|
public static void SetUnhandledExceptionMode (UnhandledExceptionMode mode)
|
|
{
|
|
//FIXME: a stub to fill
|
|
}
|
|
|
|
[MonoNotSupported ("Empty stub.")]
|
|
public static void SetUnhandledExceptionMode (UnhandledExceptionMode mode, bool threadScope)
|
|
{
|
|
//FIXME: a stub to fill
|
|
}
|
|
|
|
[MonoNotSupported ("Only applies when Winforms is being hosted by an unmanaged app.")]
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static void UnregisterMessageLoop ()
|
|
{
|
|
}
|
|
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static void RaiseIdle (EventArgs e)
|
|
{
|
|
XplatUI.RaiseIdle (e);
|
|
}
|
|
|
|
public static void Restart ()
|
|
{
|
|
//FIXME: ClickOnce stuff using the Update or UpdateAsync methods.
|
|
//FIXME: SecurityPermission: Restart () requires IsUnrestricted permission.
|
|
|
|
if (Assembly.GetEntryAssembly () == null)
|
|
throw new NotSupportedException ("The method 'Restart' is not supported by this application type.");
|
|
|
|
string mono_path = null;
|
|
|
|
//Get mono path
|
|
PropertyInfo gac = typeof (Environment).GetProperty ("GacPath", BindingFlags.Static | BindingFlags.NonPublic);
|
|
MethodInfo get_gac = null;
|
|
if (gac != null)
|
|
get_gac = gac.GetGetMethod (true);
|
|
|
|
if (get_gac != null) {
|
|
string gac_path = Path.GetDirectoryName ((string)get_gac.Invoke (null, null));
|
|
string mono_prefix = Path.GetDirectoryName (Path.GetDirectoryName (gac_path));
|
|
|
|
if (XplatUI.RunningOnUnix) {
|
|
mono_path = Path.Combine (mono_prefix, "bin/mono");
|
|
if (!File.Exists (mono_path))
|
|
mono_path = "mono";
|
|
} else {
|
|
mono_path = Path.Combine (mono_prefix, "bin\\mono.bat");
|
|
|
|
if (!File.Exists (mono_path))
|
|
mono_path = Path.Combine (mono_prefix, "bin\\mono.exe");
|
|
|
|
if (!File.Exists (mono_path))
|
|
mono_path = Path.Combine (mono_prefix, "mono\\mono\\mini\\mono.exe");
|
|
|
|
if (!File.Exists (mono_path))
|
|
throw new FileNotFoundException (string.Format ("Windows mono path not found: '{0}'", mono_path));
|
|
}
|
|
}
|
|
|
|
//Get command line arguments
|
|
StringBuilder argsBuilder = new StringBuilder ();
|
|
string[] args = Environment.GetCommandLineArgs ();
|
|
for (int i = 0; i < args.Length; i++)
|
|
{
|
|
argsBuilder.Append (string.Format ("\"{0}\" ", args[i]));
|
|
}
|
|
string arguments = argsBuilder.ToString ();
|
|
ProcessStartInfo procInfo = Process.GetCurrentProcess ().StartInfo;
|
|
|
|
if (mono_path == null) { //it is .NET on Windows
|
|
procInfo.FileName = args[0];
|
|
procInfo.Arguments = arguments.Remove (0, args[0].Length + 3); //1 space and 2 quotes
|
|
}
|
|
else {
|
|
procInfo.Arguments = arguments;
|
|
procInfo.FileName = mono_path;
|
|
}
|
|
|
|
procInfo.WorkingDirectory = Environment.CurrentDirectory;
|
|
|
|
Application.Exit ();
|
|
Process.Start (procInfo);
|
|
}
|
|
|
|
public static void Exit ()
|
|
{
|
|
Exit (new CancelEventArgs ());
|
|
}
|
|
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static void Exit (CancelEventArgs e)
|
|
{
|
|
ArrayList forms_to_close;
|
|
|
|
lock (forms) {
|
|
forms_to_close = new ArrayList (forms);
|
|
|
|
foreach (Form f in forms_to_close) {
|
|
// Give each form a chance to cancel the Application.Exit
|
|
e.Cancel = f.FireClosingEvents (CloseReason.ApplicationExitCall, false);
|
|
|
|
if (e.Cancel)
|
|
return;
|
|
|
|
f.suppress_closing_events = true;
|
|
f.Close ();
|
|
f.Dispose ();
|
|
}
|
|
}
|
|
|
|
XplatUI.PostQuitMessage (0);
|
|
}
|
|
|
|
public static void ExitThread()
|
|
{
|
|
CloseForms(Thread.CurrentThread);
|
|
// this might not be right - need to investigate (somehow) if a WM_QUIT message is generated here
|
|
XplatUI.PostQuitMessage(0);
|
|
}
|
|
|
|
public static ApartmentState OleRequired ()
|
|
{
|
|
//throw new NotImplementedException("OLE Not supported by this ShiftUI implementation");
|
|
return ApartmentState.Unknown;
|
|
}
|
|
|
|
public static void OnThreadException (Exception t)
|
|
{
|
|
if (MWFThread.Current.HandlingException) {
|
|
/* we're already handling an exception and we got
|
|
another one? print it out and exit, this means
|
|
we've got a runtime/SWF bug. */
|
|
Console.WriteLine (t);
|
|
// Don't use Application.Exit here, since it may cause a stack overflow
|
|
// in certain cases. It's however hard to reproduce since it seems to
|
|
// be depending on when the GC kicks in.
|
|
Environment.Exit(1);
|
|
}
|
|
|
|
try {
|
|
MWFThread.Current.HandlingException = true;
|
|
|
|
if (Application.ThreadException != null) {
|
|
Application.ThreadException(null, new ThreadExceptionEventArgs(t));
|
|
return;
|
|
}
|
|
|
|
if (SystemInformation.UserInteractive) {
|
|
Form form = new ThreadExceptionDialog (t);
|
|
form.ShowDialog ();
|
|
} else {
|
|
Console.WriteLine (t.ToString ());
|
|
Application.Exit ();
|
|
}
|
|
} finally {
|
|
MWFThread.Current.HandlingException = false;
|
|
}
|
|
}
|
|
|
|
public static void RemoveMessageFilter (IMessageFilter value)
|
|
{
|
|
lock (message_filters) {
|
|
message_filters.Remove (value);
|
|
}
|
|
}
|
|
|
|
public static void Run ()
|
|
{
|
|
Run (new ApplicationContext ());
|
|
}
|
|
|
|
public static void Run (Form mainForm)
|
|
{
|
|
Run (new ApplicationContext (mainForm));
|
|
}
|
|
|
|
internal static void FirePreRun ()
|
|
{
|
|
EventHandler handler = PreRun;
|
|
if (handler != null)
|
|
handler (null, EventArgs.Empty);
|
|
}
|
|
|
|
public static void Run (ApplicationContext context)
|
|
{
|
|
// If a sync context hasn't been created by now, create
|
|
// a default one
|
|
if (SynchronizationContext.Current == null)
|
|
SynchronizationContext.SetSynchronizationContext (new SynchronizationContext ());
|
|
|
|
RunLoop (false, context);
|
|
|
|
// Reset the sync context back to the default
|
|
if (SynchronizationContext.Current is WindowsFormsSynchronizationContext)
|
|
WindowsFormsSynchronizationContext.Uninstall ();
|
|
}
|
|
|
|
private static void DisableFormsForModalLoop (Queue toplevels, ApplicationContext context)
|
|
{
|
|
Form f;
|
|
|
|
lock (forms) {
|
|
IEnumerator control = forms.GetEnumerator ();
|
|
|
|
while (control.MoveNext ()) {
|
|
f = (Form)control.Current;
|
|
|
|
// Don't disable the main form.
|
|
if (f == context.MainForm) {
|
|
continue;
|
|
}
|
|
|
|
// Don't disable any children of the main form.
|
|
// These do not have to be MDI children.
|
|
Widget current = f;
|
|
bool is_child_of_main = false; ;
|
|
|
|
do {
|
|
if (current.Parent == context.MainForm) {
|
|
is_child_of_main = true;
|
|
break;
|
|
}
|
|
current = current.Parent;
|
|
} while (current != null);
|
|
|
|
if (is_child_of_main)
|
|
continue;
|
|
|
|
// Disable the rest
|
|
if (f.IsHandleCreated && XplatUI.IsEnabled (f.Handle)) {
|
|
#if DebugRunLoop
|
|
Console.WriteLine(" Disabling form {0}", f);
|
|
#endif
|
|
XplatUI.EnableWindow (f.Handle, false);
|
|
toplevels.Enqueue (f);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
private static void EnableFormsForModalLoop (Queue toplevels, ApplicationContext context)
|
|
{
|
|
while (toplevels.Count > 0) {
|
|
#if DebugRunLoop
|
|
Console.WriteLine(" Re-Enabling form form {0}", toplevels.Peek());
|
|
#endif
|
|
Form c = (Form) toplevels.Dequeue ();
|
|
if (c.IsHandleCreated) {
|
|
XplatUI.EnableWindow (c.window.Handle, true);
|
|
context.MainForm = c;
|
|
}
|
|
}
|
|
#if DebugRunLoop
|
|
Console.WriteLine(" Done with the re-enable");
|
|
#endif
|
|
}
|
|
|
|
internal static void RunLoop (bool Modal, ApplicationContext context)
|
|
{
|
|
Queue toplevels;
|
|
MSG msg;
|
|
Object queue_id;
|
|
MWFThread thread;
|
|
ApplicationContext previous_thread_context;
|
|
|
|
thread = MWFThread.Current;
|
|
|
|
/*
|
|
* There is a NotWorking test for this, but since we are using this method both for Form.ShowDialog as for ApplicationContexts we'll
|
|
* fail on nested ShowDialogs, so disable the check for the moment.
|
|
*/
|
|
//if (thread.MessageLoop) {
|
|
// throw new InvalidOperationException ("Starting a second message loop on a single thread is not a valid operation. Use Form.ShowDialog instead.");
|
|
//}
|
|
|
|
msg = new MSG();
|
|
|
|
if (context == null)
|
|
context = new ApplicationContext();
|
|
|
|
previous_thread_context = thread.Context;
|
|
thread.Context = context;
|
|
|
|
if (context.MainForm != null) {
|
|
context.MainForm.is_modal = Modal;
|
|
context.MainForm.context = context;
|
|
context.MainForm.closing = false;
|
|
context.MainForm.Visible = true; // Cannot use Show() or scaling gets confused by menus
|
|
// XXX the above line can be used to close the form. another problem with our handling of Show/Activate.
|
|
if (context.MainForm != null)
|
|
context.MainForm.Activate();
|
|
}
|
|
|
|
#if DebugRunLoop
|
|
Console.WriteLine("Entering RunLoop(Modal={0}, Form={1})", Modal, context.MainForm != null ? context.MainForm.ToString() : "NULL");
|
|
#endif
|
|
|
|
if (Modal) {
|
|
toplevels = new Queue ();
|
|
DisableFormsForModalLoop (toplevels, context);
|
|
|
|
// FIXME - need activate?
|
|
/* make sure the MainForm is enabled */
|
|
if (context.MainForm != null) {
|
|
XplatUI.EnableWindow (context.MainForm.Handle, true);
|
|
XplatUI.SetModal(context.MainForm.Handle, true);
|
|
}
|
|
} else {
|
|
toplevels = null;
|
|
}
|
|
|
|
queue_id = XplatUI.StartLoop(Thread.CurrentThread);
|
|
thread.MessageLoop = true;
|
|
|
|
bool quit = false;
|
|
|
|
while (!quit && XplatUI.GetMessage(queue_id, ref msg, IntPtr.Zero, 0, 0)) {
|
|
Message m = Message.Create(msg.hwnd, (int)msg.message, msg.wParam, msg.lParam);
|
|
|
|
if (Application.FilterMessage (ref m))
|
|
continue;
|
|
|
|
switch((Msg)msg.message) {
|
|
case Msg.WM_KEYDOWN:
|
|
case Msg.WM_SYSKEYDOWN:
|
|
case Msg.WM_CHAR:
|
|
case Msg.WM_SYSCHAR:
|
|
case Msg.WM_KEYUP:
|
|
case Msg.WM_SYSKEYUP:
|
|
Widget c;
|
|
c = Widget.FromHandle(msg.hwnd);
|
|
|
|
// If we have a control with keyboard capture (usually a *Strip)
|
|
// give it the message, and then drop the message
|
|
if (keyboard_capture != null) {
|
|
// WM_SYSKEYUP does not make it into ProcessCmdKey, so do it here
|
|
if ((Msg)m.Msg == Msg.WM_SYSKEYDOWN)
|
|
if (m.WParam.ToInt32() == (int)Keys.Menu) {
|
|
keyboard_capture.GetTopLevelToolStrip ().Dismiss (ToolStripDropDownCloseReason.Keyboard);
|
|
continue;
|
|
}
|
|
|
|
m.HWnd = keyboard_capture.Handle;
|
|
|
|
switch (keyboard_capture.PreProcessWidgetMessageInternal (ref m)) {
|
|
case PreProcessWidgetstate.MessageProcessed:
|
|
continue;
|
|
case PreProcessWidgetstate.MessageNeeded:
|
|
case PreProcessWidgetstate.MessageNotNeeded:
|
|
if (((m.Msg == (int)Msg.WM_KEYDOWN || m.Msg == (int)Msg.WM_CHAR) && !keyboard_capture.ProcessWidgetMnemonic ((char)m.WParam))) {
|
|
if (c == null || !ControlOnToolStrip (c))
|
|
continue;
|
|
else
|
|
m.HWnd = msg.hwnd;
|
|
} else
|
|
continue;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (((c != null) && c.PreProcessWidgetMessageInternal (ref m) != PreProcessWidgetstate.MessageProcessed) ||
|
|
(c == null)) {
|
|
goto default;
|
|
}
|
|
break;
|
|
|
|
case Msg.WM_LBUTTONDOWN:
|
|
case Msg.WM_MBUTTONDOWN:
|
|
case Msg.WM_RBUTTONDOWN:
|
|
if (keyboard_capture != null) {
|
|
Widget c2 = Widget.FromHandle (msg.hwnd);
|
|
|
|
// the target is not a winforms control (an embedded control, perhaps), so
|
|
// release everything
|
|
if (c2 == null) {
|
|
ToolStripManager.FireAppClicked ();
|
|
goto default;
|
|
}
|
|
|
|
// If we clicked a ToolStrip, we have to make sure it isn't
|
|
// the one we are on, or any of its parents or children
|
|
// If we clicked off the dropped down menu, release everything
|
|
if (c2 is ToolStrip) {
|
|
if ((c2 as ToolStrip).GetTopLevelToolStrip () != keyboard_capture.GetTopLevelToolStrip ())
|
|
ToolStripManager.FireAppClicked ();
|
|
} else {
|
|
if (c2.Parent != null)
|
|
if (c2.Parent is ToolStripDropDownMenu)
|
|
if ((c2.Parent as ToolStripDropDownMenu).GetTopLevelToolStrip () == keyboard_capture.GetTopLevelToolStrip ())
|
|
goto default;
|
|
if (c2.TopLevelWidget == null)
|
|
goto default;
|
|
|
|
ToolStripManager.FireAppClicked ();
|
|
}
|
|
}
|
|
|
|
goto default;
|
|
|
|
case Msg.WM_QUIT:
|
|
quit = true; // make sure we exit
|
|
break;
|
|
default:
|
|
XplatUI.TranslateMessage (ref msg);
|
|
XplatUI.DispatchMessage (ref msg);
|
|
break;
|
|
}
|
|
|
|
// If our Form doesn't have a handle anymore, it means it was destroyed and we need to *wait* for WM_QUIT.
|
|
if ((context.MainForm != null) && (!context.MainForm.IsHandleCreated))
|
|
continue;
|
|
|
|
// Handle exit, Form might have received WM_CLOSE and set 'closing' in response.
|
|
if ((context.MainForm != null) && (context.MainForm.closing || (Modal && !context.MainForm.Visible))) {
|
|
if (!Modal) {
|
|
XplatUI.PostQuitMessage (0);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#if DebugRunLoop
|
|
Console.WriteLine (" RunLoop loop left");
|
|
#endif
|
|
|
|
thread.MessageLoop = false;
|
|
XplatUI.EndLoop (Thread.CurrentThread);
|
|
|
|
if (Modal) {
|
|
Form old = context.MainForm;
|
|
|
|
context.MainForm = null;
|
|
|
|
EnableFormsForModalLoop (toplevels, context);
|
|
|
|
if (context.MainForm != null && context.MainForm.IsHandleCreated) {
|
|
XplatUI.SetModal (context.MainForm.Handle, false);
|
|
}
|
|
#if DebugRunLoop
|
|
Console.WriteLine (" Done with the SetModal");
|
|
#endif
|
|
old.RaiseCloseEvents (true, false);
|
|
old.is_modal = false;
|
|
}
|
|
|
|
#if DebugRunLoop
|
|
Console.WriteLine ("Leaving RunLoop(Modal={0}, Form={1})", Modal, context.MainForm != null ? context.MainForm.ToString() : "NULL");
|
|
#endif
|
|
|
|
if (context.MainForm != null) {
|
|
context.MainForm.context = null;
|
|
context.MainForm = null;
|
|
}
|
|
|
|
thread.Context = previous_thread_context;
|
|
|
|
if (!Modal)
|
|
thread.Exit();
|
|
}
|
|
|
|
#endregion // Public Static Methods
|
|
|
|
#region Events
|
|
|
|
public static event EventHandler ApplicationExit;
|
|
|
|
public static event EventHandler Idle {
|
|
add {
|
|
XplatUI.Idle += value;
|
|
}
|
|
remove {
|
|
XplatUI.Idle -= value;
|
|
}
|
|
}
|
|
|
|
public static event EventHandler ThreadExit;
|
|
public static event ThreadExceptionEventHandler ThreadException;
|
|
|
|
// These are used externally by the UIA framework
|
|
internal static event EventHandler FormAdded;
|
|
internal static event EventHandler PreRun;
|
|
|
|
#pragma warning disable 0067
|
|
[MonoTODO]
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static event EventHandler EnterThreadModal;
|
|
|
|
[MonoTODO]
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public static event EventHandler LeaveThreadModal;
|
|
#pragma warning restore 0067
|
|
|
|
#endregion // Events
|
|
|
|
#region Public Delegates
|
|
|
|
//[EditorBrowsable (EditorBrowsableState.Advanced)]
|
|
public delegate bool MessageLoopCallback ();
|
|
|
|
#endregion
|
|
|
|
#region Internal Properties
|
|
internal static ToolStrip KeyboardCapture {
|
|
get { return keyboard_capture; }
|
|
set { keyboard_capture = value; }
|
|
}
|
|
|
|
internal static bool VisualStylesEnabled {
|
|
get { return visual_styles_enabled; }
|
|
}
|
|
#endregion
|
|
|
|
#region Internal Methods
|
|
|
|
internal static void AddForm (Form f)
|
|
{
|
|
lock (forms)
|
|
forms.Add (f);
|
|
// Signal that a Form has been added to this
|
|
// Application. Used by UIA to detect new Forms that
|
|
// need a11y support. This event may be fired even if
|
|
// the form has already been added, so clients should
|
|
// account for that when handling this signal.
|
|
if (FormAdded != null)
|
|
FormAdded (f, null);
|
|
}
|
|
|
|
internal static void RemoveForm (Form f)
|
|
{
|
|
lock (forms)
|
|
forms.Remove (f);
|
|
}
|
|
|
|
private static bool ControlOnToolStrip (Widget c)
|
|
{
|
|
Widget p = c.Parent;
|
|
|
|
while (p != null) {
|
|
if (p is ToolStrip)
|
|
return true;
|
|
|
|
p = p.Parent;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Takes a starting path, appends company name, product name, and
|
|
// product version. If the directory doesn't exist, create it
|
|
private static string CreateDataPath (string basePath)
|
|
{
|
|
string path;
|
|
|
|
path = Path.Combine (basePath, CompanyName);
|
|
path = Path.Combine (path, ProductName);
|
|
path = Path.Combine (path, ProductVersion);
|
|
|
|
if (!Directory.Exists (path))
|
|
Directory.CreateDirectory (path);
|
|
|
|
return path;
|
|
}
|
|
#endregion
|
|
}
|
|
}
|