Official Website:http://www.twain.org

一、简介

TWAIN 数据源管理程序 (DSM) 工业标准的软件库,用于从静态图像设备提取图像。绝大部分的扫描仪和数码相机都提供了 TWAIN 驱动程序,利用统一的 TWAIN 接口,应用程序可以非常方便地从这些设备中获取图像。

二、使用步骤

互联网上关于 TWAIN 编程的中文资料很少,代码更是难找到,因为我不得不仔细阅读了 www.twain.org 提供的 TWAIN Specification。下面说说使用 TWAIN 接口获取图像的简要步骤。

Windows 系统中存在一个 TWAIN_32.dll,所有的 TWAIN 操作都是通过这个 DLL 导出的 DSM_Entry 函数来实现的 (说实在话,我个人觉得 TWAIN 接口设计得太差了,看看 MS 的WIA,逻辑上非常清晰)。这个函数的声明如下:

TW_UINT16 FAR PASCAL DSM_Entry(
            pTW_IDENTITY pOrigin,   // Source of message
            pTW_IDENTITY pDest,     // Destination of message
            TW_UINT32 DG,           // Data group ID: DG_xxxx
            TW_UINT16 DAT,          // Data argument type: DAT_xxxx
            TW_UINT16 MSG,          // Message ID: MSG_xxxx
            TW_MEMREF pData         // Pointer to data
            );

1. 打开 DSM (Data Source Manager: 数据源管理器)

TWAIN 是一个数据源管理程序,应用程序首先要使用 MSG_OPENDSM 消息,打开数据源管理器。这里需要指定一个窗口句柄,应用程序应该在此窗口的消息循环中处理 TWAIN 消息 (MSG_PROCESSEVENT)。

2. 选择 DS (Data Source: 数据源)

因为一个系统中可能存在多个 TWAIN 设备,因此必须选择一个数据源。选择数据源通常有两种方式:  选择默认数据源 (MSG_GETDEFAULT) 和显示选择数据源对话框,由用户来选择数据源 (MSG_USERSELECT)。

3. 打开 DS

使用 MSG_OPENDS 消息打开数据源。

4. 设置参数

消息为 MSG_SET,设置各种参数,如获取方式、图像数量等。有些参数由设备驱动支持才有效。

5. 显示扫描界面

使用 MSG_ENABLEDS 消息,显示设备驱动提供的用户界面 (UI)。

6. 获取图像

如果用户选择扫描什么的,可以在窗口的消息循环中获取到这个事件 (MSG_XFERREADY)。此时,应用程序可以通过 DAT_SETUPFILEXFER 设置文件名,然后用 DAT_IMAGEFILEXFER 获取图像到文件中。

7. 关闭扫描界面

在窗口的消息循环中获取到 MSG_CLOSEDSREQ 或 MSG_CLOSEDSOK 消息,可以关闭扫描界面 (MSG_DISABLEDS)。

8. 关闭 DS

消息为 MSG_CLOSEDS。

9. 关闭数据源
   
    消息为 MSG_CLOSEDSM。

三、CTwainHelper 助手类

为了使用方便,我写了一个静态 TWAIN 助手类 CTwainHelper。使用 CTwainHelper 的五个函数,就可以简单地从 TWAIN 设备获取图像到文件中。使用方法如下:

  1. 调用 CTwainHelper::Initialize() 确定是否有可用的设备。
  2. 在窗口消息循环中,调用 CTwainHelper::ProcessMessage() 处理 TWAIN 消息。
  3. 要获取图像时,调用 CTwainHelper::GetImage()。
  4. 如果图像已准备好 (如用户确定扫描图像),窗口会收到 WM_COMMAND 消息,wParam 为 IDC_TwainHelper。此时应用程序可以调用 CTwainHelper::TransferImage() 获取图像到文件中。

具体使用方法请参看示例代码。
   
    CTwainHelper 可以在 Visual C++ 6.x/7.x 工程中使用,支持 UNICODE 编译。因为是静态类,要改写成 C 代码只需要做一点点少量的工作。

四、后话

当然,上面只是一种常用的步骤。其实应用程序完全可以自定义所有的步骤,比如不使用 TWAIN 驱动提供扫描对话框而直接扫描,或者扫描图像到内存中等等。详细情况请参考 TWAIN Specification,步骤大同小异,消息和参数千差万别,仔细看看应该很容易的。

如果没有 TWAIN 设备又要进行 TWAIN 程序开发,可以到 TWAIN 官方网站下载 TWAIN Developers Toolkit,安装后会有一个虚拟的 TWAIN 设备。不过应用程序在这个虚拟 TWAIN 设备中正常工作,不代表一定能在实际的 TWAIN 设备正常使用,这点需要注意。以前 CTwainHelper 就碰到过这样的情况  在虚拟 TWAIN 设备中明明是好的,在我的扫描仪上却不能扫描图像。检查后发现,原来设置了不支持的参数。
   
    最后,TWAIN 是 Technology Without A Interesting Name 缩写,直译为没有“没有让人感兴趣名字的技术”,真是一个让人摸不着头脑的名字。

TWAIN 助手类: CTwainHelper (包含示例代码 35K)
    TWAIN 官方网站: http://www.twain.org
    TWAIN 头文件: http://www.twain.org/devfiles/twain.h
    TWAIN Specification: http://www.twain.org/docs/Spec1_9_197.pdf
    TWAIN Developers Toolkit: http://www.twain.org/devfiles/twainkit.exe

Load the Source Manager and Get the DSM_Entry (State 1 to 2)
Open the Source Manager (State 2 to 3)
Select the Source (during State 3)
Open the Source (State 3 to 4)
Negotiate Capabilities with the Source (during State 4)
Request the Acquisition of Data from the Source (State 4 to 5)
Recognize that the Data Transfer is Ready (State 5 to 6)
Start and Perform the Transfer (State 6 to 7)
Conclude the Transfer (State 7 to 6 to 5)
Disconnect the TWAIN Session (State 5 to 1 in sequence)

VC++ Call Twain

Header

#ifndef __TWAINCPP_
#define __TWAINCPP_#include "Twain.h"#define TWCPP_ANYCOUNT        (-1)
#define TWCPP_CANCELTHIS    (1)
#define TWCPP_CANCELALL        (2)
#define TWCPP_DOTRANSFER    (0)class CTwain
{
public:void Deskew(LPSTR lpFilename);CTwain(HWND hWnd = NULL);virtual ~CTwain();BOOL InitTwain(HWND hWnd);void ReleaseTwain();/*  This routine must be implemented by the dervied class After setting the required values in the m_AppId structure,the derived class should call the parent class implementationRefer Pg: 51 of the Twain Specification version 1.8*/virtual void GetIdentity();virtual BOOL SelectSource();virtual BOOL OpenSource(TW_IDENTITY *pSource=NULL);virtual int  ShouldTransfer(TW_IMAGEINFO& info) { return TWCPP_DOTRANSFER;};void SetPara(int nFileFormat,LPCTSTR lpFileName,int nAD,int nPage,int nStep,BOOL bDeskdw);void GetPara(UINT &nPage);BOOL ProcessMessage(MSG msg);BOOL SelectDefaultSource();BOOL IsValidDriver() const;BOOL SourceSelected() const {return m_bSourceSelected;} ;BOOL DSMOpen() const;BOOL DSOpen() const;BOOL SourceEnabled() const { return m_bSourceEnabled;};BOOL ModalUI() const { return m_bModalUI; };TW_INT16 GetRC() const { return m_returnCode; };TW_STATUS GetStatus() const { return m_Status; };BOOL SetImageCount(TW_INT16 nCount = 1);BOOL Acquire(int numImages=1);void DoFileTransfer();void CloseDSM();
protected:BOOL CallTwainProc(pTW_IDENTITY pOrigin,pTW_IDENTITY pDest,TW_UINT32 DG,TW_UINT16 DAT,TW_UINT16 MSG,TW_MEMREF pData);void CloseDS();BOOL GetCapability(TW_CAPABILITY& twCap,TW_UINT16 cap,TW_UINT16 conType=TWON_DONTCARE16);BOOL GetCapability(TW_UINT16 cap,TW_UINT32& value);BOOL SetCapability(TW_UINT16 cap,TW_UINT16 value,BOOL sign=FALSE);BOOL SetCapability(TW_CAPABILITY& twCap);BOOL EnableSource(BOOL showUI = TRUE);BOOL GetImageInfo(TW_IMAGEINFO& info);virtual BOOL DisableSource();virtual BOOL CanClose()  { return TRUE; };void TranslateMessage(TW_EVENT& twEvent);void TransferImage();BOOL EndTransfer();void CancelTransfer();BOOL ShouldContinue();BOOL GetImage(TW_IMAGEINFO& info);//    virtual void CopyImage(HANDLE hBitmap,TW_IMAGEINFO& info)=0;protected:HINSTANCE m_hTwainDLL;DSMENTRYPROC m_pDSMProc;TW_IDENTITY m_AppId;TW_IDENTITY m_Source;TW_STATUS m_Status;TW_INT16  m_returnCode;HWND m_hMessageWnd;BOOL m_bSourceSelected;BOOL m_bDSMOpen;BOOL m_bDSOpen;BOOL m_bSourceEnabled;BOOL m_bModalUI;

Cpp

#include "stdafx.h"
#include "twaincpp.h"
#include "showpic.h"
#include "tiff2pdfdll.h"CTwain::CTwain(HWND hWnd)
{m_hTwainDLL = NULL;m_pDSMProc = NULL;m_bSourceSelected = FALSE;m_bDSOpen = m_bDSMOpen = FALSE;m_bSourceEnabled = FALSE;m_bModalUI = TRUE;m_nImageCount = TWCPP_ANYCOUNT;if(hWnd){InitTwain(hWnd);}
}CTwain::~CTwain()
{ReleaseTwain();
}/*
初始化TWAIN 接口. 构造函数中已经调用,不过如果调用了ReleaseTwain的话就需要再次调用.
hWnd是接受Twain消息的窗口句柄,通常应该是主应用程序的窗口句柄
*/
BOOL CTwain::InitTwain(HWND hWnd)
{
char libName[512];if(IsValidDriver()) {return TRUE;}memset(&m_AppId,0,sizeof(m_AppId));if(!IsWindow(hWnd)){return FALSE;}m_hMessageWnd = hWnd;strcpy(libName,"TWAIN_32.DLL");m_hTwainDLL  = LoadLibrary(libName);if(m_hTwainDLL != NULL){if(!(m_pDSMProc = (DSMENTRYPROC)GetProcAddress(m_hTwainDLL,MAKEINTRESOURCE(1)))){FreeLibrary(m_hTwainDLL);m_hTwainDLL = NULL;}}if(IsValidDriver()){GetIdentity();m_bDSMOpen= CallTwainProc(&m_AppId,NULL,DG_CONTROL,DAT_PARENT,MSG_OPENDSM,(TW_MEMREF)&m_hMessageWnd);return TRUE;}else{return FALSE;}
}/*
释放Twain接口,除非不再使用Twain接口,否则就无需调用
*/
void CTwain::ReleaseTwain()
{if(IsValidDriver()){CloseDSM();FreeLibrary(m_hTwainDLL);m_hTwainDLL = NULL;m_pDSMProc = NULL;}
}/*
如果正确的装载了驱动程序,就返回TRUE
*/
BOOL CTwain::IsValidDriver() const
{return (m_hTwainDLL && m_pDSMProc);
}BOOL CTwain::CallTwainProc(pTW_IDENTITY pOrigin,pTW_IDENTITY pDest,TW_UINT32 DG,TW_UINT16 DAT,TW_UINT16 MSG,TW_MEMREF pData)
{if(IsValidDriver()){USHORT ret_val;ret_val = (*m_pDSMProc)(pOrigin,pDest,DG,DAT,MSG,pData);m_returnCode = ret_val;if(ret_val != TWRC_SUCCESS){(*m_pDSMProc)(pOrigin,pDest,DG_CONTROL,DAT_STATUS,MSG_GET,&m_Status);}return (ret_val == TWRC_SUCCESS);}else{m_returnCode = TWRC_FAILURE;return FALSE;}
}/*
设置Twain的一些入口参数
*/
void CTwain::GetIdentity()
{// Expects all the fields in m_AppId to be set except for the id field.m_AppId.Id = 0; // Initialize to 0 (Source Managerm_AppId.Version.MajorNum = 1; //Your app's version numberm_AppId.Version.MinorNum = 5;m_AppId.Version.Language = TWLG_USA;m_AppId.Version.Country = TWCY_USA;strcpy (m_AppId.Version.Info, "3.5");m_AppId.ProtocolMajor = TWON_PROTOCOLMAJOR;m_AppId.ProtocolMinor = TWON_PROTOCOLMINOR;m_AppId.SupportedGroups = DG_IMAGE | DG_CONTROL;strcpy (m_AppId.Manufacturer, "MICSS");strcpy (m_AppId.ProductFamily, "Generic");strcpy (m_AppId.ProductName, "Twain Test");}/*
调用此函数显示选择扫描设备窗口
*/
BOOL CTwain::SelectSource()
{memset(&m_Source,0,sizeof(m_Source));if(!SourceSelected()){SelectDefaultSource();}if(CallTwainProc(&m_AppId,NULL,DG_CONTROL,DAT_IDENTITY,MSG_USERSELECT,&m_Source)){m_bSourceSelected = TRUE;}return m_bSourceSelected;
}/*
用于选择缺省扫描设备
*/
BOOL CTwain::SelectDefaultSource()
{m_bSourceSelected = CallTwainProc(&m_AppId,NULL,DG_CONTROL,DAT_IDENTITY,MSG_GETDEFAULT,&m_Source);return m_bSourceSelected;
}/*
关闭数据源(即扫描仪)
*/
void CTwain::CloseDS()
{if(DSOpen()){DisableSource();CallTwainProc(&m_AppId,NULL,DG_CONTROL,DAT_IDENTITY,MSG_CLOSEDS,(TW_MEMREF)&m_Source);m_bDSOpen = FALSE;}
}/*
关闭数据源管理器
*/
void CTwain::CloseDSM()
{if(DSMOpen()){CloseDS();CallTwainProc(&m_AppId,NULL,DG_CONTROL,DAT_PARENT,MSG_CLOSEDSM,(TW_MEMREF)&m

转载于:https://www.cnblogs.com/yefengmeander/archive/2013/01/06/2887940.html

Twain Practice相关推荐

  1. Ten Simple Rules for Effective Statistical Practice

    http://journals.plos.org/ploscompbiol/article?id=10.1371/journal.pcbi.1004961 Introduction Several m ...

  2. HDU1089-1096 A+B for Input-Output Practice 系列问题(输入输出格式练习)

    HDU1089 A+B for Input-output Practice (1) 问题描述:每行输入一对整数a和b,每行对应输出a与b的和. Sample Input 1 5 10 20 Sampl ...

  3. Programmer of Practice Manual

    这是我以前再读研究生的时候写的东东,希望搞计算机的同学,教计算机本科生学习技术的文章(非算法类) 粘在这里纪念一下. 大一寒假 结构化编程基础: 图书:<How to C> 实践过程:完成 ...

  4. TWAIN Specification Chapter 4 “Advanced Application Implementation”译——应用程序端的高级实现...

    本文是对TWAIN规范的第四章<应用程序端的高级实现>的翻译.因工作需要了解TWAIN,所以顺便译了一下.这是私人工作,您可以参考,但本人不保证不存在翻译的差错或不合宜.如果您发现有不妥的 ...

  5. Database design best practice(1):关于primary key及其它

    1. The job of the primary key is to uniquely identify records, not to store business data ; any use ...

  6. 《Git in Practice》作者访谈:关于Git的八个问题

    Mike McQuaid的著作<Git in Pratice>是一份具有动手实践性的指导,它从实践性的角度介绍了Git,提供了超过60种在操作Git库时会使用到的技巧和命令,并且为有经验的 ...

  7. chrome浏览器上传文件延迟_扫描识别工具Dynamic Web TWAIN使用教程:移动浏览器捕获(下)...

    本篇文章将继续上一篇文章为大家介绍Dynamic Web TWAIN关于移动浏览器捕获的使用教程. 步骤4 首先尝试使用移动浏览器页面 此页面适用于桌面浏览器和移动浏览器,在桌面Chrome中,如下图 ...

  8. Remoting Practice Sample

    Remoting Practice Sample 这两天研究了一下Remoting, 改了MSDN 的例子. 那个例子不是很方便. 我做了如下改进: 整个sample做成一个solution 用了wi ...

  9. 从任何兼容 TWAIN 的设备获取图象的控件Dynamic Web TWAIN

    Dynamic Web TWAIN 使得你可以从任何兼容 TWAIN 的设备获取图象,并且上载扫描过的图象到网页服务器.是特别针对网页应用程序设计而提出的真实的网页扫描解决方案.它已经通过了 Lock ...

最新文章

  1. Java Socket 学习
  2. dataframe常用处理
  3. CSS :active 伪类
  4. 虹软java接摄像头_虹软人脸识别SDK(java+linux/window) 初试
  5. 如何给 mongodb 设置密码
  6. [vue] 怎么缓存当前打开的路由组件,缓存后想更新当前组件怎么办呢?
  7. Java基础学习总结(29)——浅谈Java中的Set、List、Map的区别
  8. c语言判断字符串合法标识符,HDU 2024 C语言合法标识符(以及一些关于输入和ctype.h的内容)...
  9. 你能想到几种方式实现数组扁平化(越多越好)
  10. C++ map的基本和高级操作
  11. 【九度OJ】题目1084:整数拆分
  12. ZOJ 1606 Count the Colors (线段数染色)
  13. Android 四大组件学习之Activity七
  14. php重定向下载地址,用PHP强制下载然后重定向
  15. 整合spring cloud云架构 - SSO单点登录之OAuth2.0登录流程
  16. 必应搜索引擎怎么了?
  17. adb wifi 连接设备
  18. 轩逸酷我音乐显示服务器错误,第14代轩逸车机系统体验:基本满足日常需求 随车流量模糊不清...
  19. 泛型的基础概念,T和?的使用方法和区别
  20. 机器人系统设计与制作:Python语言实现2.5 用Blender制作机器人的三维模型

热门文章

  1. SpringBoot整合Disconf
  2. 第五九八章 全息头盔出问题了
  3. 数字逻辑中的德·摩根定理证明
  4. 惠州学院采购JKTD-1000型铁电材料测试仪
  5. 1000个人临终前的遗言
  6. 小程序代理商是如何赚钱的?
  7. VS Code无法安装Go扩展依赖包问题解决及剖析
  8. 微前端-micro-app 使用 onresize和addEventListener区别
  9. 那些你所不知道的小技巧:硬盘照片恢复
  10. CollectionView多选