winform摄像头拍照 C#利用摄像头拍照

这是我的第一篇博文,决定以后每个程序都要记录下来,方便以后查阅!

本人小菜一名,本程序也是查阅了网上各位前辈的博客和百度知道所整理出来的一个小程序。

第一次写有点不知道从何写起,先贴一张程序图吧。

winform摄像头拍照   C#利用摄像头拍照

程序功能,导入Excel用户模板,模板图如下

winform摄像头拍照   C#利用摄像头拍照

通过身份证查找用户,根据准考证好生成图片名。

下面开始进入程序

利用摄像头拍照需要调用win32的avicap32.dll和user32.dll。

avicap32.dll:Windows NT 4.0  avicap32.dll是Windows API应用程序接口相关模块,用于对摄像头和其它视频硬件进行AⅥ电影和视频的截取。

user32.dll:user32.dll是Windows用户界面相关应用程序接口,用于包括Windows处理,基本用户界面等特性,如创建窗口和发送消息。

代码部分

  private int hHwnd = ;
private const int port = ;
private DataTable dtExcel;
private string imgName;
public CameraForm()
{
InitializeComponent();
}
public struct videohdr_tag
{
public byte[] lpData;
public int dwBufferLength;
public int dwBytesUsed;
public int dwTimeCaptured;
public int dwUser;
public int dwFlags;
public int[] dwReserved; }
  ///   <summary>
/// 必需的设计器变量。
/// </summary> [DllImport("avicap32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
public static extern int capCreateCaptureWindowA([MarshalAs(UnmanagedType.VBByRefStr)] ref string lpszWindowName, int dwStyle, int x, int y, int nWidth, short nHeight, int hWndParent, int nID);
[DllImport("avicap32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
public static extern bool capGetDriverDescriptionA(short wDriver, [MarshalAs(UnmanagedType.VBByRefStr)] ref string lpszName, int cbName, [MarshalAs(UnmanagedType.VBByRefStr)] ref string lpszVer, int cbVer);
[DllImport("user32", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
public static extern bool DestroyWindow(int hndw);
[DllImport("user32", EntryPoint = "SendMessageA", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
public static extern int SendMessage(int hwnd, int wMsg, int wParam, [MarshalAs(UnmanagedType.AsAny)] object lParam);
[DllImport("user32", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
public static extern int SetWindowPos(int hwnd, int hWndInsertAfter, int x, int y, int cx, int cy, int wFlags);
[DllImport("vfw32.dll")]
public static extern string capVideoStreamCallback(int hwnd, videohdr_tag videohdr_tag);
[DllImport("vicap32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]

接下来就要开始要调用摄像头打开

         /// <summary>
/// 开启摄像头
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button1_Click(object sender, EventArgs e)
{
if (hHwnd == )
this.OpenCapture();
else
MessageBox.Show("摄像头已经开启!","提示");
}
      /// <summary>
/// 打开摄像头
/// </summary>
private void OpenCapture()
{
int intWidth = this.panel1.Width;
int intHeight = this.panel1.Height;
int intDevice = ;
string refDevice = intDevice.ToString();
//创建视频窗口并得到句柄
hHwnd = CameraForm.capCreateCaptureWindowA(ref refDevice, , , , , , this.panel1.Handle.ToInt32(), );
if (CameraForm.SendMessage(hHwnd, 0x40a, intDevice, ) > )
{
CameraForm.SendMessage(this.hHwnd, 0x435, -, );
CameraForm.SendMessage(this.hHwnd, 0x434, 0x42, );
CameraForm.SendMessage(this.hHwnd, 0x432, -, );
CameraForm.SetWindowPos(this.hHwnd, , , , intWidth, intHeight, );
}
else
{
CameraForm.DestroyWindow(this.hHwnd);
}
}

capCreateCaptureWindowA的作用是创建一个视频窗口,摄像头捕捉到的视频图像在此窗口内显示,函数返回值就是代表此窗口的句柄。

接下来关闭摄像头

         /// <summary>
///关闭摄像头
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button2_Click(object sender, EventArgs e)
{
//停止视频注销视频句柄
CameraForm.SendMessage(this.hHwnd, 0x40b, , );
CameraForm.DestroyWindow(this.hHwnd);
hHwnd = ;
}

然后是拍照

   /// <summary>
/// 截图
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button3_Click(object sender, EventArgs e)
{
if (hHwnd == )
{
MessageBox.Show("请先开启摄像头!","提示");
return;
}
Preview dlg = new Preview();
try
{
CameraForm.SendMessage(this.hHwnd, 0x41e, , );
IDataObject obj1 = Clipboard.GetDataObject();
if (obj1.GetDataPresent(typeof(Bitmap)))
{
Rectangle rect = new Rectangle(, , , );
Image image1 = (Image)obj1.GetData(typeof(Bitmap));
Bitmap bit = (Bitmap)image1;
//MessageBox.Show( bit.Height.ToString());
//MessageBox.Show(bit.Width.ToString()); bit = bit.Clone(new Rectangle(, , , ), System.Drawing.Imaging.PixelFormat.Format32bppArgb);
bit = (Bitmap)GetThumbnailImage(bit, , );
dlg.Img = bit;
dlg.ImgName = imgName;
}
}
catch
{
}
dlg.ShowDialog();
}
  /// <summary>
/// 将某个图像生成指定尺寸的图像缩放图
/// </summary>
/// <param name="myBitmap">原图像</param>
/// <param name="width">新的图像宽</param>
/// <param name="height">新的图像高</param>
/// <returns></returns>
private Image GetThumbnailImage(Bitmap myBitmap, int width, int height)
{ //生成图像的缩放图
Image.GetThumbnailImageAbort myCallback =
new Image.GetThumbnailImageAbort(ThumbnailCallback); Image myThumbnail = myBitmap.GetThumbnailImage(
width, height, myCallback, IntPtr.Zero);
//this.pictureBox1.Image = myThumbnail; return myThumbnail;
}
public bool ThumbnailCallback()
{
return false;
}

最后是保存截图

  private void button1_Click(object sender, EventArgs e)
{
SaveFileDialog SaveFileDialog1 = new SaveFileDialog();
SaveFileDialog1.FileName = imgName;
SaveFileDialog1.Filter = "Image Files(*.JPG;*.GIF)|*.JPG;*.GIF|All files (*.*)|*.*";
if (SaveFileDialog1.ShowDialog() == DialogResult.OK)
{
img.Save(SaveFileDialog1.FileName, ImageFormat.Bmp);
this.Close();
}
}

下面是操作Excel

首先判断Excel版本

  /// <summary>
/// 检测Excel版本信息
/// </summary>
/// <returns></returns>
public static double JongCheckExcelVer()
{
Type objExcelType = Type.GetTypeFromProgID("Excel.Application");
if (objExcelType == null)
{
return ;
}
object objApp = Activator.CreateInstance(objExcelType);
if (objApp == null)
{
return ;
}
object objVer = objApp.GetType().InvokeMember("Version", BindingFlags.GetProperty, null, objApp, null);
double iVer = Convert.ToDouble(objVer.ToString());
objVer = null;
objApp = null;
objExcelType = null;
GC.Collect();
return iVer;
} public static String JongGetExcelVerStr()
{
String s1;
double excelver;
excelver = JongCheckExcelVer();// ExistsExcelRegedit();
s1 = " Office ";
if (excelver == )
{
MessageBox.Show("無法識別Excel的版本", "錯誤", MessageBoxButtons.OK, MessageBoxIcon.Information);
s1 = "無法識別 office 版本";
}
else if (excelver >= ) s1 += "2010或以上";
else if (excelver >= ) s1 += "";
else if (excelver >= ) s1 += "";
else if (excelver >= ) s1 += "XP";
else if (excelver >= ) s1 += "";
else if (excelver >= ) s1 += "";
else if (excelver >= ) s1 += ""; return s1;
}

接着是导入到DataTable

   /// <summary>
/// 导入excel到datata
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button5_Click(object sender, EventArgs e)
{
System.Windows.Forms.OpenFileDialog fd = new OpenFileDialog();
try
{
if (fd.ShowDialog() == DialogResult.OK)
{
DataSet ds = new DataSet(); string strConn = ""; if (JongCheckExcelVer() >= )
{
strConn = "Provider=Microsoft.Ace.OLEDB.12.0;" + "Data Source=" + fd.FileName + ";" + "Extended Properties='Excel 12.0;HDR=Yes;IMEX=1;'";
}
else if (JongCheckExcelVer() >= )
{
strConn = "Provider=Microsoft.Jet.OLEDB.4.0;" + "Data Source=" + fd.FileName + ";" + "Extended Properties='Excel 8.0;HDR=Yes;IMEX=1;'";
}
else if (JongCheckExcelVer() == )
{
MessageBox.Show("无法识别Excel的版本", "错误", MessageBoxButtons.OK, MessageBoxIcon.Information);
} OleDbConnection conn = new OleDbConnection(strConn);
conn.Open();
string strExcel = "";
//string strSheet = "";
string strSheetName = "";
OleDbDataAdapter myCommand = null;
DataTable dtExcelName = conn.GetOleDbSchemaTable(System.Data.OleDb.OleDbSchemaGuid.Tables, null); strSheetName = dtExcelName.Rows[][].ToString(); strExcel = string.Format("select * from [{0}$]", "Sheet1"); //Bicycle为excel中工作薄
myCommand = new OleDbDataAdapter(strExcel, strConn);
myCommand.Fill(ds, "Sheet1");
System.Data.DataTable dt = new System.Data.DataTable();
dt = ds.Tables[];
dtExcel = dt;
MessageBox.Show("导入成功!","提示");
}
else
MessageBox.Show("导入失败!","提示");
}
catch
{
MessageBox.Show("请选择相应的excel文件", "警告");
}
}

本程序代码大部分来之互联网,如有雷同,纯属必然。 可以加群讨论,群里有各种源码,资料: 215269573

上一篇:不要着急改代码,先想想--centos 6.8下编译安装tmux


下一篇:【spring 5】AOP:spring中对于AOP的的实现