2014-04-04 06:52:11 +08:00
|
|
|
TWAIN Application-Side Library
|
2014-04-03 07:01:21 +08:00
|
|
|
==============================
|
|
|
|
|
2014-04-04 06:52:11 +08:00
|
|
|
Info
|
2014-04-03 07:01:21 +08:00
|
|
|
--------------------------------------
|
2014-05-01 07:38:46 +08:00
|
|
|
This is a library created to make working with [TWAIN](http://twain.org/) interface possible in dotnet.
|
|
|
|
This project has these features/goals:
|
2014-04-03 07:01:21 +08:00
|
|
|
|
2014-05-01 07:38:46 +08:00
|
|
|
* Targets latest TWAIN version (2.3 at this writing)
|
2014-04-13 21:30:39 +08:00
|
|
|
* Supports all the TWAIN functions in the spec
|
2014-05-25 21:29:44 +08:00
|
|
|
* Optionally hosts an internal message loop so there's no need to hook into application UI thread
|
2014-04-04 06:52:11 +08:00
|
|
|
|
2014-04-15 07:51:36 +08:00
|
|
|
The solution contains tester projects in winform, wpf, and even (gasp!) console.
|
2014-09-15 19:24:13 +08:00
|
|
|
A nuget package is also [available here](https://www.nuget.org/packages/ntwain)
|
|
|
|
(NOTE: this doc describes v3. For older version go to Source and choose v2 branch.)
|
2014-04-08 10:50:09 +08:00
|
|
|
|
|
|
|
Using the lib
|
|
|
|
--------------------------------------
|
|
|
|
To properly use this lib you will need to be reasonably familiar with the TWAIN spec
|
2014-09-12 09:14:41 +08:00
|
|
|
and understand how it works in general (especially capability).
|
2014-05-01 07:38:46 +08:00
|
|
|
The spec can be downloaded from [twain.org](http://twain.org/).
|
2014-04-13 21:30:39 +08:00
|
|
|
|
2014-05-24 07:18:07 +08:00
|
|
|
Except for those that have been abstracted away with .net equivalents, most triplet operations are
|
2014-05-01 07:38:46 +08:00
|
|
|
provided as-is so you will need to know when and how to use them.
|
2014-05-24 07:18:07 +08:00
|
|
|
There are no high-level, single-line scan-a-page-for-me-now functions yet.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
|
|
|
The main class to use is TwainSession. You can either use it directly by subscribing
|
2014-09-12 09:14:41 +08:00
|
|
|
to the important events or sub-class it and override the On* methods related to those events.
|
2014-05-01 07:38:46 +08:00
|
|
|
The sample projects contain both usages. Note that an application process should only
|
2014-09-12 09:14:41 +08:00
|
|
|
have one active (open) TwainSession at a time.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
// can use the utility method to create appId or make one yourself
|
|
|
|
var appId = TWIdentity.CreateFromAssembly(DataGroups.Image, Assembly.GetExecutingAssembly());
|
|
|
|
|
|
|
|
// new it up and handle events
|
|
|
|
var session = new TwainSession(appId);
|
|
|
|
|
|
|
|
session.TransferReady += ...
|
|
|
|
session.DataTransferred += ...
|
|
|
|
|
2014-05-24 07:18:07 +08:00
|
|
|
// finally open it
|
|
|
|
session.Open();
|
|
|
|
|
2014-05-01 07:38:46 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
TwainSession class provides many events, but these 2 are the most important
|
|
|
|
|
|
|
|
* TransferReady - fired before a transfer occurs. You can cancel the current transfer
|
|
|
|
or all subsequent transfers using the event object.
|
|
|
|
* DataTransferred - fired after the transfer has occurred. The data available depends on
|
2015-02-19 09:02:39 +08:00
|
|
|
what you've specified using the TWAIN API before starting the transfer. If using image
|
2015-02-19 10:06:07 +08:00
|
|
|
native transfer, the event arg provides a quick GetNativeImage() method to convert the
|
|
|
|
data to a System.Drawing.Image.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-12-30 07:24:23 +08:00
|
|
|
NOTE: do not try to close the source/session in the handler of these 2 events or something
|
|
|
|
unpredictable will happen. Either let the scan run its course or cancel the scan using the flag
|
|
|
|
in the TransferReady event arg.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-05-24 07:18:07 +08:00
|
|
|
Once you've setup and opened the session, you can get available sources, pick one to use,
|
2014-12-30 07:24:23 +08:00
|
|
|
and call Open() to start using it as the example below.
|
2014-05-24 07:18:07 +08:00
|
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
|
2014-09-12 09:14:41 +08:00
|
|
|
// choose and open the first source found
|
2014-09-15 19:24:13 +08:00
|
|
|
// note that TwainSession implements IEnumerable<DataSource> so we can do this
|
|
|
|
DataSource myDS = session.FirstOrDefault();
|
2014-05-24 07:18:07 +08:00
|
|
|
myDS.Open();
|
|
|
|
|
|
|
|
```
|
|
|
|
|
2014-11-30 11:43:07 +08:00
|
|
|
At this point you can use all the typical TWAIN triplet API for working with a data source.
|
|
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
|
|
|
|
// all exposed triplet operations are defined through these properties.
|
|
|
|
// if the operation you want is not available, that most likely means
|
|
|
|
// it's not for consumer use or there's an equivalent API in this lib.
|
|
|
|
myDS.DGControl...;
|
|
|
|
myDS.DGImage...;
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
Additionally, the DataSource class itself has some handy pre-defined wrappers for common capability
|
|
|
|
negotiation. You can use the wrapper properties to see what capabilities or their operations are
|
|
|
|
supported. You also won't have to keep track of what value types to use since the wrapper defines it
|
|
|
|
for you. Most capabilities only require a simple single value to set
|
|
|
|
and the wrapper makes it easy to do that (see example below):
|
|
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
|
|
|
|
// The wrapper has many methods that corresponds to the TWAIN capability triplet msgs like
|
2015-02-19 10:55:07 +08:00
|
|
|
// GetValues(), GetCurrent(), GetDefault(), SetValue(), etc.
|
2014-11-30 11:43:07 +08:00
|
|
|
// (see TWAIN pdf doc for reference)
|
|
|
|
|
|
|
|
|
|
|
|
// This example sets pixel type of scanned image to BW and
|
|
|
|
// IPixelType is the wrapper property on the data source.
|
|
|
|
// (note: the name of the wrapper property is the same as the CapabilityId enum)
|
|
|
|
PixelType myValue = PixelType.BlackWhite;
|
|
|
|
|
|
|
|
if (myDS.ICapPixelType.CanSet &&
|
2015-02-19 10:55:07 +08:00
|
|
|
myDS.ICapPixelType.GetValues().Contains(myValue))
|
2014-11-30 11:43:07 +08:00
|
|
|
{
|
2015-02-19 10:55:07 +08:00
|
|
|
myDS.ICapPixelType.SetValue(myValue);
|
2014-11-30 11:43:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-09-12 09:14:41 +08:00
|
|
|
When you're ready to get into transfer mode, just call Enable() on the source object.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-05-24 07:18:07 +08:00
|
|
|
```
|
|
|
|
#!c#
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-09-12 10:24:27 +08:00
|
|
|
myDS.Enable(...);
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-05-24 07:18:07 +08:00
|
|
|
```
|
|
|
|
|
2014-11-30 11:43:07 +08:00
|
|
|
After transfer has completed (remember that you are notified of this with the SourceDisabled event from session)
|
2014-05-24 07:18:07 +08:00
|
|
|
and you're done with TWAIN, you can close the source and the session in sequence to clean things up.
|
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
|
|
|
|
myDS.Close();
|
|
|
|
session.Close();
|
|
|
|
|
|
|
|
```
|
2014-05-01 07:38:46 +08:00
|
|
|
|
2014-04-17 08:39:30 +08:00
|
|
|
|
|
|
|
Caveats
|
|
|
|
--------------------------------------
|
2015-02-19 09:02:39 +08:00
|
|
|
At the moment the DataTransferredEventArgs only provides conversion routine to
|
2015-02-19 10:06:07 +08:00
|
|
|
an image when using native transfer.
|
2015-02-19 09:02:39 +08:00
|
|
|
If other transfer methods are used you'll have to deal with them yourself.
|
2014-04-15 07:51:36 +08:00
|
|
|
|
2014-12-30 07:24:23 +08:00
|
|
|
If you just call session.Open() without passing a message loop hook argument, an
|
|
|
|
internal message loop will be started behind the scenes. When this happens the session events may be raised from another thread.
|
2014-05-25 21:29:44 +08:00
|
|
|
If you would like things marshalled to a UI thread then set the experimental SynchronizationContext property
|
2014-12-30 07:24:23 +08:00
|
|
|
to the one from your preferred UI thread.
|
2014-05-01 07:38:46 +08:00
|
|
|
|
|
|
|
```
|
|
|
|
#!c#
|
|
|
|
// set this while in a UI thread
|
|
|
|
session.SynchronizationContext = SynchronizationContext.Current;
|
|
|
|
|
|
|
|
```
|
2014-05-24 07:18:07 +08:00
|
|
|
Note that on certain scanner drivers this may hang the
|
|
|
|
application due to their use of modal dialogs, so if you find yourself in that position
|
|
|
|
you'll have to find another way to synchronize data to UI threads.
|
|
|
|
|
2014-04-19 21:10:15 +08:00
|
|
|
|
|
|
|
64-bit OS
|
|
|
|
--------------------------------------
|
|
|
|
If the application process is running in 64-bit then you will need to make sure you have the
|
2014-05-01 07:38:46 +08:00
|
|
|
newer data source manager (twaindsm.dll) from below installed.
|
2014-05-24 07:18:07 +08:00
|
|
|
|
2014-04-19 21:10:15 +08:00
|
|
|
[DSM from TWAIN.org](http://sourceforge.net/projects/twain-dsm/files/TWAIN%20DSM%202%20Win/)
|
|
|
|
|
2014-09-12 09:14:41 +08:00
|
|
|
In fact, installing the new DSM is recommended whether you're running in 64-bit or not.
|
|
|
|
|
|
|
|
If the scanner's TWAIN driver is still 32-bit (and most likely it will be) then you'll have no choice but to
|
2014-12-30 07:24:23 +08:00
|
|
|
compile the exe using the NTwain.dll in x86.
|