C# Activex调用USB摄像头--附带源码
前言
最近在整理一些自己写过的东西,也算是重新熟悉一下并且优化一下吧。
需求:获取本地USB摄像头视频显示,并且获取图片数据给底层做人脸识别。
记得当时直接采用H5已经做好了,调试好了。。。。结果放上去使用发现必须需要证书才可以,
然后因为某些原因(没办法自己写一个ssl证书)只能重写了一个之前使用Activex做的USB控件。
H5调用USB摄像头参考:https://segmentfault.com/a/1190000011793960
开发
闲话:DLL缺少搜索找不到,推荐找dll https://www.zhaodll.com/
引用:AForge.dll,AForge.Video.DirectShow.dll,AForge.Video.dll
首先创建一个用户控件>放入一个pictureBox1>放入一个lable
将用户控件调成灰色(明显。。。)大小随便后面可以调整,pictureBox1在父容器停靠,lab用于错误提示,效果如下:

添加一个 IObjectSafety 接口。
/*
IObjectSafety 接口用于通知浏览器:“浏览器,我是安全的”
*/
[ComImport, Guid("887FC3C3-A970-4A36-92EF-D4EB31541C40")]
[InterfaceTypeAttribute(ComInterfaceType.InterfaceIsIUnknown)]
public interface IObjectSafety
{
[PreserveSig]
int GetInterfaceSafetyOptions(ref Guid riid, [MarshalAs(UnmanagedType.U4)] ref int pdwSupportedOptions, [MarshalAs(UnmanagedType.U4)] ref int pdwEnabledOptions);
[PreserveSig()]
int SetInterfaceSafetyOptions(ref Guid riid, [MarshalAs(UnmanagedType.U4)] int dwOptionSetMask, [MarshalAs(UnmanagedType.U4)] int dwEnabledOptions);
}
添加一个基础类,用于数据返回处理。
public class USBInfo
{
/// <summary>
/// 返回消息
/// </summary>
public class ReturnMessage
{
public bool result { get; set; }//结果
public string message { get; set; }//消息
}
/// <summary>
/// 返回视频分辨率
/// </summary>
public class GetResolvingPower
{
public int index { get; set; }//下标 -1(不存在摄像头),-2(发生意外错误)
public VideoCapabilities Capabilities { get; set; }//信息
public string error { get; set; }//错误
}
}
用户控件中实现:
/// <summary>
/// 继承IObjectSafety
/// </summary>
[ProgId("USBCamera")]
[ClassInterface(ClassInterfaceType.AutoDual)]
[Guid("887FC3C3-A970-4A36-92EF-D4EB31541C40")]
[ComVisible(true)]
public partial class USBCamera : UserControl, IObjectSafety
{
FilterInfoCollection videoDevices;//设备
VideoCaptureDevice videoSource;//设备信息
MemoryStream frameData;//图片数据
static Thread th_usb;//监听线程
;//多久检查一次
static int index_usb;//第一次打开选择的分辨率
public USBCamera()
{
InitializeComponent();
frameData = new MemoryStream();
}
//句柄销毁
protected override void OnHandleDestroyed(EventArgs e)
{
Stop();
base.OnHandleDestroyed(e);
}
/// <summary>
/// 设置窗体相关大小
/// </summary>
/// <param name="width">宽</param>
/// <param name="height">高</param>
/// <param name="fontformat">字体熟悉</param>
/// <param name="fontsize">字体大小</param>
/// <param name="promptInfo">提示信息</param>
/// <param name="sleep">监听间隔</param>
public void FormSize(int width, int height, string fontFormat, int fontSize, string promptInfo, int sleep)
{
this.Size = new Size(width, height);//窗体大小
this.lab_Prompt.Font = new Font(fontFormat, fontSize, FontStyle.Bold); //字体熟悉
this.lab_Prompt.ForeColor = Color.Red;//字体颜色
this.lab_Prompt.Text = promptInfo;//提示信息
//设置字体高宽
);
);
this.lab_Prompt.Location = new System.Drawing.Point(lab_width, lab_height);
sl_usb = sleep;//监听时间
}
/// <summary>
/// 打开视频
/// </summary>
/// <param name="index">USB可用分辨率数组下标</param>
public void Start(int index)
{
index_usb = index;
if (videoSource != null)
{
videoSource.NewFrame -= new NewFrameEventHandler(video_NewFrame);
videoSource.SignalToStop();
videoSource = null;
}
// 创建视频源
Trace.WriteLine("usbcamera have cameras" + videoDevices.Count.ToString());
videoSource = ].MonikerString);
// 设置新帧事件处理程序
videoSource.NewFrame += new NewFrameEventHandler(video_NewFrame);
// 启动视频源
try
{
videoSource.VideoResolution = this.videoSource.VideoCapabilities[index];
videoSource.Start();
if (th_usb == null || !th_usb.IsAlive)
{
th_usb = new Thread(GetUSB);
th_usb.Start();
}
this.lab_Prompt.Text = null;
}
catch (Exception ex)
{
Trace.WriteLine("usbcamera Start()遇到错误" + ex.Message + ex.StackTrace);
}
}
/// <summary>
/// 停止视频
/// </summary>
public void Stop()
{
if (videoSource != null)
{
videoSource.NewFrame -= new NewFrameEventHandler(video_NewFrame);
//当不再需要捕捉时,发出停止的信号
videoSource.SignalToStop();
videoSource = null;
Trace.WriteLine("usbcamera stop()....");
if (pictureBox1.Image != null)//清空控件
{
pictureBox1.Image.Dispose();
pictureBox1.Image = null;
}
this.lab_Prompt.Text = null;
}
}
/// <summary>
/// 视频帧获取回调
/// </summary>
/// <param name="sender"></param>
/// <param name="eventArgs"></param>
private void video_NewFrame(object sender, NewFrameEventArgs eventArgs)
{
try
{
lab_Prompt.Text = null;
Trace.WriteLine("usbcamera video_NewFrame()....");
eventArgs.Frame.Save(frameData, ImageFormat.Bmp);
if (pictureBox1.InvokeRequired)
{
//写入数据
pictureBox1.BeginInvoke((MethodInvoker)delegate ()
{
try
{
if (pictureBox1.Image != null)
{
Image img = pictureBox1.Image;
img.Dispose();
}
pictureBox1.Image = new Bitmap(frameData);
}
catch (Exception ex)
{
if (pictureBox1.Image != null)
{
pictureBox1.Image.Dispose();
pictureBox1.Image = null;
}
}
});
}
else
{
pictureBox1.Image = new Bitmap(frameData);
}
}
catch (Exception ex)
{
if (pictureBox1.Image != null)
{
pictureBox1.Image.Dispose();
pictureBox1.Image = null;
}
}
}
/// <summary>
/// 获取USB(USB连接不正常),不存在给出提示,存在重新Start
/// </summary>
public void GetUSB()
{
while (true)
{
var usb = new FilterInfoCollection(FilterCategory.VideoInputDevice);
)
{
Stop();
lab_Prompt.BeginInvoke((MethodInvoker)delegate ()
{
this.lab_Prompt.Text = "未连接上摄像头,检查连接是否正常。";
});
}
else
{
//重新连接上 执行Start
if (!string.IsNullOrWhiteSpace(this.lab_Prompt.Text))
{
Start(index_usb);
}
}
Thread.Sleep(sl_usb);
}
}
#region USB视频操作
/// <summary>
/// 测试Activex是否可用
/// </summary>
/// <returns>true/false</returns>
public bool Connection()
{
return true;
}
/// <summary>
/// 获取截图
/// </summary>
/// <returns></returns>
public ReturnMessage GetScreenshots()
{
try
{
return new ReturnMessage()
{
result = true,
message = Convert.ToBase64String(frameData.ToArray())
};
}
catch (Exception EX)
{
Trace.WriteLine("usbcamera GetImage() 错误...." + EX.Message + EX.StackTrace);
return new ReturnMessage() { result = false, message = EX.Message + " " + EX.StackTrace };
}
}
/// <summary>
/// 获取USB摄像头分辨率
/// </summary>
/// <returns>json(GetResolvingPower)</returns>
public string GetResolution()
{
var list = new List<GetResolvingPower>();
try
{
videoDevices = new FilterInfoCollection(FilterCategory.VideoInputDevice);
)
{
videoSource = ].MonikerString);
; i < videoSource.VideoCapabilities.Length; i++)
{
list.Add(new GetResolvingPower
{
index = i,
Capabilities = videoSource.VideoCapabilities[i],
});
}
}
else
{
list.Add(new GetResolvingPower
{
index = -,
Capabilities = null,
error = "不存在USB摄像头",
});
}
}
catch (Exception EX)
{
list.Add(new GetResolvingPower
{
index = -,
Capabilities = null,
error = EX.Source + " " + EX.Message
});
}
return Newtonsoft.Json.JsonConvert.SerializeObject(list);
}
#endregion
#region IObjectSafety 成员直接复制
private const string _IID_IDispatch = "{00020400-0000-0000-C000-000000000046}";
private const string _IID_IDispatchEx = "{a6ef9860-c720-11d0-9337-00a0c90dcaa9}";
private const string _IID_IPersistStorage = "{0000010A-0000-0000-C000-000000000046}";
private const string _IID_IPersistStream = "{00000109-0000-0000-C000-000000000046}";
private const string _IID_IPersistPropertyBag = "{37D84F60-42CB-11CE-8135-00AA004BB851}";
private const int INTERFACESAFE_FOR_UNTRUSTED_CALLER = 0x00000001;
private const int INTERFACESAFE_FOR_UNTRUSTED_DATA = 0x00000002;
;
private const int E_FAIL = unchecked((int)0x80004005);
private const int E_NOINTERFACE = unchecked((int)0x80004002);
private bool _fSafeForScripting = true;
private bool _fSafeForInitializing = true;
public int GetInterfaceSafetyOptions(ref Guid riid, ref int pdwSupportedOptions, ref int pdwEnabledOptions)
{
int Rslt = E_FAIL;
string strGUID = riid.ToString("B");
pdwSupportedOptions = INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA;
switch (strGUID)
{
case _IID_IDispatch:
case _IID_IDispatchEx:
Rslt = S_OK;
pdwEnabledOptions = ;
if (_fSafeForScripting == true)
pdwEnabledOptions = INTERFACESAFE_FOR_UNTRUSTED_CALLER;
break;
case _IID_IPersistStorage:
case _IID_IPersistStream:
case _IID_IPersistPropertyBag:
Rslt = S_OK;
pdwEnabledOptions = ;
if (_fSafeForInitializing == true)
pdwEnabledOptions = INTERFACESAFE_FOR_UNTRUSTED_DATA;
break;
default:
Rslt = E_NOINTERFACE;
break;
}
return Rslt;
}
public int SetInterfaceSafetyOptions(ref Guid riid, int dwOptionSetMask, int dwEnabledOptions)
{
int Rslt = E_FAIL;
string strGUID = riid.ToString("B");
switch (strGUID)
{
case _IID_IDispatch:
case _IID_IDispatchEx:
if (((dwEnabledOptions & dwOptionSetMask) == INTERFACESAFE_FOR_UNTRUSTED_CALLER) && (_fSafeForScripting == true))
Rslt = S_OK;
break;
case _IID_IPersistStorage:
case _IID_IPersistStream:
case _IID_IPersistPropertyBag:
if (((dwEnabledOptions & dwOptionSetMask) == INTERFACESAFE_FOR_UNTRUSTED_DATA) && (_fSafeForInitializing == true))
Rslt = S_OK;
break;
default:
Rslt = E_NOINTERFACE;
break;
}
return Rslt;
}
#endregion
}
用于测试的HTML页面代码:
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title></title>
<meta charset="utf-8" />
<script>
//初始化 设置USB控件大小和显示
var array = new Array();
window.onload = function () {
var USBCamera = document.getElementById("USBCamera");
USBCamera.Stop(); //停止视频
array = JSON.parse(document.getElementById("USBCamera").GetResolution());//获取USB分辨率
/*
判断获取数据是否成功(USB连接是否正常)
大于1为成功,摄像头基本有多个分辨率
也可以判断返回值中index是否为负数
获取到的分辨率最大的为第一个下标为:0
*/
if (array.length > 1) {
//分辨率下标 宽 高
var index = 0, width = 0, height = 0;
for (var i = 0; i < array.length; i++) {
var size = array[i].Capabilities.FrameSize;
if (size.Width > width && size.Height > height) {
index = i;
width = size.Width;
height = size.Height;
}
}
//此处我获取分辨率的最大值赋值,实际可以更改
USBCamera.FormSize(width, height, "宋体", 9, "", 10000);//初始化
USBCamera.Start(index);//打开摄像头
} else {
if (array[0].index == -1) { //-1未连接
USBCamera.FromSize(400, 400, "宋体", 9, "摄像头未连接", 10000);
} if (array[0].index == -2) { //-2 获取是发生错误
USBCamera.FromSize(400, 400, "宋体", 9, "摄像头连接有异常,请联系管理人员!", 10000);
}
}
}
//关闭时执行Stop
window.BeforeUnloadEvent = function () {
document.getElementById("USBCamera").Stop();
}
//测试连接
function TestConnection() {
alert(document.getElementById("USBCamera").Connection());
}
//打开摄像头
function StartCamera() {
document.getElementById("USBCamera").Start(0);
}
//关闭摄像头
function StopCamera() {
document.getElementById("USBCamera").Stop();
}
//获取截图
function GetScreenshots() {
var imgInfo = document.getElementById("USBCamera").GetScreenshots();
if (imgInfo.result) {
document.getElementById("DisplayScreenshots").src = "data:image/jpg;base64," + imgInfo.message;
} else {
alert("截图获取失败");
}
}
//获取分辨率
function GetResolution() {
console.log(document.getElementById("USBCamera").GetResolution());
}
</script>
</head>
<body>
<div>
<button onclick="TestConnection()">测试连接</button>
<button onclick="StartCamera()">打开摄像头</button>
<button onclick="StopCamera()">关闭摄像头</button>
<button onclick="GetScreenshots()">获取截图</button>
<button onclick="GetResolution()">获取分辨率</button>
</div>
<hr />
<h3>截图显示</h3>
<img style="width:300px;height:300px;" id="DisplayScreenshots" />
<div style="float:right;width:70%">
<object id="USBCamera" classid="clsid:887FC3C3-A970-4A36-92EF-D4EB31541C40">
<div class="alert alert-danger">视频控件载入失败!如未安装,请<a href="InstallationPackage/Setup1.msi">下载安装</a>。</div>
</object>
</div>
</body>
</html>
实现的效果图:


结语
VS使用的2015社区版,社区版。。。有点尴尬所以打包使用的 Visual Studio Installer
一回生二回熟,第一次写这个满脸蒙蔽。
不过这次稍微整理了一下,改动了一点,后续准备找个人脸识别的开源接口融合进来。
下面给个下载地址,可以直接调试,需要注意使用IE,由于没有加证书,IE需要设置。
工具>Internet选项
>安全>加入受信任的站点>
>自定义级别>ActiveX控件和插件>ActivexX控件自动提示>启用
>对未标记为可安全执行的脚本的ActivexX控件初始化并执行脚本>启用
百度云盘链接:下载
C# Activex调用USB摄像头--附带源码的更多相关文章
- Log4Net 日志配置[附带源码]
前述 园子里有许多人对log4net这款开源的日志记录控件有很多介绍.在这里个人再做一次总结,希望对以后有所帮助,需要的时候可以直接使用,减少查阅资料的时间.利用log4net可以方便地将日志信息记录 ...
- 【Hook技术】实现从"任务管理器"中保护进程不被关闭 + 附带源码 + 进程保护知识扩展
[Hook技术]实现从"任务管理器"中保护进程不被关闭 + 附带源码 + 进程保护知识扩展 公司有个监控程序涉及到进程的保护问题,需要避免用户通过任务管理器结束掉监控进程,这里使用 ...
- iOS 指南针的制作 附带源码
iOS 指南针的制作 附带源码 代码下载地址: http://vdisk.weibo.com/s/HK4yE http://pan.baidu.com/share/link?shareid=7 ...
- Linux下USB suspend/resume源码分析【转】
转自:http://blog.csdn.net/aaronychen/article/details/3928479 Linux下USB suspend/resume源码分析 Author:aaron ...
- 【轮子狂魔】抛弃IIS,打造个性的Web Server - WebAPI/Lua/MVC(附带源码)
引言 此篇是<[轮子狂魔]抛弃IIS,向天借个HttpListener - 基础篇(附带源码)>的续篇,也可以说是提高篇,如果你对HttpListener不甚了解的话,建议先看下基础篇. ...
- SpringMVC关于json、xml自动转换的原理研究[附带源码分析 --转
SpringMVC关于json.xml自动转换的原理研究[附带源码分析] 原文地址:http://www.cnblogs.com/fangjian0423/p/springMVC-xml-json-c ...
- Python机器学习经典实例电子版和附带源码
Python机器学习经典实例电子版和附带源码 下载:https://pan.baidu.com/s/1m6ODNJk--PWHW8Vdsdjs-g 提取码:nyc0 分享更多python数据分析相关电 ...
- 简单的c#winform象棋游戏(附带源码)
算法源自网络(网络源码连接:http://www.mycodes.net/161/6659.htm) 整体思路:用二维数组构建棋盘每一个数组元素封装为一个picturebox附带若干属性(例如:棋 ...
- SpringMVC拦截器详解[附带源码分析]
目录 前言 重要接口及类介绍 源码分析 拦截器的配置 编写自定义的拦截器 总结 总结 前言 SpringMVC是目前主流的Web MVC框架之一. 如果有同学对它不熟悉,那么请参考它的入门blog:h ...
随机推荐
- webpack开发小总结
webpack开发前端的时候往往是单独自己的服务器: 1.express 带上 webpack-dev-middleware(自己实现了热更新,而且在memory-fileSystem,不会产生多余文 ...
- Golang异常处理-panic与recover
Golang异常处理-panic与recover 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在程序设计中,容错是相当重要的一部分工作,在 Go中它是通过错误处理来实现的,err ...
- python操作txt文件中数据教程[4]-python去掉txt文件行尾换行
python操作txt文件中数据教程[4]-python去掉txt文件行尾换行 觉得有用的话,欢迎一起讨论相互学习~Follow Me 参考文章 python操作txt文件中数据教程[1]-使用pyt ...
- spring中bean配置和注入场景分析
bean与spring容器的关系 Bean配置信息定义了Bean的实现及依赖关系,Spring容器根据各种形式的Bean配置信息在容器内部建立Bean定义注册表,然后根据注册表加载.实例化Bean,并 ...
- javascript沙箱模式
沙箱模式解决了命名空间模式的如下几个缺点: 1.对单个全局变量的依赖变成了应用程序的全局变量依赖.在命名空间模式中,是没有办法使同一个应用程序或库的2个版本运行在同一个页面中.2.对这种以点分割的名字 ...
- 20155206 2016-2017-2 《Java程序设计》第6周学习总结
20155206 2016-2017-2 <Java程序设计>第6周学习总结 教材学习内容总结 串流设计 流(Stream)是对「输入输出」的抽象,注意「输入输出」是相对程序而言的. Ja ...
- 利用iis虚拟目录实现文件服务器功能
要求说明: 通过网站上传文件保存到统一的文件服务器上. 服务器说明: 1.文件服务器以下称为FilesServer,IP地址为:192.168.1.213 2.Web服务器为以下称为WebServer ...
- android MeasureSpec的三个测量模式
1.MeasureSpec含义 其实可以去看MeasureSpec的文档,里面对MeasureSpec的作用介绍得很清楚.MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureS ...
- 在Windows环境中利用Responder工具窃取NTLMv2哈希
在Windows环境中利用Responder工具窃取NTLMv2哈希 翻译自:https://github.com/incredibleindishell/Windows-AD-environment ...
- mysql 字符编码设置
安装mysql时如果字符编码为默认值latin1,则需要修改为utf8以便支持中文数据. 命令如下: 1.显示数据库字符集 mysql> show create database test;+- ...