2014-04-17 19:11:39 +08:00
|
|
|
|
using NTwain.Properties;
|
|
|
|
|
using NTwain.Triplets;
|
2014-04-15 07:30:25 +08:00
|
|
|
|
using System;
|
2014-04-16 06:57:03 +08:00
|
|
|
|
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-17 08:39:30 +08:00
|
|
|
|
|
2014-04-19 21:10:15 +08:00
|
|
|
|
public void EnsureStarted(WindowsHook.WndProcHook hook)
|
2014-04-15 07:30:25 +08:00
|
|
|
|
{
|
|
|
|
|
if (!_started)
|
|
|
|
|
{
|
2014-04-16 06:57:03 +08:00
|
|
|
|
// 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(() =>
|
|
|
|
|
{
|
2014-04-16 06:57:03 +08:00
|
|
|
|
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();
|
2014-04-17 08:39:30 +08:00
|
|
|
|
_started = false;
|
2014-04-15 07:30:25 +08:00
|
|
|
|
}));
|
|
|
|
|
loopThread.IsBackground = true;
|
|
|
|
|
loopThread.SetApartmentState(ApartmentState.STA);
|
|
|
|
|
loopThread.Start();
|
2014-04-16 06:57:03 +08:00
|
|
|
|
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)
|
|
|
|
|
{
|
2014-04-17 19:11:39 +08:00
|
|
|
|
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)
|
|
|
|
|
{
|
2014-04-17 19:11:39 +08:00
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|