2014-09-12 10:24:27 +08:00
using NTwain.Data ;
using NTwain.Internals ;
2015-02-19 09:02:39 +08:00
using NTwain.Interop ;
2014-09-12 10:24:27 +08:00
using NTwain.Triplets ;
using System ;
using System.Collections.Generic ;
using System.Diagnostics ;
using System.Globalization ;
using System.Linq ;
2014-09-12 19:16:10 +08:00
using System.Runtime.InteropServices ;
2014-09-12 10:24:27 +08:00
using System.Text ;
using System.Threading ;
namespace NTwain
{
// for internal pieces since the main twain session file is getting too long
2014-09-12 19:16:10 +08:00
partial class TwainSession : ITwainSessionInternal , IWinMessageFilter
2014-09-12 10:24:27 +08:00
{
#region ITwainSessionInternal Members
MessageLoopHook _msgLoopHook ;
MessageLoopHook ITwainSessionInternal . MessageLoopHook { get { return _msgLoopHook ; } }
/// <summary>
/// Gets the app id used for the session.
/// </summary>
/// <value>The app id.</value>
TWIdentity ITwainSessionInternal . AppId { get { return _appId ; } }
2016-02-23 07:55:29 +08:00
bool _closeRequested ;
bool ITwainSessionInternal . CloseDSRequested { get { return _closeRequested ; } }
2015-01-08 20:05:22 +08:00
void ITwainSessionInternal . UpdateCallback ( )
{
if ( State < 4 )
{
_callbackObj = null ;
}
else
{
ReturnCode rc = ReturnCode . Failure ;
// per the spec (8-10) apps for 2.2 or higher uses callback2 so try this first
if ( _appId . ProtocolMajor > 2 | | ( _appId . ProtocolMajor > = 2 & & _appId . ProtocolMinor > = 2 ) )
{
var cb = new TWCallback2 ( HandleCallback ) ;
rc = ( ( ITwainSessionInternal ) this ) . DGControl . Callback2 . RegisterCallback ( cb ) ;
if ( rc = = ReturnCode . Success )
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Registered callback2 OK." ) ;
2015-01-08 20:05:22 +08:00
_callbackObj = cb ;
}
}
if ( rc ! = ReturnCode . Success )
{
// always try old callback
var cb = new TWCallback ( HandleCallback ) ;
rc = ( ( ITwainSessionInternal ) this ) . DGControl . Callback . RegisterCallback ( cb ) ;
if ( rc = = ReturnCode . Success )
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Registered callback OK." ) ;
2015-01-08 20:05:22 +08:00
_callbackObj = cb ;
}
}
}
}
2014-09-12 10:24:27 +08:00
void ITwainSessionInternal . ChangeState ( int newState , bool notifyChange )
{
_state = newState ;
if ( notifyChange )
{
OnPropertyChanged ( "State" ) ;
SafeAsyncSyncableRaiseOnEvent ( OnStateChanged , StateChanged ) ;
}
}
ICommittable ITwainSessionInternal . GetPendingStateChanger ( int newState )
{
return new TentativeStateCommitable ( this , newState ) ;
}
2014-09-15 19:24:13 +08:00
void ITwainSessionInternal . ChangeCurrentSource ( DataSource source )
2014-09-12 10:24:27 +08:00
{
CurrentSource = source ;
2016-03-24 06:08:58 +08:00
DisableReason = Message . Null ;
2014-09-12 10:24:27 +08:00
OnPropertyChanged ( "CurrentSource" ) ;
SafeAsyncSyncableRaiseOnEvent ( OnSourceChanged , SourceChanged ) ;
}
void ITwainSessionInternal . SafeSyncableRaiseEvent ( DataTransferredEventArgs e )
{
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , e ) ;
}
void ITwainSessionInternal . SafeSyncableRaiseEvent ( TransferErrorEventArgs e )
{
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , e ) ;
}
void ITwainSessionInternal . SafeSyncableRaiseEvent ( TransferReadyEventArgs e )
{
SafeSyncableRaiseOnEvent ( OnTransferReady , TransferReady , e ) ;
}
DGAudio _dgAudio ;
DGAudio ITwainSessionInternal . DGAudio
{
get
{
if ( _dgAudio = = null ) { _dgAudio = new DGAudio ( this ) ; }
return _dgAudio ;
}
}
DGControl _dgControl ;
2014-11-14 10:49:12 +08:00
DGControl ITripletControl . DGControl { get { return DGControl ; } }
protected DGControl DGControl
2014-09-12 10:24:27 +08:00
{
get
{
if ( _dgControl = = null ) { _dgControl = new DGControl ( this ) ; }
return _dgControl ;
}
}
2014-11-14 10:49:12 +08:00
2014-09-12 10:24:27 +08:00
DGImage _dgImage ;
2014-11-14 10:49:12 +08:00
DGImage ITripletControl . DGImage { get { return DGImage ; } }
protected DGImage DGImage
2014-09-12 10:24:27 +08:00
{
get
{
if ( _dgImage = = null ) { _dgImage = new DGImage ( this ) ; }
return _dgImage ;
}
}
DGCustom _dgCustom ;
2014-11-14 10:49:12 +08:00
DGCustom ITripletControl . DGCustom { get { return DGCustom ; } }
protected DGCustom DGCustom
2014-09-12 10:24:27 +08:00
{
get
{
if ( _dgCustom = = null ) { _dgCustom = new DGCustom ( this ) ; }
return _dgCustom ;
}
}
/// <summary>
/// Enables the source to start transferring.
/// </summary>
/// <param name="mode">The mode.</param>
/// <param name="modal">if set to <c>true</c> any driver UI will display as modal.</param>
/// <param name="windowHandle">The window handle if modal.</param>
/// <returns></returns>
ReturnCode ITwainSessionInternal . EnableSource ( SourceEnableMode mode , bool modal , IntPtr windowHandle )
{
2016-02-23 07:55:29 +08:00
_closeRequested = false ;
2016-03-23 18:06:39 +08:00
DisableReason = Message . Null ;
2014-09-12 10:24:27 +08:00
var rc = ReturnCode . Failure ;
2017-04-24 22:11:41 +08:00
_msgLoopHook ? . Invoke ( ( ) = >
2014-09-12 10:24:27 +08:00
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Thread {0}: EnableSource with {1}." , Thread . CurrentThread . ManagedThreadId , mode ) ;
2014-09-12 10:24:27 +08:00
_twui = new TWUserInterface ( ) ;
_twui . ShowUI = mode = = SourceEnableMode . ShowUI ;
_twui . ModalUI = modal ;
_twui . hParent = windowHandle ;
2015-01-08 20:05:22 +08:00
if ( mode = = SourceEnableMode . ShowUIOnly )
2014-09-12 10:24:27 +08:00
{
2015-01-08 20:05:22 +08:00
rc = ( ( ITwainSessionInternal ) this ) . DGControl . UserInterface . EnableDSUIOnly ( _twui ) ;
2014-09-12 10:24:27 +08:00
}
2015-01-08 20:05:22 +08:00
else
2014-09-12 10:24:27 +08:00
{
2015-01-08 20:05:22 +08:00
rc = ( ( ITwainSessionInternal ) this ) . DGControl . UserInterface . EnableDS ( _twui ) ;
2014-09-12 10:24:27 +08:00
}
} ) ;
return rc ;
}
bool _disabling ;
ReturnCode ITwainSessionInternal . DisableSource ( )
{
var rc = ReturnCode . Failure ;
if ( ! _disabling ) // temp hack as a workaround to this being called from multiple threads (xfer logic & closedsreq msg)
{
_disabling = true ;
try
{
2017-04-24 22:11:41 +08:00
_msgLoopHook ? . Invoke ( ( ) = >
2014-09-12 10:24:27 +08:00
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Thread {0}: DisableSource." , Thread . CurrentThread . ManagedThreadId ) ;
2014-09-12 10:24:27 +08:00
rc = ( ( ITwainSessionInternal ) this ) . DGControl . UserInterface . DisableDS ( _twui ) ;
if ( rc = = ReturnCode . Success )
{
SafeAsyncSyncableRaiseOnEvent ( OnSourceDisabled , SourceDisabled ) ;
}
} ) ;
}
finally
{
_disabling = false ;
}
}
return rc ;
}
#endregion
2014-09-12 19:16:10 +08:00
#region IWinMessageFilter Members
/// <summary>
2015-01-04 00:29:38 +08:00
/// Checks and handles the message if it's a TWAIN message.
2014-09-12 19:16:10 +08:00
/// </summary>
/// <param name="hwnd">The window handle.</param>
/// <param name="msg">The message.</param>
/// <param name="wParam">The w parameter.</param>
/// <param name="lParam">The l parameter.</param>
/// <returns>
/// true if handled internally.
/// </returns>
public bool IsTwainMessage ( IntPtr hwnd , int msg , IntPtr wParam , IntPtr lParam )
{
bool handled = false ;
// this handles the message from a typical WndProc message loop and check if it's from the TWAIN source.
if ( _state > = 5 )
{
// transform it into a pointer for twain
IntPtr msgPtr = IntPtr . Zero ;
try
{
2015-02-19 09:02:39 +08:00
var winMsg = new MESSAGE ( hwnd , msg , wParam , lParam ) ;
2014-09-12 19:16:10 +08:00
// no need to do another lock call when using marshal alloc
msgPtr = Marshal . AllocHGlobal ( Marshal . SizeOf ( winMsg ) ) ;
Marshal . StructureToPtr ( winMsg , msgPtr , false ) ;
var evt = new TWEvent ( ) ;
evt . pEvent = msgPtr ;
if ( handled = ( ( ( ITwainSessionInternal ) this ) . DGControl . Event . ProcessEvent ( evt ) = = ReturnCode . DSEvent ) )
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Thread {0}: HandleWndProcMessage at state {1} with MSG={2}." , Thread . CurrentThread . ManagedThreadId , State , evt . TWMessage ) ;
2014-09-12 19:16:10 +08:00
HandleSourceMsg ( evt . TWMessage ) ;
}
}
finally
{
if ( msgPtr ! = IntPtr . Zero ) { Marshal . FreeHGlobal ( msgPtr ) ; }
}
}
return handled ;
}
#endregion
#region handle twain ds message
ReturnCode HandleCallback ( TWIdentity origin , TWIdentity destination , DataGroups dg , DataArgumentType dat , Message msg , IntPtr data )
{
if ( origin ! = null & & CurrentSource ! = null & & origin . Id = = CurrentSource . Identity . Id & & _state > = 5 )
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Thread {0}: CallbackHandler at state {1} with MSG={2}." , Thread . CurrentThread . ManagedThreadId , State , msg ) ;
2014-09-12 19:16:10 +08:00
// spec says we must handle this on the thread that enabled the DS.
// by using the internal dispatcher this will be the case.
2015-01-04 00:29:38 +08:00
// In any event the trick to get this thing working is to return from the callback first
// before trying to process the msg or there will be unpredictable errors.
2017-04-24 22:11:41 +08:00
_msgLoopHook ? . BeginInvoke ( ( ) = >
2014-09-12 19:16:10 +08:00
{
HandleSourceMsg ( msg ) ;
} ) ;
return ReturnCode . Success ;
}
return ReturnCode . Failure ;
}
// final method that handles msg from the source, whether it's from wndproc or callbacks
void HandleSourceMsg ( Message msg )
{
2015-07-04 23:18:03 +08:00
PlatformInfo . Current . Log . Debug ( "Got TWAIN msg " + msg ) ;
2014-09-12 19:16:10 +08:00
switch ( msg )
{
case Message . XferReady :
if ( State < 6 )
{
State = 6 ;
}
TransferLogic . DoTransferRoutine ( this ) ;
break ;
case Message . DeviceEvent :
TWDeviceEvent de ;
var rc = ( ( ITwainSessionInternal ) this ) . DGControl . DeviceEvent . Get ( out de ) ;
if ( rc = = ReturnCode . Success )
{
SafeSyncableRaiseOnEvent ( OnDeviceEvent , DeviceEvent , new DeviceEventArgs ( de ) ) ;
}
break ;
case Message . CloseDSReq :
case Message . CloseDSOK :
2016-03-23 18:06:39 +08:00
DisableReason = msg ;
2014-09-12 19:16:10 +08:00
// even though it says closeDS it's really disable.
// dsok is sent if source is enabled with uionly
// some sources send this at other states so do a step down
if ( State > 5 )
{
2016-02-23 07:55:29 +08:00
// rather than do a close here let the transfer logic handle the close down now
//ForceStepDown(4);
_closeRequested = true ;
2014-09-12 19:16:10 +08:00
}
else if ( State = = 5 )
{
// needs this state check since some source sends this more than once
( ( ITwainSessionInternal ) this ) . DisableSource ( ) ;
}
break ;
}
}
#endregion
2014-09-12 10:24:27 +08:00
}
}