ntwain/README.md

173 lines
6.0 KiB
Markdown
Raw Normal View History

TWAIN Application-Side Library
2014-04-03 07:01:21 +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
The solution contains tester projects in winform, wpf, and even (gasp!) console.
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
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
// 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
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
Caveats
--------------------------------------
At the moment the DataTransferredEventArgs only provides conversion routine to
2015-02-19 10:06:07 +08:00
an image when using native transfer.
If other transfer methods are used you'll have to deal with them yourself.
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.