ntwain/NTwain/MessageLoop.cs

159 lines
4.8 KiB
C#
Raw Normal View History

using NTwain.Properties;
using NTwain.Triplets;
2014-04-15 07:30:25 +08:00
using System;
using System.Diagnostics;
2014-04-19 21:10:15 +08:00
using System.Runtime.InteropServices;
2014-04-15 07:30:25 +08:00
using System.Threading;
using System.Windows.Interop;
using System.Windows.Threading;
namespace NTwain
{
/// <summary>
/// Provides a message loop for old TWAIN to post or new TWAIN to synchronize callbacks.
/// </summary>
class MessageLoop
{
static MessageLoop _instance = new MessageLoop();
public static MessageLoop Instance { get { return _instance; } }
Dispatcher _dispatcher;
bool _started;
2014-04-19 21:10:15 +08:00
WindowsHook _hook;
2014-04-15 07:30:25 +08:00
private MessageLoop() { }
2014-04-19 21:10:15 +08:00
public void EnsureStarted(WindowsHook.WndProcHook hook)
2014-04-15 07:30:25 +08:00
{
if (!_started)
{
// using this terrible hack so the new thread will start running before this function returns
var hack = new ManualResetEvent(false);
2014-04-15 07:30:25 +08:00
var loopThread = new Thread(new ThreadStart(() =>
{
Debug.WriteLine("NTwain message loop started.");
2014-04-15 07:30:25 +08:00
_dispatcher = Dispatcher.CurrentDispatcher;
2014-04-19 21:10:15 +08:00
if (!Dsm.IsOnMono)
2014-04-15 07:30:25 +08:00
{
2014-04-19 21:10:15 +08:00
_hook = new WindowsHook(hook);
2014-04-15 07:30:25 +08:00
}
2014-04-19 21:10:15 +08:00
hack.Set();
2014-04-15 07:30:25 +08:00
Dispatcher.Run();
_started = false;
2014-04-15 07:30:25 +08:00
}));
loopThread.IsBackground = true;
loopThread.SetApartmentState(ApartmentState.STA);
loopThread.Start();
hack.WaitOne();
hack.Close();
2014-04-15 07:30:25 +08:00
_started = true;
}
}
public IntPtr LoopHandle
{
get
{
2014-04-19 21:10:15 +08:00
return _hook == null ? IntPtr.Zero : _hook.Handle;
2014-04-15 07:30:25 +08:00
}
}
public void BeginInvoke(Action action)
{
if (_dispatcher == null) { throw new InvalidOperationException(Resources.MsgLoopUnavailble); }
2014-04-16 18:53:05 +08:00
_dispatcher.BeginInvoke(DispatcherPriority.Normal, action);
2014-04-15 07:30:25 +08:00
}
public void Invoke(Action action)
{
if (_dispatcher == null) { throw new InvalidOperationException(Resources.MsgLoopUnavailble); }
2014-04-16 18:53:05 +08:00
if (_dispatcher.CheckAccess())
2014-04-15 07:30:25 +08:00
{
2014-04-16 18:53:05 +08:00
action();
}
else
{
//_dispatcher.Invoke(DispatcherPriority.Normal, action);
var man = new ManualResetEvent(false);
_dispatcher.BeginInvoke(DispatcherPriority.Normal, new Action(() =>
2014-04-15 07:30:25 +08:00
{
2014-04-16 18:53:05 +08:00
try
2014-04-15 07:30:25 +08:00
{
action();
2014-04-16 18:53:05 +08:00
}
finally
{
2014-04-15 07:30:25 +08:00
man.Set();
2014-04-16 18:53:05 +08:00
}
}));
man.WaitOne();
man.Close();
2014-04-15 07:30:25 +08:00
}
}
2014-04-19 21:10:15 +08:00
}
/// <summary>
/// Abstracts out wnd proc hook on Windows from MessageLoop class.
/// This allows things to not depend on PresentationCore.dll at runtime on mono.
/// Not that everything works yet in mono but it's something.
/// </summary>
class WindowsHook
{
public WindowsHook(WndProcHook hook)
{
// hook into windows msg loop for old twain to post msgs.
// the style values are purely guesses here with
// CS_NOCLOSE, WS_DISABLED, and WS_EX_NOACTIVATE
var win = new HwndSource(0x0200, 0x8000000, 0x8000000, 0, 0, "NTWAIN_LOOPER", IntPtr.Zero);
Handle = win.Handle;
_hook = hook;
win.AddHook(WndProc);
}
public delegate void WndProcHook(ref MESSAGE winMsg, ref bool handled);
2014-04-15 07:30:25 +08:00
2014-04-19 21:10:15 +08:00
WndProcHook _hook;
private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
2014-04-15 07:30:25 +08:00
{
2014-04-19 21:10:15 +08:00
if (_hook != null)
{
var winmsg = new MESSAGE(hwnd, msg, wParam, lParam);
_hook(ref winmsg, ref handled);
}
return IntPtr.Zero;
2014-04-15 07:30:25 +08:00
}
2014-04-19 21:10:15 +08:00
public IntPtr Handle { get; private set; }
/// <summary>
/// The MSG structure in Windows for TWAIN use.
/// </summary>
[StructLayout(LayoutKind.Sequential)]
public struct MESSAGE
2014-04-15 07:30:25 +08:00
{
2014-04-19 21:10:15 +08:00
public MESSAGE(IntPtr hwnd, int message, IntPtr wParam, IntPtr lParam)
{
_hwnd = hwnd;
_message = (uint)message;
_wParam = wParam;
_lParam = lParam;
_time = 0;
_x = 0;
_y = 0;
}
IntPtr _hwnd;
uint _message;
IntPtr _wParam;
IntPtr _lParam;
uint _time;
int _x;
int _y;
2014-04-15 07:30:25 +08:00
}
}
}