|
ScantoImage.Vb
Imports System
Imports System.Text
Imports System.Runtime.InteropServices
Imports System.Drawing
Imports System.Collections
Imports System.ComponentModel
Namespace ScanSoft
<StructLayout(LayoutKind.Sequential, Pack:=2)> Friend Class BITMAPINFOHEADER
Public biSize As Integer
Public biWidth As Integer
Public biHeight As Integer
Public biPlanes As Short
Public biBitCount As Short
Public biCompression As Integer
Public biSizeImage As Integer
Public biXPelsPerMeter As Integer
Public biYPelsPerMeter As Integer
Public biClrUsed As Integer
Public biClrImportant As Integer
End Class
Public Class scanToImage
<DllImport("gdi32.dll", ExactSpelling:=True)> Friend Shared Function SetDIBitsToDevice(ByVal hdc As IntPtr, ByVal xdst As Integer, ByVal ydst As Integer, ByVal width As Integer, ByVal height As Integer, ByVal xsrc As Integer, ByVal ysrc As Integer, ByVal start As Integer, ByVal lines As Integer, ByVal bitsptr As IntPtr, ByVal bmiptr As IntPtr, ByVal color As Integer) As Integer
End Function
<DllImport("kernel32.dll", ExactSpelling:=True)> Friend Shared Function GlobalLock(ByVal handle As IntPtr) As IntPtr
End Function
<DllImport("kernel32.dll", ExactSpelling:=True)> Friend Shared Function GlobalFree(ByVal handle As IntPtr) As IntPtr
End Function
<DllImport("kernel32.dll", CharSet:=CharSet.Auto)> Public Shared Sub OutputDebugString(ByVal outstr As String)
End Sub
Dim bmi As BITMAPINFOHEADER
Dim bmprect As Rectangle
Dim dibhand As IntPtr
Dim bmpptr As IntPtr
Dim pixptr As IntPtr
Public Sub New(ByVal dibhandp As IntPtr)
bmprect = New Rectangle(0, 0, 0, 0)
dibhand = dibhandp
bmpptr = GlobalLock(dibhand)
pixptr = GetPixelInfo(bmpptr)
End Sub
Protected Function GetPixelInfo(ByVal bmpptr As IntPtr) As IntPtr
bmi = New BITMAPINFOHEADER
Marshal.PtrToStructure(bmpptr, bmi)
bmprect.X = bmprect.Y = 0
bmprect.Width = bmi.biWidth
bmprect.Height = bmi.biHeight
If (bmi.biSizeImage = 0) Then
bmi.biSizeImage = Int((((bmi.biWidth * bmi.biBitCount) + 31) & Hex(Not (31))) / 2 ^ 3) * bmi.biHeight
End If
Dim p As Integer = bmi.biClrUsed
If ((p = 0) And (bmi.biBitCount <= 8)) Then
p = Int(1 * 2 ^ bmi.biBitCount)
End If
p = (p * 4) + bmi.biSize + CType(bmpptr.ToInt32, Integer)
Return New IntPtr(p)
End Function
Public Function ImgToBitmap(ByVal dibhandp As IntPtr) As Bitmap
bmprect = New Rectangle(0, 0, 0, 0)
dibhand = dibhandp
bmpptr = GlobalLock(dibhand)
pixptr = GetPixelInfo(bmpptr)
Dim TempBMP As Bitmap = New Bitmap(bmprect.Width, bmprect.Height)
Dim TempGrap As Graphics = Graphics.FromImage(TempBMP)
Dim hdc As IntPtr = TempGrap.GetHdc
SetDIBitsToDevice(hdc, 0, 0, bmprect.Width, bmprect.Height, 0, 0, 0, bmprect.Height, pixptr, bmpptr, 0)
TempGrap.ReleaseHdc(hdc)
TempGrap.Dispose()
GlobalFree(dibhand)
dibhand = IntPtr.Zero
Return (TempBMP)
End Function
End Class
End Namespace
|
|
|
|
|
Hello,
Guys please help. I am stuck. I've read all posts here, but I did not find solution closer than this.
I need form with two buttons. First is for select source, and second is for scanning of one picture and then to get return of that picture as image.
I solved first task by calling "tw.Select()", but second task I do not know how to accomplish.
I have tried everything.
For example:
"Dim newpic As scanToImage = New scanToImage(0)
picBox.Image = newpic.ImgToBitmap(0)"
Please help me. I have almost completed app for my office archive. Just need a class that I can call to perform scanning and then to put image from return to picturebox and store it to database.
Thank You very very very much.
I lose my mind on trying to understand this twain stuff but always more I look more get confused.
Sorry for my bad English.
Grateful forever
|
|
|
|
|
svincetic1 wrote: Guys please help. I am stuck. I've read all posts here, but I did not find solution closer than this.
I need form with two buttons. First is for select source, and second is for scanning of one picture and then to get return of that picture as image.
I solved first task by calling "tw.Select()", but second task I do not know how to accomplish.
I have tried everything.
find this code snippet
Case TwainCommand.TransferReady
Dim pics As ArrayList = tw.TransferPictures()
EndingScan()
tw.CloseSrc()
picnumber += 1
Dim i As Integer
For i = 0 To pics.Count - 1 Step 1
Dim img As IntPtr = CType(pics(i), IntPtr)
Dim newpic As PicForm = New PicForm(img)
newpic.MdiParent = Me
Dim picnum As Integer = i + 1
newpic.Text = "ScanPass" + picnumber.ToString() + "_Pic" + picnum.ToString()
and make it
Case TwainCommand.TransferReady
Dim pics As ArrayList = tw.TransferPictures()
EndingScan()
tw.CloseSrc()
picnumber += 1
Dim i As Integer
For i = 0 To pics.Count - 1 Step 1
Dim img As IntPtr = CType(pics(i), IntPtr)
Dim newpic As PicForm = New PicForm(img)
newpic.MdiParent = Me
Dim picnum As Integer = i + 1
newpic.Text = "ScanPass" + picnumber.ToString() + "_Pic" + picnum.ToString()
GdiPlusLib.Gdip.SaveDIBAs((newpic.Text & ".bmp"), PicForm.bmpptr, PicForm.pixptr)
newpic.show()
hope i helped
|
|
|
|
|
how can I turn it into a pdf rather than an image . Also is there a way to call the auto select of the scanner software rather than opening it's GUI. I have a problem eliminating the extra white edges and i keep trying to crop the image after it's created but i can't include all colors since some pixels turn into shades of white and gray . PLEASE HELPP - I need to complete this project ASAP
|
|
|
|
|
I want to be able to scan from an HTML page using this .net code, but it seems to need some .net permissions in the .net configuration console on the client machines.
¿Is there a way to change this code in order to generate an .ocx file?
|
|
|
|
|
Hi
I've some problems with configure the scan area. I use code wrote by cmchuan
[StructLayout(LayoutKind.Sequential, Pack = 2)]
internal struct TwFrame {
public int Left;
public int Top;
public int Right;
public int Bottom;
}
//set the scan area using a call like this
TwImageLayout layout = new TwImageLayout();
//Get the default layout
TwRC rc = DSilayout(appid, srcds, TwDG.Image, TwDAT.ImageLayout, TwMSG.Get, Layout);
Layout.Frame.Top = 0; //Set the top margin
Layout.Frame.Left = 0; //Set left margin
Layout.Frame.Right = 0; //Set right margin
Layout.Frame.Bottom = 0; //Set bottom margin
Layout.FrameNumber = 1;
Layout.PageNumber = 1;
Layout.DocumentNumber = 1;
rc = DSilayout(appid, srcds, TwDG.Image, TwDAT.ImageLayout, TwMSG.Set, layout);
TwStatus s = new TwStatus();
rc = DSstatus(appid, srcds, TwDG.Control, TwDAT.Status, TwMSG.Get, s);
if( rc != TwRC.Success )
{
CloseSrc();
return false;
}
I change margins but it's always scan all area. What should i do to scan small fragment?
|
|
|
|
|
I have post full VB.NET code in above.
|
|
|
|
|
Could You mark fragment responsible for selecting and setuping area?
Regards
alien250
|
|
|
|
|
One the things that I found difficult about developing my Twain application was that I often could not get the excellent code posted here to work. Often I would get 'success' returned from trying to set the dpi or the bitdepth or other capabilities but the effect was zero.
I have noticed that many others have the same problem. I finally went to the web site (http://www.twain.org/docs/Spec1_9_197.pdf) for Twain and actually read the document setting out the standards.
I was developing an Optical Mark Recognition application for which I had written an algorithm and was adding the Twain standard so I could scan and read forms. I needed to set the DPI, the bit depth and the resolution and tried using the code here to do so. I could scan but was unsuccessful in setting the different elements. The application would return 'success' but the final scan woull not reflect the changes.
It turns out that the capabilities of the scanner should be set in sequence according to the standard. For example, you have to set the pixel type before the bitdepth before the x and y resolution. This is strictly enforced on many systems, particularly on high end scanners like the Fujitsu I am using.
A system will return 'success' because you have made the right call but do nothing because you have not set the caps in the right order.
Following the changes in line with the Twain standard I have just successfully scanned and processed 500 forms consisting of a total of 3,000 pages.
Hopefully this will work for others.
Alan
'Nemo me impune Lacessit'
|
|
|
|
|
hello there, i saw ur article, it seems u r the person that can help.
i m making an application to scan multiple forms, 100 pages per minute using twain in black & white.
so far the application made scans only 1 page, i want to scan and save multiple pages to the directory i want. can u guide me what steps shall i perform in order to achieve that,
|
|
|
|
|
Did anyone find a way how to scan image directly without showing user interface of the source? Is that property related with CAP_AUTOSCAN capability, if so how i will set this capability to true using the interface above written by NetMaster.
I need ur helps...
|
|
|
|
|
The following is a function that populates an internal ArrayList(_Sources) of TwIdentity and returns number of sources found. You can then use this ArrayList to select the source of choice.
public int GetDataSources()
{
//ArrayList ds = new ArrayList();
_Sources = new ArrayList();
TwIdentity scanner = new TwIdentity();
TwIdentity defScanner = new TwIdentity();
CloseSrc();
if (appid.Id == IntPtr.Zero)
{
Init(hwnd);
if (appid.Id == IntPtr.Zero)
return _Sources.Count;
}
//Get the default
if (DSMident(appid, IntPtr.Zero, TwDG.Control, TwDAT.Identity, TwMSG.GetDefault, defScanner) != TwRC.Success)
return _Sources.Count;
if (DSMident(appid, IntPtr.Zero, TwDG.Control, TwDAT.Identity, TwMSG.GetFirst, scanner) == TwRC.Success)
{
//Select default data source
if (scanner.Id == defScanner.Id)
_SelectedSource = _Sources.Count;
_Sources.Add(scanner);
//Loop through and find all sources and add to array
scanner = new TwIdentity();
while (DSMident(appid, IntPtr.Zero, TwDG.Control, TwDAT.Identity, TwMSG.GetNext, scanner) == TwRC.Success)
{
//Select default data source
if (scanner.Id == defScanner.Id)
_SelectedSource = _Sources.Count;
_Sources.Add(scanner);
scanner = new TwIdentity();
}
}
return _Sources.Count;
}
To Select the desired Datasource you could to do the following:
public bool SelectSource(int SourceIndex)
{
//Select correct datasource
srcds = (TwIdentity)_Sources[SourceIndex];
//Close Datasource
CloseSrc();
//Invoke set command of datasource
rc = DSMident(appid, IntPtr.Zero, TwDG.Control, TwDAT.Identity, TwMSG.Set, srcds);
if (rc != TwRC.Success)
return false;
}
Hope that helps
|
|
|
|
|
hi ,,
I want to change the way to select source device and for that i have added a combobox in my application and to work with this i found your code which may help me but when i implemented it give some error.
1.what is "SelectedSource" in your code
2.SelectSource function error--not all code paths return a value.
but i have done some little modification
which give only one error that is "SelectedSource"
please help me which place i have to add this function and further procedure for implement this feature.
please help me
|
|
|
|
|
hello, i have download the code and learn many thing for it.
i want to change the scan UI in the code, but don't know TWAIN exactly.
i find the Acquire sub is very important, and some structure or Inherits some function from TWAIN will finish it.
thank you very much for your help
MSN: zhangjiuheng@hotmail.com
web: www.hengjiu.cc
MSN: zhangjiuheng@hotmail.com
web: www.hengjiu.cc
|
|
|
|
|
thank you for this great article
i want to handle scanner event like paperjam in my application but i dont know how to do this i tried some way like setting deviceevent capability
but it doesnt work and returnz sfailure message when im setting
please some body help me
|
|
|
|
|
Hi,
Is there any way to suppress the user interfaces from twain driver?
First, I want to handle the error while scanning, opening data source or closing data source. Is this can do by not showing UI?
Second is, while scanning, scanning process progressbar is pops up. Can we invisible this UI?
Urgent help needed!!
bg
|
|
|
|
|
|
|
thanks for the reference, I've moved that code to http://www.codeplex.com/openTwain as gotdotnet is closing.
|
|
|
|
|
This class extends what m@u had posted.
[SecurityPermission(SecurityAction.Assert, Flags=SecurityPermissionFlag.UnmanagedCode)]
[FileIOPermission(SecurityAction.Assert, Unrestricted=true)]
public class TwainManager : NativeWindow, IDisposable
{
[DllImport("kernel32.dll", ExactSpelling = true)]
internal static extern IntPtr GlobalLock(IntPtr handle);
[DllImport("kernel32.dll", ExactSpelling = true)]
internal static extern IntPtr GlobalFree(IntPtr handle);
private Twain _scanner;
private List<fileinfo> _files;
private TwainConfig _scannerConfig;
private bool _ready = false;
private byte[] _binaryImage;
public byte[] BinaryImage
{
get
{
return _binaryImage;
}
}
public bool Ready
{
get
{
return _ready;
}
}
public TwainManager()
{
_scannerConfig = TwainConfig.GetConfig();
CreateParams cp = new CreateParams();
cp.Parent = new IntPtr(-3);
//cp.ClassName = "Window";
cp.Caption = "";
this.CreateHandle(cp);
_scanner = new Twain();
_scanner.Init(Handle);
if (string.IsNullOrEmpty(_scannerConfig.ActiveDevice))
_scannerConfig.ActiveDevice = _scanner.ActiveDeviceName;
else
{
if (!_scanner.SelectByName(_scannerConfig.ActiveDevice))
{
_scanner.Init(Handle);
_scannerConfig.ActiveDevice = _scanner.ActiveDeviceName;
}
}
_scannerConfig.Save();
}
public bool SelectScanner()
{
_ready = _scanner.Select();
if (_ready)
{
_scannerConfig.ActiveDevice = _scanner.ActiveDeviceName;
_scannerConfig.Save();
}
return _ready;
}
private void AddScan()
{
if (_scanner.Acquire() != TwRC.Success)
{
EndScanning();
throw new Exception("Document could not be scanned");
}
}
public void Scan()
{
if (_ready)
{
_files = new List<fileinfo>();
_binaryImage = null;
if (_scanner.Acquire() != TwRC.Success)
{
EndScanning();
throw new Exception("Document could not be scanned");
}
}
else
{
if (SelectScanner())
Scan();
}
}
private void AnalyzeImage(IntPtr pic)
{
IntPtr bmpPtr = IntPtr.Zero;
IntPtr pixPtr = IntPtr.Zero;
string tempFile = Path.GetTempFileName().Replace(".tmp", ".tif");
try
{
if (pic.Equals(IntPtr.Zero))
{
EndScanning();
_scanner.CloseSrc();
return;
}
bmpPtr = GlobalLock(pic);
pixPtr = GetPixelInfo(bmpPtr);
GdiPlusLib.SaveDIBAs(tempFile, bmpPtr, pixPtr);
FileInfo fi = new FileInfo(tempFile);
if (fi.Exists)
_files.Add(fi);
}
catch (Exception ex)
{
throw ex;
}
}
private IntPtr GetPixelInfo(IntPtr bmpptr)
{
Rectangle bmprect = default(Rectangle);
BITMAPINFOHEADER bmi = new BITMAPINFOHEADER();
Marshal.PtrToStructure(bmpptr, bmi);
bmprect.X = bmprect.Y = 0;
bmprect.Width = bmi.biWidth;
bmprect.Height = bmi.biHeight;
if (bmi.biSizeImage == 0)
bmi.biSizeImage = ((((bmi.biWidth * bmi.biBitCount) + 31) & ~31) >> 3) * bmi.biHeight;
int p = bmi.biClrUsed;
if ((p == 0) && (bmi.biBitCount <= 8))
p = 1 << bmi.biBitCount;
p = (p * 4) + bmi.biSize + (int)bmpptr;
return (IntPtr)p;
}
protected override void WndProc(ref Message m)
{
if (_scanner != null)
{
TwainCommand cmd = _scanner.PassMessage(ref m);
if (cmd == TwainCommand.Not)
return;
switch (cmd)
{
case TwainCommand.CloseRequest:
EndScanning();
_scanner.CloseSrc();
break;
case TwainCommand.CloseOk:
EndScanning();
_scanner.CloseSrc();
break;
case TwainCommand.DeviceEvent:
break;
case TwainCommand.TransferReady:
ScanImage();
break;
}
}
base.WndProc(ref m);
}
private void ScanImage()
{
IntPtr pic = IntPtr.Zero;
try
{
pic = _scanner.TransferPicture();
AnalyzeImage(pic);
if (_scanner.HasMorePages)
ScanImage();
else
{
_scanner.CloseSrc();
if (DialogResult.No == MessageBox.Show(
"Would you like to add more pages?", "Continue Scanning?",
MessageBoxButtons.YesNo,
MessageBoxIcon.Question))
{
BuildTiff();
}
else
{
AddScan();
}
}
}
catch (Exception ex)
{
throw ex;
}
}
private void EndScanning()
{
}
private EncoderParameters GetEncoderParameters(EncoderValue SaveFlag, EncoderValue Compression)
{
EncoderParameters enParams = new EncoderParameters(2);
enParams.Param[0] = new EncoderParameter(System.Drawing.Imaging.Encoder.SaveFlag, (long)SaveFlag);
enParams.Param[1] = new EncoderParameter(System.Drawing.Imaging.Encoder.Compression, (long)Compression);
return enParams;
}
private ImageCodecInfo GetEncoderInfo(String mimeType)
{
ImageCodecInfo[] encoders;
encoders = ImageCodecInfo.GetImageEncoders();
for (int i = 0; i < encoders.Length; i++)
{
if (String.Compare(mimeType, encoders[i].MimeType, true) == 0)
{
return encoders[i];
}
}
return null;
}
private void BuildTiff()
{
string tiffPath = Path.GetTempFileName().Replace(".tmp", ".tif");
Bitmap b = null;
if (_files.Count > 0)
{
b = (Bitmap)Bitmap.FromFile(_files[0].FullName);
if (b.PixelFormat == PixelFormat.Format1bppIndexed)
b.Save(tiffPath, GetEncoderInfo("image/tiff"), GetEncoderParameters(EncoderValue.MultiFrame, EncoderValue.CompressionCCITT4));
else
b.Save(tiffPath, GetEncoderInfo("image/tiff"), GetEncoderParameters(EncoderValue.MultiFrame, EncoderValue.CompressionLZW));
for (int i = 1; i < _files.Count; i++)
{
Image image = Image.FromFile(_files[i].FullName);
if (image.PixelFormat == PixelFormat.Format1bppIndexed)
b.SaveAdd(image, GetEncoderParameters(EncoderValue.FrameDimensionPage, EncoderValue.CompressionCCITT4));
else
b.SaveAdd(image, GetEncoderParameters(EncoderValue.FrameDimensionPage, EncoderValue.CompressionLZW));
}
b.Dispose();
b = null;
}
else
throw new Exception("No document scanned.");
FileStream fs = new FileStream(tiffPath, FileMode.Open, FileAccess.Read);
_binaryImage = new Byte[fs.Length];
fs.Read(_binaryImage, 0, (int)fs.Length);
fs.Close();
fs.Dispose();
fs = null;
b = null;
if (_binaryImage == null || _binaryImage.Length < 1)
throw new Exception("No document could be scanned.");
}
#region IDisposable Member
public void Dispose()
{
try
{
if(_scanner != null)
_scanner.CloseSrc();
}
catch { }
_scanner = null;
}
#endregion
}
Breaking your theories
Sully
|
|
|
|
|
I have created a Class Library. I scan and save the image to a file. Then I read from the file to display in a picturebox. When I use the library in a form, it scans, saves image to a file and displays the image in the picturebox.
When I use the DLL (library) in a HTML page, it scans, but the (user specified) file is not created. Instead, it creates a file in TEMP called TwainDS.tif and then the program hangs.
Anyone have any idea why?
|
|
|
|
|
Interesting!
I am having a very similar scenario. My problem is that when I run the code from an ActiveX wrapper, the scanner never receives the command to scan. In other words, the PreFilterMessage never catches any message! I even tried with WndProc method, but no success. But when the code is ran from directly (running the .exe or debugging within Visual Studio), all works perfectly!
I assume it has something to do with the fact that the parent process (owner) is IE8... I am not sure...
Any ideas?
|
|
|
|
|
Hi,I have the same issue,any ideas for me?
Thanks advanced.
|
|
|
|
|
Hi!
I have a problem with this code(Twain Manager for .NET ActiveX)
What is TwainConfig?
Could you public or mail complete source of ActiveX decision?
Thanks.
Vasig
|
|
|
|
|
Hi,
I want to list available scanners in my own listbox, not in twain's gui. Becuse i want to use alias names of scanners.
Is there anyway to get scanner list shown in the SelctSource() dialog box?
Thanks.
bg
|
|
|
|
|