2014-04-06 04:48:44 +08:00
using NTwain.Data ;
2014-04-21 04:57:38 +08:00
using NTwain.Internals ;
2014-04-17 19:11:39 +08:00
using NTwain.Properties ;
2014-04-06 04:48:44 +08:00
using NTwain.Triplets ;
using System ;
using System.Collections.Generic ;
using System.ComponentModel ;
2014-04-06 06:33:21 +08:00
using System.Diagnostics ;
2014-04-06 09:54:08 +08:00
using System.IO ;
2014-04-06 06:33:21 +08:00
using System.Runtime.InteropServices ;
using System.Threading ;
2014-04-06 04:48:44 +08:00
namespace NTwain
{
2014-04-06 06:33:21 +08:00
/// <summary>
2014-04-20 20:10:20 +08:00
/// Basic class for interfacing with TWAIN. You should only have one of this per application process.
2014-04-06 06:33:21 +08:00
/// </summary>
2014-04-06 09:54:08 +08:00
public class TwainSession : ITwainStateInternal , ITwainOperation
2014-04-06 04:48:44 +08:00
{
/// <summary>
2014-04-17 19:11:39 +08:00
/// Initializes a new instance of the <see cref="TwainSession" /> class.
2014-04-06 04:48:44 +08:00
/// </summary>
/// <param name="appId">The app id.</param>
/// <exception cref="System.ArgumentNullException"></exception>
2014-04-06 09:54:08 +08:00
public TwainSession ( TWIdentity appId )
2014-04-06 04:48:44 +08:00
{
if ( appId = = null ) { throw new ArgumentNullException ( "appId" ) ; }
2014-04-21 04:57:38 +08:00
2014-04-06 06:33:21 +08:00
_appId = appId ;
2014-04-06 20:22:59 +08:00
( ( ITwainStateInternal ) this ) . ChangeState ( 1 , false ) ;
2014-04-06 04:48:44 +08:00
EnforceState = true ;
2014-04-15 07:04:48 +08:00
2014-04-19 21:10:15 +08:00
MessageLoop . Instance . EnsureStarted ( HandleWndProcMessage ) ;
2014-04-06 04:48:44 +08:00
}
2014-04-06 06:33:21 +08:00
TWIdentity _appId ;
object _callbackObj ; // kept around so it doesn't get gc'ed
TWUserInterface _twui ;
2014-04-06 04:48:44 +08:00
2014-04-10 19:30:00 +08:00
static readonly CapabilityId [ ] _emptyCapList = new CapabilityId [ 0 ] ;
2014-04-06 08:14:19 +08:00
private IList < CapabilityId > _supportedCaps ;
/// <summary>
/// Gets the supported caps for the current source.
/// </summary>
/// <value>
/// The supported caps.
/// </value>
public IList < CapabilityId > SupportedCaps
{
get
{
if ( _supportedCaps = = null & & State > 3 )
{
_supportedCaps = this . GetCapabilities ( ) ;
}
2014-04-10 19:30:00 +08:00
return _supportedCaps ? ? _emptyCapList ;
2014-04-06 08:14:19 +08:00
}
private set
{
_supportedCaps = value ;
RaisePropertyChanged ( "SupportedCaps" ) ;
}
}
2014-04-17 08:39:30 +08:00
/// <summary>
/// EXPERIMENTAL. Gets or sets the optional synchronization context.
/// This allows events to be raised on the thread
/// associated with the context.
/// </summary>
/// <value>
/// The synchronization context.
/// </value>
public SynchronizationContext SynchronizationContext { get ; set ; }
2014-04-06 08:14:19 +08:00
2014-04-06 04:48:44 +08:00
#region ITwainStateInternal Members
2014-04-06 06:33:21 +08:00
/// <summary>
/// Gets the app id used for the session.
/// </summary>
/// <value>The app id.</value>
2014-04-16 18:53:05 +08:00
TWIdentity ITwainStateInternal . AppId { get { return _appId ; } }
2014-04-06 06:33:21 +08:00
2014-04-06 04:48:44 +08:00
/// <summary>
/// Gets or sets a value indicating whether calls to triplets will verify the current twain session state.
/// </summary>
/// <value>
/// <c>true</c> if state value is enforced; otherwise, <c>false</c>.
/// </value>
public bool EnforceState { get ; set ; }
void ITwainStateInternal . ChangeState ( int newState , bool notifyChange )
{
2014-04-06 08:14:19 +08:00
_state = newState ;
2014-04-06 04:48:44 +08:00
if ( notifyChange )
{
RaisePropertyChanged ( "State" ) ;
2014-04-17 08:39:30 +08:00
SafeAsyncSyncableRaiseOnEvent ( OnStateChanged , StateChanged ) ;
2014-04-06 04:48:44 +08:00
}
}
2014-04-21 04:57:38 +08:00
ICommittable ITwainStateInternal . GetPendingStateChanger ( int newState )
2014-04-06 04:48:44 +08:00
{
return new TentativeStateCommitable ( this , newState ) ;
}
void ITwainStateInternal . ChangeSourceId ( TWIdentity sourceId )
{
SourceId = sourceId ;
RaisePropertyChanged ( "SourceId" ) ;
2014-04-17 08:39:30 +08:00
SafeAsyncSyncableRaiseOnEvent ( OnSourceChanged , SourceChanged ) ;
2014-04-06 04:48:44 +08:00
}
#endregion
#region ITwainState Members
/// <summary>
/// Gets the source id used for the session.
/// </summary>
/// <value>
/// The source id.
/// </value>
public TWIdentity SourceId { get ; private set ; }
2014-04-06 08:14:19 +08:00
int _state ;
2014-04-06 04:48:44 +08:00
/// <summary>
/// Gets the current state number as defined by the TWAIN spec.
/// </summary>
/// <value>
/// The state.
/// </value>
2014-04-06 08:14:19 +08:00
public int State
{
get { return _state ; }
protected set
{
if ( value > 0 & & value < 8 )
{
_state = value ;
RaisePropertyChanged ( "State" ) ;
2014-04-17 08:39:30 +08:00
SafeAsyncSyncableRaiseOnEvent ( OnStateChanged , StateChanged ) ;
2014-04-06 08:14:19 +08:00
}
}
}
2014-04-06 04:48:44 +08:00
#endregion
#region ITwainOperation Members
DGAudio _dgAudio ;
/// <summary>
/// Gets the triplet operations defined for audio data group.
/// </summary>
public DGAudio DGAudio
{
get
{
if ( _dgAudio = = null ) { _dgAudio = new DGAudio ( this ) ; }
return _dgAudio ;
}
}
DGControl _dgControl ;
/// <summary>
/// Gets the triplet operations defined for control data group.
/// </summary>
public DGControl DGControl
{
get
{
if ( _dgControl = = null ) { _dgControl = new DGControl ( this ) ; }
return _dgControl ;
}
}
DGImage _dgImage ;
/// <summary>
/// Gets the triplet operations defined for image data group.
/// </summary>
public DGImage DGImage
{
get
{
if ( _dgImage = = null ) { _dgImage = new DGImage ( this ) ; }
return _dgImage ;
}
}
#endregion
#region INotifyPropertyChanged Members
/// <summary>
/// Occurs when a property value changes.
/// </summary>
public event PropertyChangedEventHandler PropertyChanged ;
/// <summary>
/// Raises the <see cref="PropertyChanged"/> event.
/// </summary>
/// <param name="propertyName">Name of the property.</param>
protected void RaisePropertyChanged ( string propertyName )
{
2014-04-17 08:39:30 +08:00
if ( SynchronizationContext = = null )
{
try
{
var hand = PropertyChanged ;
if ( hand ! = null ) { hand ( this , new PropertyChangedEventArgs ( propertyName ) ) ; }
}
catch { }
}
else
{
SynchronizationContext . Post ( o = >
{
try
{
var hand = PropertyChanged ;
if ( hand ! = null ) { hand ( this , new PropertyChangedEventArgs ( propertyName ) ) ; }
}
catch { }
} , null ) ;
}
2014-04-06 04:48:44 +08:00
}
#endregion
2014-04-06 06:33:21 +08:00
#region privileged calls that causes state change in TWAIN
/// <summary>
/// Opens the data source manager. This must be the first method used
2014-04-13 21:30:39 +08:00
/// before using other TWAIN functions. Calls to this must be followed by <see cref="CloseManager"/> when done with a TWAIN session.
2014-04-06 06:33:21 +08:00
/// </summary>
/// <returns></returns>
2014-04-15 07:04:48 +08:00
public ReturnCode OpenManager ( )
2014-04-06 06:33:21 +08:00
{
2014-04-20 09:02:44 +08:00
var rc = ReturnCode . Failure ;
2014-04-15 07:30:25 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: OpenManager." , Thread . CurrentThread . ManagedThreadId ) ) ;
rc = DGControl . Parent . OpenDsm ( MessageLoop . Instance . LoopHandle ) ;
if ( rc = = ReturnCode . Success )
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
// if twain2 then get memory management functions
if ( ( _appId . DataFunctionalities & DataFunctionalities . Dsm2 ) = = DataFunctionalities . Dsm2 )
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
TWEntryPoint entry ;
rc = DGControl . EntryPoint . Get ( out entry ) ;
if ( rc = = ReturnCode . Success )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager = entry ;
2014-04-15 07:30:25 +08:00
Debug . WriteLine ( "Using TWAIN2 memory functions." ) ;
}
else
{
CloseManager ( ) ;
}
2014-04-06 06:33:21 +08:00
}
}
2014-04-15 07:30:25 +08:00
} ) ;
2014-04-06 06:33:21 +08:00
return rc ;
}
/// <summary>
/// Closes the data source manager.
/// </summary>
/// <returns></returns>
public ReturnCode CloseManager ( )
{
2014-04-20 20:10:20 +08:00
var rc = ReturnCode . Failure ;
2014-04-15 07:30:25 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
{
Debug . WriteLine ( string . Format ( "Thread {0}: CloseManager." , Thread . CurrentThread . ManagedThreadId ) ) ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:30:25 +08:00
rc = DGControl . Parent . CloseDsm ( MessageLoop . Instance . LoopHandle ) ;
} ) ;
2014-04-06 06:33:21 +08:00
return rc ;
}
/// <summary>
/// Loads the specified source into main memory and causes its initialization.
2014-04-20 20:10:20 +08:00
/// Calls to this must be followed by
2014-04-13 21:30:39 +08:00
/// <see cref="CloseSource" /> when not using it anymore.
2014-04-06 06:33:21 +08:00
/// </summary>
/// <param name="sourceProductName">Name of the source.</param>
/// <returns></returns>
2014-04-20 20:10:20 +08:00
/// <exception cref="System.ArgumentException">sourceProductName</exception>
2014-04-06 06:33:21 +08:00
public ReturnCode OpenSource ( string sourceProductName )
{
2014-04-17 19:11:39 +08:00
if ( string . IsNullOrEmpty ( sourceProductName ) ) { throw new ArgumentException ( Resources . SourceRequired , "sourceProductName" ) ; }
2014-04-06 06:33:21 +08:00
2014-04-20 20:10:20 +08:00
var rc = ReturnCode . Failure ;
2014-04-15 07:30:25 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
{
Debug . WriteLine ( string . Format ( "Thread {0}: OpenSource." , Thread . CurrentThread . ManagedThreadId ) ) ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:30:25 +08:00
var source = new TWIdentity ( ) ;
source . ProductName = sourceProductName ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:30:25 +08:00
rc = DGControl . Identity . OpenDS ( source ) ;
} ) ;
2014-04-06 06:33:21 +08:00
return rc ;
}
/// <summary>
/// When an application is finished with a Source, it must formally close the session between them
/// using this operation. This is necessary in case the Source only supports connection with a single
/// application (many desktop scanners will behave this way). A Source such as this cannot be
/// accessed by other applications until its current session is terminated
/// </summary>
/// <returns></returns>
public ReturnCode CloseSource ( )
{
2014-04-20 09:02:44 +08:00
var rc = ReturnCode . Failure ;
2014-04-15 07:30:25 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: CloseSource." , Thread . CurrentThread . ManagedThreadId ) ) ;
rc = DGControl . Identity . CloseDS ( ) ;
if ( rc = = ReturnCode . Success )
{
SupportedCaps = null ;
}
} ) ;
2014-04-06 06:33:21 +08:00
return rc ;
}
/// <summary>
2014-04-13 21:30:39 +08:00
/// Enables the source to start transferring.
2014-04-06 06:33:21 +08:00
/// </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>
/// <exception cref="ArgumentNullException">context</exception>
2014-04-15 07:04:48 +08:00
public ReturnCode EnableSource ( SourceEnableMode mode , bool modal , IntPtr windowHandle )
2014-04-06 06:33:21 +08:00
{
2014-04-20 09:02:44 +08:00
var rc = ReturnCode . Failure ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:04:48 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: EnableSource." , Thread . CurrentThread . ManagedThreadId ) ) ;
2014-04-15 07:04:48 +08:00
// app v2.2 or higher uses callback2
if ( _appId . ProtocolMajor > = 2 & & _appId . ProtocolMinor > = 2 )
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:04:48 +08:00
var cb = new TWCallback2 ( HandleCallback ) ;
var rc2 = DGControl . Callback2 . RegisterCallback ( cb ) ;
if ( rc2 = = ReturnCode . Success )
{
Debug . WriteLine ( "Registered callback2 OK." ) ;
_callbackObj = cb ;
}
2014-04-06 06:33:21 +08:00
}
2014-04-15 07:04:48 +08:00
else
{
var cb = new TWCallback ( HandleCallback ) ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:04:48 +08:00
var rc2 = DGControl . Callback . RegisterCallback ( cb ) ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:04:48 +08:00
if ( rc2 = = ReturnCode . Success )
{
Debug . WriteLine ( "Registered callback OK." ) ;
_callbackObj = cb ;
}
2014-04-06 06:33:21 +08:00
}
2014-04-15 07:04:48 +08:00
_twui = new TWUserInterface ( ) ;
_twui . ShowUI = mode = = SourceEnableMode . ShowUI ;
_twui . ModalUI = modal ;
_twui . hParent = windowHandle ;
2014-04-06 06:33:21 +08:00
2014-04-15 07:04:48 +08:00
if ( mode = = SourceEnableMode . ShowUIOnly )
{
rc = DGControl . UserInterface . EnableDSUIOnly ( _twui ) ;
}
else
{
rc = DGControl . UserInterface . EnableDS ( _twui ) ;
}
} ) ;
return rc ;
2014-04-06 06:33:21 +08:00
}
/// <summary>
/// Disables the source to end data acquisition.
/// </summary>
/// <returns></returns>
2014-04-06 08:14:19 +08:00
protected ReturnCode DisableSource ( )
2014-04-06 06:33:21 +08:00
{
2014-04-20 09:02:44 +08:00
var rc = ReturnCode . Failure ;
2014-04-15 07:04:48 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:30:25 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: DisableSource." , Thread . CurrentThread . ManagedThreadId ) ) ;
2014-04-15 07:04:48 +08:00
rc = DGControl . UserInterface . DisableDS ( _twui ) ;
if ( rc = = ReturnCode . Success )
{
2014-04-19 21:10:15 +08:00
_callbackObj = null ;
2014-04-17 08:39:30 +08:00
SafeAsyncSyncableRaiseOnEvent ( OnSourceDisabled , SourceDisabled ) ;
2014-04-15 07:04:48 +08:00
}
} ) ;
2014-04-06 06:33:21 +08:00
return rc ;
}
/// <summary>
/// Forces the stepping down of an opened source when things gets out of control.
/// Used when session state and source state become out of sync.
/// </summary>
/// <param name="targetState">State of the target.</param>
public void ForceStepDown ( int targetState )
{
Debug . WriteLine ( string . Format ( "Thread {0}: ForceStepDown." , Thread . CurrentThread . ManagedThreadId ) ) ;
2014-04-15 07:30:25 +08:00
bool origFlag = EnforceState ;
EnforceState = false ;
// From the twain spec
// Stepping Back Down the States
// DG_CONTROL / DAT_PENDINGXFERS / MSG_ENDXFER → state 7 to 6
// DG_CONTROL / DAT_PENDINGXFERS / MSG_RESET → state 6 to 5
// DG_CONTROL / DAT_USERINTERFACE / MSG_DISABLEDS → state 5 to 4
// DG_CONTROL / DAT_IDENTITY / MSG_CLOSEDS → state 4 to 3
// Ignore the status returns from the calls prior to the one yielding the desired state. For instance, if a
// call during scanning returns TWCC_SEQERROR and the desire is to return to state 5, then use the
// following commands.
// DG_CONTROL / DAT_PENDINGXFERS / MSG_ENDXFER → state 7 to 6
// DG_CONTROL / DAT_PENDINGXFERS / MSG_RESET → state 6 to 5
// Being sure to confirm that DG_CONTROL / DAT_PENDINGXFERS / MSG_RESET returned
// success, the return status from DG_CONTROL / DAT_PENDINGXFERS / MSG_ENDXFER may
// be ignored.
2014-04-15 07:51:36 +08:00
MessageLoop . Instance . Invoke ( ( ) = >
2014-04-15 07:30:25 +08:00
{
2014-04-15 07:51:36 +08:00
if ( targetState < 7 )
{
DGControl . PendingXfers . EndXfer ( new TWPendingXfers ( ) ) ;
}
if ( targetState < 6 )
{
DGControl . PendingXfers . Reset ( new TWPendingXfers ( ) ) ;
}
if ( targetState < 5 )
{
DisableSource ( ) ;
}
if ( targetState < 4 )
{
CloseSource ( ) ;
}
if ( targetState < 3 )
{
CloseManager ( ) ;
}
} ) ;
2014-04-15 07:30:25 +08:00
EnforceState = origFlag ;
2014-04-06 06:33:21 +08:00
}
#endregion
2014-04-06 04:48:44 +08:00
#region custom events and overridables
2014-04-06 07:39:41 +08:00
/// <summary>
/// Occurs when <see cref="State"/> has changed.
/// </summary>
public event EventHandler StateChanged ;
/// <summary>
/// Occurs when <see cref="SourceId"/> has changed.
/// </summary>
public event EventHandler SourceChanged ;
/// <summary>
/// Occurs when source has been disabled (back to state 4).
/// </summary>
public event EventHandler SourceDisabled ;
/// <summary>
/// Occurs when the source has generated an event.
/// </summary>
public event EventHandler < DeviceEventArgs > DeviceEvent ;
/// <summary>
/// Occurs when a data transfer is ready.
/// </summary>
public event EventHandler < TransferReadyEventArgs > TransferReady ;
/// <summary>
/// Occurs when data has been transferred.
/// </summary>
public event EventHandler < DataTransferredEventArgs > DataTransferred ;
2014-04-12 23:07:13 +08:00
/// <summary>
/// Occurs when an error has been encountered during transfer.
/// </summary>
public event EventHandler < TransferErrorEventArgs > TransferError ;
2014-04-06 04:48:44 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Raises event and if applicable marshal it synchronously to the <see cref="SynchronizationContext" /> thread.
2014-04-06 04:48:44 +08:00
/// </summary>
2014-04-17 08:39:30 +08:00
/// <typeparam name="TEventArgs">The type of the event arguments.</typeparam>
/// <param name="onEventFunc">The on event function.</param>
/// <param name="handler">The handler.</param>
2014-04-17 19:11:39 +08:00
/// <param name="e">The TEventArgs instance containing the event data.</param>
2014-04-17 08:39:30 +08:00
void SafeSyncableRaiseOnEvent < TEventArgs > ( Action < TEventArgs > onEventFunc , EventHandler < TEventArgs > handler , TEventArgs e ) where TEventArgs : EventArgs
2014-04-06 07:39:41 +08:00
{
2014-04-17 08:39:30 +08:00
var syncer = SynchronizationContext ;
if ( syncer = = null )
2014-04-06 07:39:41 +08:00
{
2014-04-06 08:50:13 +08:00
try
{
2014-04-17 08:39:30 +08:00
onEventFunc ( e ) ;
if ( handler ! = null ) { handler ( this , e ) ; }
2014-04-06 08:50:13 +08:00
}
catch { }
2014-04-06 07:39:41 +08:00
}
2014-04-17 08:39:30 +08:00
else
2014-04-06 07:39:41 +08:00
{
2014-04-17 08:39:30 +08:00
syncer . Send ( o = >
2014-04-06 08:50:13 +08:00
{
2014-04-17 08:39:30 +08:00
try
{
onEventFunc ( e ) ;
if ( handler ! = null ) { handler ( this , e ) ; }
}
catch { }
} , null ) ;
2014-04-06 07:39:41 +08:00
}
}
2014-04-06 04:48:44 +08:00
2014-04-06 06:33:21 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Raises event and if applicable marshal it asynchronously to the <see cref="SynchronizationContext"/> thread.
2014-04-06 06:33:21 +08:00
/// </summary>
2014-04-17 08:39:30 +08:00
/// <param name="onEventFunc">The on event function.</param>
/// <param name="handler">The handler.</param>
void SafeAsyncSyncableRaiseOnEvent ( Action onEventFunc , EventHandler handler )
2014-04-06 07:39:41 +08:00
{
2014-04-17 08:39:30 +08:00
var syncer = SynchronizationContext ;
if ( syncer = = null )
2014-04-06 07:39:41 +08:00
{
2014-04-06 08:50:13 +08:00
try
{
2014-04-17 08:39:30 +08:00
onEventFunc ( ) ;
if ( handler ! = null ) { handler ( this , EventArgs . Empty ) ; }
2014-04-06 08:50:13 +08:00
}
catch { }
2014-04-06 07:39:41 +08:00
}
2014-04-17 08:39:30 +08:00
else
{
syncer . Post ( o = >
{
try
{
onEventFunc ( ) ;
if ( handler ! = null ) { handler ( this , EventArgs . Empty ) ; }
}
catch { }
} , null ) ;
}
2014-04-06 07:39:41 +08:00
}
2014-04-06 06:33:21 +08:00
2014-04-06 07:39:41 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Called when <see cref="State"/> changed.
/// </summary>
protected virtual void OnStateChanged ( ) { }
/// <summary>
/// Called when <see cref="SourceId"/> changed.
/// </summary>
protected virtual void OnSourceChanged ( ) { }
/// <summary>
/// Called when source has been disabled (back to state 4).
/// </summary>
protected virtual void OnSourceDisabled ( ) { }
/// <summary>
/// Called when the source has generated an event.
2014-04-06 07:39:41 +08:00
/// </summary>
/// <param name="e">The <see cref="DeviceEventArgs"/> instance containing the event data.</param>
2014-04-17 08:39:30 +08:00
protected virtual void OnDeviceEvent ( DeviceEventArgs e ) { }
2014-04-06 06:33:21 +08:00
2014-04-06 07:39:41 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Called when a data transfer is ready.
2014-04-06 07:39:41 +08:00
/// </summary>
/// <param name="e">The <see cref="TransferReadyEventArgs"/> instance containing the event data.</param>
2014-04-17 08:39:30 +08:00
protected virtual void OnTransferReady ( TransferReadyEventArgs e ) { }
2014-04-06 06:33:21 +08:00
2014-04-06 07:39:41 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Called when data has been transferred.
2014-04-06 07:39:41 +08:00
/// </summary>
/// <param name="e">The <see cref="DataTransferredEventArgs"/> instance containing the event data.</param>
2014-04-17 08:39:30 +08:00
protected virtual void OnDataTransferred ( DataTransferredEventArgs e ) { }
2014-04-06 07:39:41 +08:00
2014-04-12 23:07:13 +08:00
/// <summary>
2014-04-17 08:39:30 +08:00
/// Called when an error has been encountered during transfer.
2014-04-12 23:07:13 +08:00
/// </summary>
/// <param name="e">The <see cref="TransferErrorEventArgs"/> instance containing the event data.</param>
2014-04-17 08:39:30 +08:00
protected virtual void OnTransferError ( TransferErrorEventArgs e ) { }
2014-04-06 07:39:41 +08:00
#endregion
2014-04-06 06:33:21 +08:00
2014-04-06 19:23:49 +08:00
#region TWAIN logic during xfer work
2014-04-06 06:33:21 +08:00
2014-04-15 07:04:48 +08:00
//[EnvironmentPermissionAttribute(SecurityAction.LinkDemand)]
2014-04-21 04:57:38 +08:00
void HandleWndProcMessage ( ref WindowsHook . MESSAGE winMsg , ref bool handled )
2014-04-06 07:39:41 +08:00
{
2014-04-15 07:04:48 +08:00
// this handles the message from a typical WndProc message loop and check if it's from the TWAIN source.
2014-04-15 07:51:36 +08:00
if ( State > = 5 )
2014-04-06 07:39:41 +08:00
{
// transform it into a pointer for twain
IntPtr msgPtr = IntPtr . Zero ;
try
{
// no need to do another lock call when using marshal alloc
2014-04-19 21:10:15 +08:00
msgPtr = Marshal . AllocHGlobal ( Marshal . SizeOf ( winMsg ) ) ;
Marshal . StructureToPtr ( winMsg , msgPtr , false ) ;
2014-04-06 07:39:41 +08:00
2014-04-15 07:51:36 +08:00
var evt = new TWEvent ( ) ;
2014-04-06 07:39:41 +08:00
evt . pEvent = msgPtr ;
2014-04-15 07:04:48 +08:00
if ( handled = ( DGControl . Event . ProcessEvent ( evt ) = = ReturnCode . DSEvent ) )
2014-04-06 09:14:46 +08:00
{
2014-04-06 09:54:08 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: HandleWndProcMessage at state {1} with MSG={2}." , Thread . CurrentThread . ManagedThreadId , State , evt . TWMessage ) ) ;
2014-04-15 07:04:48 +08:00
2014-04-15 07:30:25 +08:00
HandleSourceMsg ( evt . TWMessage ) ;
2014-04-06 09:14:46 +08:00
}
2014-04-06 07:39:41 +08:00
}
finally
{
if ( msgPtr ! = IntPtr . Zero ) { Marshal . FreeHGlobal ( msgPtr ) ; }
}
}
2014-04-06 06:33:21 +08:00
}
2014-04-06 07:39:41 +08:00
ReturnCode HandleCallback ( TWIdentity origin , TWIdentity destination , DataGroups dg , DataArgumentType dat , Message msg , IntPtr data )
2014-04-06 06:33:21 +08:00
{
if ( origin ! = null & & SourceId ! = null & & origin . Id = = SourceId . Id )
{
2014-04-06 09:54:08 +08:00
Debug . WriteLine ( string . Format ( "Thread {0}: CallbackHandler at state {1} with MSG={2}." , Thread . CurrentThread . ManagedThreadId , State , msg ) ) ;
2014-04-15 07:04:48 +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.
2014-04-06 08:50:13 +08:00
2014-04-15 07:04:48 +08:00
MessageLoop . Instance . BeginInvoke ( ( ) = >
2014-04-06 06:33:21 +08:00
{
2014-04-15 07:04:48 +08:00
HandleSourceMsg ( msg ) ;
} ) ;
2014-04-06 06:33:21 +08:00
return ReturnCode . Success ;
}
return ReturnCode . Failure ;
}
2014-04-06 08:50:13 +08:00
// method that handles msg from the source, whether it's from wndproc or callbacks
void HandleSourceMsg ( Message msg )
2014-04-06 07:39:41 +08:00
{
2014-04-06 08:50:13 +08:00
switch ( msg )
{
case Message . XferReady :
if ( State < 6 )
{
State = 6 ;
}
DoTransferRoutine ( ) ;
break ;
case Message . DeviceEvent :
TWDeviceEvent de ;
var rc = DGControl . DeviceEvent . Get ( out de ) ;
if ( rc = = ReturnCode . Success )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnDeviceEvent , DeviceEvent , new DeviceEventArgs ( de ) ) ;
2014-04-06 08:50:13 +08:00
}
break ;
case Message . CloseDSReq :
case Message . CloseDSOK :
// 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 )
{
ForceStepDown ( 4 ) ;
}
else if ( State = = 5 )
{
// needs this state check since some source sends this more than once
DisableSource ( ) ;
}
break ;
}
}
2014-04-06 07:39:41 +08:00
2014-04-06 08:50:13 +08:00
/// <summary>
2014-04-06 19:08:52 +08:00
/// Performs the TWAIN transfer routine at state 6.
2014-04-06 08:50:13 +08:00
/// </summary>
protected virtual void DoTransferRoutine ( )
2014-04-06 19:08:52 +08:00
{
var pending = new TWPendingXfers ( ) ;
2014-04-06 09:54:08 +08:00
var rc = ReturnCode . Success ;
do
{
2014-04-07 04:36:57 +08:00
#region build and raise xfer ready
2014-04-06 19:23:49 +08:00
2014-04-06 19:08:52 +08:00
TWAudioInfo audInfo ;
2014-04-07 04:36:57 +08:00
if ( DGAudio . AudioInfo . Get ( out audInfo ) ! = ReturnCode . Success )
{
audInfo = null ;
}
TWImageInfo imgInfo ;
if ( DGImage . ImageInfo . Get ( out imgInfo ) ! = ReturnCode . Success )
{
imgInfo = null ;
}
2014-04-06 19:08:52 +08:00
// ask consumer for xfer details
var preXferArgs = new TransferReadyEventArgs
2014-04-06 09:54:08 +08:00
{
2014-04-06 19:08:52 +08:00
AudioInfo = audInfo ,
2014-04-07 04:36:57 +08:00
PendingImageInfo = imgInfo ,
2014-04-06 19:08:52 +08:00
PendingTransferCount = pending . Count ,
EndOfJob = pending . EndOfJob = = 0
} ;
2014-04-06 09:54:08 +08:00
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferReady , TransferReady , preXferArgs ) ;
2014-04-06 19:08:52 +08:00
#endregion
2014-04-07 04:36:57 +08:00
#region actually handle xfer
2014-04-06 19:08:52 +08:00
if ( preXferArgs . CancelAll )
2014-04-06 09:54:08 +08:00
{
2014-04-06 19:08:52 +08:00
rc = DGControl . PendingXfers . Reset ( pending ) ;
2014-04-06 09:54:08 +08:00
}
2014-04-06 19:08:52 +08:00
else if ( ! preXferArgs . CancelCurrent )
2014-04-06 09:54:08 +08:00
{
2014-04-07 04:36:57 +08:00
DataGroups xferGroup = DataGroups . None ;
if ( DGControl . XferGroup . Get ( ref xferGroup ) ! = ReturnCode . Success )
2014-04-06 19:08:52 +08:00
{
2014-04-07 04:36:57 +08:00
xferGroup = DataGroups . None ;
2014-04-06 19:08:52 +08:00
}
2014-04-06 09:54:08 +08:00
2014-04-07 04:36:57 +08:00
if ( ( xferGroup & DataGroups . Image ) = = DataGroups . Image )
{
2014-04-10 09:30:07 +08:00
var mech = this . GetCurrentCap ( CapabilityId . ICapXferMech ) . ConvertToEnum < XferMech > ( ) ;
2014-04-07 04:36:57 +08:00
switch ( mech )
{
case XferMech . Native :
DoImageNativeXfer ( ) ;
break ;
case XferMech . Memory :
DoImageMemoryXfer ( ) ;
break ;
case XferMech . File :
DoImageFileXfer ( ) ;
break ;
case XferMech . MemFile :
DoImageMemoryFileXfer ( ) ;
break ;
}
}
if ( ( xferGroup & DataGroups . Audio ) = = DataGroups . Audio )
{
2014-04-10 09:30:07 +08:00
var mech = this . GetCurrentCap ( CapabilityId . ACapXferMech ) . ConvertToEnum < XferMech > ( ) ;
2014-04-07 04:36:57 +08:00
switch ( mech )
{
case XferMech . Native :
DoAudioNativeXfer ( ) ;
break ;
case XferMech . File :
DoAudioFileXfer ( ) ;
break ;
}
}
}
2014-04-06 19:08:52 +08:00
rc = DGControl . PendingXfers . EndXfer ( pending ) ;
2014-04-06 09:54:08 +08:00
2014-04-07 04:36:57 +08:00
#endregion
2014-04-06 19:08:52 +08:00
} while ( rc = = ReturnCode . Success & & pending . Count ! = 0 ) ;
2014-04-06 09:54:08 +08:00
2014-04-06 19:08:52 +08:00
State = 5 ;
DisableSource ( ) ;
2014-04-07 04:36:57 +08:00
2014-04-06 19:08:52 +08:00
}
2014-04-06 09:54:08 +08:00
2014-04-07 04:36:57 +08:00
#region audio xfers
2014-04-06 19:08:52 +08:00
private void DoAudioNativeXfer ( )
{
IntPtr dataPtr = IntPtr . Zero ;
IntPtr lockedPtr = IntPtr . Zero ;
try
{
var xrc = DGAudio . AudioNativeXfer . Get ( ref dataPtr ) ;
if ( xrc = = ReturnCode . XferDone )
2014-04-06 09:54:08 +08:00
{
2014-04-06 19:08:52 +08:00
State = 7 ;
if ( dataPtr ! = IntPtr . Zero )
2014-04-06 09:54:08 +08:00
{
2014-04-21 04:57:38 +08:00
lockedPtr = Platform . MemoryManager . Lock ( dataPtr ) ;
2014-04-06 09:54:08 +08:00
}
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs { NativeData = lockedPtr } ) ;
2014-04-06 19:08:52 +08:00
}
2014-04-12 23:07:13 +08:00
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-12 23:07:13 +08:00
}
}
catch ( Exception ex )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { Exception = ex } ) ;
2014-04-06 19:08:52 +08:00
}
finally
{
State = 6 ;
// data here is allocated by source so needs to use shared mem calls
if ( lockedPtr ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Unlock ( lockedPtr ) ;
2014-04-06 19:08:52 +08:00
lockedPtr = IntPtr . Zero ;
}
if ( dataPtr ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Free ( dataPtr ) ;
2014-04-06 19:08:52 +08:00
dataPtr = IntPtr . Zero ;
}
}
}
private void DoAudioFileXfer ( )
{
2014-04-06 19:23:49 +08:00
string filePath = null ;
TWSetupFileXfer setupInfo ;
if ( DGControl . SetupFileXfer . Get ( out setupInfo ) = = ReturnCode . Success )
{
filePath = setupInfo . FileName ;
}
var xrc = DGAudio . AudioFileXfer . Get ( ) ;
if ( xrc = = ReturnCode . XferDone )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs { FileDataPath = filePath } ) ;
2014-04-06 19:23:49 +08:00
}
2014-04-12 23:07:13 +08:00
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-12 23:07:13 +08:00
}
2014-04-06 19:08:52 +08:00
}
2014-04-06 09:54:08 +08:00
2014-04-06 19:08:52 +08:00
#endregion
2014-04-06 09:54:08 +08:00
2014-04-06 19:08:52 +08:00
#region image xfers
2014-04-06 09:54:08 +08:00
2014-04-06 19:08:52 +08:00
private void DoImageNativeXfer ( )
{
IntPtr dataPtr = IntPtr . Zero ;
IntPtr lockedPtr = IntPtr . Zero ;
try
{
2014-04-06 19:23:49 +08:00
var xrc = DGImage . ImageNativeXfer . Get ( ref dataPtr ) ;
2014-04-06 19:08:52 +08:00
if ( xrc = = ReturnCode . XferDone )
2014-04-06 09:54:08 +08:00
{
2014-04-06 19:08:52 +08:00
State = 7 ;
2014-04-10 19:30:00 +08:00
TWImageInfo imgInfo ;
2014-04-17 19:11:39 +08:00
TWExtImageInfo extInfo = null ;
if ( SupportedCaps . Contains ( CapabilityId . ICapExtImageInfo ) )
{
if ( DGImage . ExtImageInfo . Get ( out extInfo ) ! = ReturnCode . Success )
{
extInfo = null ;
}
}
2014-04-10 19:30:00 +08:00
if ( DGImage . ImageInfo . Get ( out imgInfo ) ! = ReturnCode . Success )
{
imgInfo = null ;
}
2014-04-06 19:08:52 +08:00
if ( dataPtr ! = IntPtr . Zero )
2014-04-06 09:54:08 +08:00
{
2014-04-21 04:57:38 +08:00
lockedPtr = Platform . MemoryManager . Lock ( dataPtr ) ;
2014-04-06 09:54:08 +08:00
}
2014-04-17 19:11:39 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs
{
NativeData = lockedPtr ,
ImageInfo = imgInfo ,
ExImageInfo = extInfo
} ) ;
if ( extInfo ! = null ) { extInfo . Dispose ( ) ; }
2014-04-06 19:08:52 +08:00
}
2014-04-12 23:07:13 +08:00
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-12 23:07:13 +08:00
}
}
catch ( Exception ex )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { Exception = ex } ) ;
2014-04-06 19:08:52 +08:00
}
finally
{
State = 6 ;
// data here is allocated by source so needs to use shared mem calls
if ( lockedPtr ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Unlock ( lockedPtr ) ;
2014-04-06 19:08:52 +08:00
lockedPtr = IntPtr . Zero ;
2014-04-06 09:54:08 +08:00
}
2014-04-06 19:08:52 +08:00
if ( dataPtr ! = IntPtr . Zero )
2014-04-06 09:54:08 +08:00
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Free ( dataPtr ) ;
2014-04-06 19:08:52 +08:00
dataPtr = IntPtr . Zero ;
2014-04-06 09:54:08 +08:00
}
2014-04-06 19:08:52 +08:00
}
}
2014-04-06 09:54:08 +08:00
2014-04-06 19:23:49 +08:00
private void DoImageFileXfer ( )
2014-04-06 19:08:52 +08:00
{
2014-04-06 19:23:49 +08:00
string filePath = null ;
TWSetupFileXfer setupInfo ;
if ( DGControl . SetupFileXfer . Get ( out setupInfo ) = = ReturnCode . Success )
{
filePath = setupInfo . FileName ;
}
var xrc = DGImage . ImageFileXfer . Get ( ) ;
if ( xrc = = ReturnCode . XferDone )
{
2014-04-10 19:30:00 +08:00
TWImageInfo imgInfo ;
2014-04-17 19:11:39 +08:00
TWExtImageInfo extInfo = null ;
if ( SupportedCaps . Contains ( CapabilityId . ICapExtImageInfo ) )
{
if ( DGImage . ExtImageInfo . Get ( out extInfo ) ! = ReturnCode . Success )
{
extInfo = null ;
}
}
2014-04-10 19:30:00 +08:00
if ( DGImage . ImageInfo . Get ( out imgInfo ) ! = ReturnCode . Success )
{
imgInfo = null ;
}
2014-04-17 19:11:39 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs
{
FileDataPath = filePath ,
ImageInfo = imgInfo ,
ExImageInfo = extInfo
} ) ;
if ( extInfo ! = null ) { extInfo . Dispose ( ) ; }
2014-04-06 19:23:49 +08:00
}
2014-04-12 23:07:13 +08:00
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-12 23:07:13 +08:00
}
2014-04-06 19:08:52 +08:00
}
2014-04-06 09:54:08 +08:00
2014-04-06 20:22:59 +08:00
private void DoImageMemoryXfer ( )
2014-04-06 19:08:52 +08:00
{
2014-04-10 19:30:00 +08:00
TWSetupMemXfer memInfo ;
if ( DGControl . SetupMemXfer . Get ( out memInfo ) = = ReturnCode . Success )
{
TWImageMemXfer xferInfo = new TWImageMemXfer ( ) ;
try
{
2014-04-10 19:52:46 +08:00
// how to tell if going to xfer in strip vs tile?
// if tile don't allocate memory in app?
2014-04-12 07:25:48 +08:00
2014-04-10 19:30:00 +08:00
xferInfo . Memory = new TWMemory
{
Flags = MemoryFlags . AppOwns | MemoryFlags . Pointer ,
Length = memInfo . Preferred ,
2014-04-21 04:57:38 +08:00
TheMem = Platform . MemoryManager . Allocate ( memInfo . Preferred )
2014-04-10 19:30:00 +08:00
} ;
2014-04-12 07:25:48 +08:00
// do the unthinkable and keep all xferred batches in memory,
// possibly defeating the purpose of mem xfer
// unless compression is used.
// todo: use array instead of memory stream?
using ( MemoryStream xferredData = new MemoryStream ( ) )
2014-04-10 19:30:00 +08:00
{
2014-04-12 07:25:48 +08:00
var xrc = ReturnCode . Success ;
do
2014-04-10 19:30:00 +08:00
{
2014-04-12 07:25:48 +08:00
xrc = DGImage . ImageMemFileXfer . Get ( xferInfo ) ;
if ( xrc = = ReturnCode . Success | |
xrc = = ReturnCode . XferDone )
2014-04-10 19:30:00 +08:00
{
2014-04-12 07:25:48 +08:00
State = 7 ;
// optimize and allocate buffer only once instead of inside the loop?
byte [ ] buffer = new byte [ ( int ) xferInfo . BytesWritten ] ;
IntPtr lockPtr = IntPtr . Zero ;
try
2014-04-10 19:30:00 +08:00
{
2014-04-21 04:57:38 +08:00
lockPtr = Platform . MemoryManager . Lock ( xferInfo . Memory . TheMem ) ;
2014-04-12 07:25:48 +08:00
Marshal . Copy ( lockPtr , buffer , 0 , buffer . Length ) ;
xferredData . Write ( buffer , 0 , buffer . Length ) ;
}
finally
{
if ( lockPtr ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Unlock ( lockPtr ) ;
2014-04-12 07:25:48 +08:00
}
2014-04-10 19:30:00 +08:00
}
}
2014-04-12 07:25:48 +08:00
} while ( xrc = = ReturnCode . Success ) ;
2014-04-10 19:30:00 +08:00
2014-04-12 07:25:48 +08:00
if ( xrc = = ReturnCode . XferDone )
2014-04-10 19:30:00 +08:00
{
2014-04-12 07:25:48 +08:00
TWImageInfo imgInfo ;
2014-04-17 19:11:39 +08:00
TWExtImageInfo extInfo = null ;
if ( SupportedCaps . Contains ( CapabilityId . ICapExtImageInfo ) )
2014-04-12 07:25:48 +08:00
{
2014-04-17 19:11:39 +08:00
if ( DGImage . ExtImageInfo . Get ( out extInfo ) ! = ReturnCode . Success )
{
extInfo = null ;
}
2014-04-12 07:25:48 +08:00
}
2014-04-17 19:11:39 +08:00
if ( DGImage . ImageInfo . Get ( out imgInfo ) ! = ReturnCode . Success )
2014-04-12 07:25:48 +08:00
{
2014-04-17 19:11:39 +08:00
imgInfo = null ;
2014-04-12 07:25:48 +08:00
}
2014-04-17 19:11:39 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs
{
MemData = xferredData . ToArray ( ) ,
ImageInfo = imgInfo ,
ExImageInfo = extInfo
} ) ;
if ( extInfo ! = null ) { extInfo . Dispose ( ) ; }
2014-04-10 19:30:00 +08:00
}
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-10 19:30:00 +08:00
}
}
}
2014-04-12 23:07:13 +08:00
catch ( Exception ex )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { Exception = ex } ) ;
2014-04-12 23:07:13 +08:00
}
2014-04-10 19:30:00 +08:00
finally
{
State = 6 ;
if ( xferInfo . Memory . TheMem ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Free ( xferInfo . Memory . TheMem ) ;
2014-04-10 19:30:00 +08:00
}
}
}
2014-04-08 07:21:33 +08:00
}
2014-04-06 20:22:59 +08:00
2014-04-08 07:21:33 +08:00
private void DoImageMemoryFileXfer ( )
{
// since it's memory-file xfer need info from both (maybe)
2014-04-06 20:22:59 +08:00
TWSetupMemXfer memInfo ;
2014-04-08 07:21:33 +08:00
TWSetupFileXfer fileInfo ;
if ( DGControl . SetupMemXfer . Get ( out memInfo ) = = ReturnCode . Success & &
DGControl . SetupFileXfer . Get ( out fileInfo ) = = ReturnCode . Success )
2014-04-06 20:22:59 +08:00
{
TWImageMemXfer xferInfo = new TWImageMemXfer ( ) ;
2014-04-08 07:21:33 +08:00
var tempFile = Path . GetTempFileName ( ) ;
string finalFile = null ;
2014-04-06 20:22:59 +08:00
try
{
2014-04-10 19:52:46 +08:00
// no strip or tile here, just chunks
2014-04-06 20:22:59 +08:00
xferInfo . Memory = new TWMemory
{
2014-04-10 19:30:00 +08:00
Flags = MemoryFlags . AppOwns | MemoryFlags . Pointer ,
2014-04-06 20:22:59 +08:00
Length = memInfo . Preferred ,
2014-04-21 04:57:38 +08:00
TheMem = Platform . MemoryManager . Allocate ( memInfo . Preferred )
2014-04-06 20:22:59 +08:00
} ;
var xrc = ReturnCode . Success ;
2014-04-08 07:21:33 +08:00
using ( var outStream = File . OpenWrite ( tempFile ) )
2014-04-06 20:22:59 +08:00
{
2014-04-08 07:21:33 +08:00
do
2014-04-06 20:22:59 +08:00
{
2014-04-08 07:21:33 +08:00
xrc = DGImage . ImageMemFileXfer . Get ( xferInfo ) ;
if ( xrc = = ReturnCode . Success | |
xrc = = ReturnCode . XferDone )
{
2014-04-10 19:30:00 +08:00
State = 7 ;
2014-04-08 07:21:33 +08:00
byte [ ] buffer = new byte [ ( int ) xferInfo . BytesWritten ] ;
2014-04-12 07:25:48 +08:00
2014-04-10 19:30:00 +08:00
IntPtr lockPtr = IntPtr . Zero ;
try
{
2014-04-21 04:57:38 +08:00
lockPtr = Platform . MemoryManager . Lock ( xferInfo . Memory . TheMem ) ;
2014-04-10 19:30:00 +08:00
Marshal . Copy ( lockPtr , buffer , 0 , buffer . Length ) ;
}
finally
{
if ( lockPtr ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Unlock ( lockPtr ) ;
2014-04-10 19:30:00 +08:00
}
}
2014-04-08 07:21:33 +08:00
outStream . Write ( buffer , 0 , buffer . Length ) ;
}
} while ( xrc = = ReturnCode . Success ) ;
}
2014-04-06 20:22:59 +08:00
2014-04-08 07:21:33 +08:00
if ( xrc = = ReturnCode . XferDone )
{
switch ( fileInfo . Format )
{
case FileFormat . Bmp :
finalFile = Path . ChangeExtension ( tempFile , ".bmp" ) ;
break ;
case FileFormat . Dejavu :
finalFile = Path . ChangeExtension ( tempFile , ".dejavu" ) ;
break ;
case FileFormat . Exif :
finalFile = Path . ChangeExtension ( tempFile , ".exit" ) ;
break ;
case FileFormat . Fpx :
finalFile = Path . ChangeExtension ( tempFile , ".fpx" ) ;
break ;
case FileFormat . Jfif :
finalFile = Path . ChangeExtension ( tempFile , ".jpg" ) ;
break ;
case FileFormat . Jp2 :
finalFile = Path . ChangeExtension ( tempFile , ".jp2" ) ;
break ;
case FileFormat . Jpx :
finalFile = Path . ChangeExtension ( tempFile , ".jpx" ) ;
break ;
case FileFormat . Pdf :
case FileFormat . PdfA :
case FileFormat . PdfA2 :
finalFile = Path . ChangeExtension ( tempFile , ".pdf" ) ;
break ;
case FileFormat . Pict :
finalFile = Path . ChangeExtension ( tempFile , ".pict" ) ;
break ;
case FileFormat . Png :
finalFile = Path . ChangeExtension ( tempFile , ".png" ) ;
break ;
case FileFormat . Spiff :
finalFile = Path . ChangeExtension ( tempFile , ".spiff" ) ;
break ;
case FileFormat . Tiff :
case FileFormat . TiffMulti :
finalFile = Path . ChangeExtension ( tempFile , ".tif" ) ;
break ;
case FileFormat . Xbm :
finalFile = Path . ChangeExtension ( tempFile , ".xbm" ) ;
break ;
default :
finalFile = Path . ChangeExtension ( tempFile , ".unknown" ) ;
break ;
2014-04-06 20:22:59 +08:00
}
2014-04-08 07:21:33 +08:00
File . Move ( tempFile , finalFile ) ;
}
2014-04-12 23:07:13 +08:00
else
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { ReturnCode = xrc , SourceStatus = this . GetSourceStatus ( ) } ) ;
2014-04-12 23:07:13 +08:00
}
}
catch ( Exception ex )
{
2014-04-17 08:39:30 +08:00
SafeSyncableRaiseOnEvent ( OnTransferError , TransferError , new TransferErrorEventArgs { Exception = ex } ) ;
2014-04-06 20:22:59 +08:00
}
finally
{
2014-04-10 19:30:00 +08:00
State = 6 ;
2014-04-06 20:22:59 +08:00
if ( xferInfo . Memory . TheMem ! = IntPtr . Zero )
{
2014-04-21 04:57:38 +08:00
Platform . MemoryManager . Free ( xferInfo . Memory . TheMem ) ;
2014-04-06 20:22:59 +08:00
}
2014-04-08 07:21:33 +08:00
if ( File . Exists ( tempFile ) )
{
File . Delete ( tempFile ) ;
}
2014-04-06 20:22:59 +08:00
}
2014-04-06 19:08:52 +08:00
2014-04-08 07:21:33 +08:00
if ( File . Exists ( finalFile ) )
{
2014-04-10 19:30:00 +08:00
TWImageInfo imgInfo ;
2014-04-17 19:11:39 +08:00
TWExtImageInfo extInfo = null ;
if ( SupportedCaps . Contains ( CapabilityId . ICapExtImageInfo ) )
{
if ( DGImage . ExtImageInfo . Get ( out extInfo ) ! = ReturnCode . Success )
{
extInfo = null ;
}
}
2014-04-10 19:30:00 +08:00
if ( DGImage . ImageInfo . Get ( out imgInfo ) ! = ReturnCode . Success )
{
imgInfo = null ;
}
2014-04-17 19:11:39 +08:00
SafeSyncableRaiseOnEvent ( OnDataTransferred , DataTransferred , new DataTransferredEventArgs
{
FileDataPath = finalFile ,
ImageInfo = imgInfo ,
ExImageInfo = extInfo
} ) ;
if ( extInfo ! = null ) { extInfo . Dispose ( ) ; }
2014-04-08 07:21:33 +08:00
}
}
2014-04-06 07:39:41 +08:00
}
2014-04-06 04:48:44 +08:00
#endregion
2014-04-06 08:30:49 +08:00
2014-04-06 19:08:52 +08:00
#endregion
2014-04-06 04:48:44 +08:00
}
}