版博士V2.0程序
Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

2 роки тому
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951
  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Drawing;
  5. using System.Drawing.Imaging;
  6. using System.IO;
  7. using System.Linq;
  8. using System.Reflection;
  9. using System.Runtime.InteropServices;
  10. using System.Text;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. using System.Timers;
  14. using System.Windows.Forms;
  15. using MvCamCtrl.NET;
  16. using Newtonsoft.Json.Linq;
  17. using OpenCvSharp.Dnn;
  18. namespace AssistClient.Device
  19. {
  20. public class ScannerDev : IDisposable
  21. {
  22. private uint m_nRowStep = 0;
  23. [DllImport("user32.dll")]
  24. [return: MarshalAs(UnmanagedType.Bool)]
  25. private static extern bool IsWindow(IntPtr hWnd);
  26. public enum ScannerType
  27. {
  28. [Description("板卡相机")]
  29. GENTL = 0,
  30. [Description("网口相机")]
  31. CC = 1,
  32. }
  33. private ScannerType scannerType;
  34. private string bmpSavePath;
  35. public Action<WarningEnum, string> WarningEvent;
  36. /// <summary>
  37. /// 拍照回传 (1-num,文件名.bmp)
  38. /// </summary>
  39. public Action<int, string> ScanEventPath;
  40. //public Action<int, byte[]> ScanEvent;
  41. public Action<int, Bitmap> ScanEvent;
  42. /// <summary>
  43. /// 曝光
  44. /// </summary>
  45. public float ExposureTime { get; private set; }
  46. /// <summary>
  47. /// 增益
  48. /// </summary>
  49. public float Gain { get; private set; }
  50. /// <summary>
  51. /// 帧率
  52. /// </summary>
  53. public float ResultingFrameRate { get; private set; }
  54. /// <summary>
  55. /// 图片大小
  56. /// </summary>
  57. public Size size { get; private set; }
  58. /// <summary>
  59. /// 是否连续模式
  60. /// </summary>
  61. public bool isContinuousMode { get; private set; }
  62. /// <summary>
  63. /// 是否打开设备成功
  64. /// </summary>
  65. public bool IsInit { get; private set; } = false;
  66. //public string ErrInfo { get; private set; }
  67. //private System.Timers.Timer timer = new System.Timers.Timer();
  68. #region
  69. private static MyCamera.cbOutputExdelegate ImageCallback;
  70. private static MyCamera.MV_CHUNK_DATA_CONTENT stChunkInfo;// Chunk结构体信息
  71. private MyCamera.MV_GENTL_DEV_INFO_LIST m_stGentDeviceList = new MyCamera.MV_GENTL_DEV_INFO_LIST();
  72. private MyCamera.MV_GENTL_IF_INFO_LIST m_stIFInfoList = new MyCamera.MV_GENTL_IF_INFO_LIST();
  73. private MyCamera.MV_CC_DEVICE_INFO_LIST m_stCCDeviceList = new MyCamera.MV_CC_DEVICE_INFO_LIST();
  74. private MyCamera device = new MyCamera();
  75. private bool m_bGrabbing = false;//开始采集TAG
  76. private Thread m_hReceiveThread = null;
  77. //显示图像控件句柄
  78. private IntPtr previewHwnd = IntPtr.Zero;
  79. //拍照数量
  80. private int scanNum = 0;
  81. #endregion
  82. public ScannerDev(ScannerType type)
  83. {
  84. this.scannerType = type;
  85. }
  86. public bool open()
  87. {
  88. if (IsInit) return true;
  89. System.GC.Collect();
  90. int nRet;
  91. try
  92. {
  93. if (this.scannerType == ScannerType.GENTL)
  94. {
  95. nRet = MyCamera.MV_CC_EnumInterfacesByGenTL_NET(ref m_stIFInfoList, Config.Scanner_GENTL_CTI);
  96. if (File.Exists(Config.Scanner_GENTL_CTI))
  97. {
  98. if (0 != nRet || m_stIFInfoList.nInterfaceNum < 1)
  99. {
  100. WarningEvent?.Invoke(WarningEnum.High, $"Enumerate interfaces fail! ({nRet})");
  101. return false;
  102. }
  103. }
  104. //
  105. MyCamera.MV_GENTL_IF_INFO stIFInfo = (MyCamera.MV_GENTL_IF_INFO)Marshal.PtrToStructure(m_stIFInfoList.pIFInfo[0], typeof(MyCamera.MV_GENTL_IF_INFO));
  106. nRet = MyCamera.MV_CC_EnumDevicesByGenTL_NET(ref stIFInfo, ref m_stGentDeviceList);
  107. if (0 != nRet || m_stGentDeviceList.nDeviceNum < 1)
  108. {
  109. WarningEvent?.Invoke(WarningEnum.High, $"Enumerate devices fail! ({nRet})");
  110. return false;
  111. }
  112. // ch:获取选择的设备信息 | en:Get selected device information
  113. MyCamera.MV_GENTL_DEV_INFO device = (MyCamera.MV_GENTL_DEV_INFO)Marshal.PtrToStructure(m_stGentDeviceList.pDeviceInfo[0], typeof(MyCamera.MV_GENTL_DEV_INFO));
  114. // ch:打开设备 | en:Open device
  115. if (null == this.device)
  116. {
  117. this.device = new MyCamera();
  118. if (null == this.device)
  119. {
  120. WarningEvent?.Invoke(WarningEnum.High, $"Open device fail!");
  121. return false;
  122. }
  123. }
  124. nRet = this.device.MV_CC_CreateDeviceByGenTL_NET(ref device);
  125. if (MyCamera.MV_OK != nRet)
  126. {
  127. WarningEvent?.Invoke(WarningEnum.High, $"Open device fail! ({nRet})");
  128. return false;
  129. }
  130. nRet = this.device.MV_CC_OpenDevice_NET();
  131. if (MyCamera.MV_OK != nRet)
  132. {
  133. this.device.MV_CC_DestroyDevice_NET();
  134. WarningEvent?.Invoke(WarningEnum.High, $"Device open fail! ({nRet})");
  135. return false;
  136. }
  137. }
  138. else //CC
  139. {
  140. nRet = MyCamera.MV_CC_EnumDevices_NET(MyCamera.MV_GIGE_DEVICE | MyCamera.MV_USB_DEVICE, ref m_stCCDeviceList);
  141. if (0 != nRet || m_stCCDeviceList.nDeviceNum < 1)
  142. {
  143. WarningEvent?.Invoke(WarningEnum.High, $"Enumerate devices fail! ({nRet})");
  144. return false;
  145. }
  146. // ch:获取选择的设备信息
  147. MyCamera.MV_CC_DEVICE_INFO device = (MyCamera.MV_CC_DEVICE_INFO)Marshal.PtrToStructure(m_stCCDeviceList.pDeviceInfo[0], typeof(MyCamera.MV_CC_DEVICE_INFO));
  148. // ch:打开设备 | en:Open device
  149. if (null == this.device)
  150. {
  151. this.device = new MyCamera();
  152. if (null == this.device)
  153. {
  154. WarningEvent?.Invoke(WarningEnum.High, $"Open device fail!");
  155. return false;
  156. }
  157. }
  158. nRet = this.device.MV_CC_CreateDevice_NET(ref device);
  159. if (MyCamera.MV_OK != nRet)
  160. {
  161. WarningEvent?.Invoke(WarningEnum.High, $"Open device fail! ({nRet})");
  162. return false;
  163. }
  164. nRet = this.device.MV_CC_OpenDevice_NET();
  165. if (MyCamera.MV_OK != nRet)
  166. {
  167. this.device.MV_CC_DestroyDevice_NET();
  168. WarningEvent?.Invoke(WarningEnum.High, $"Open device fail! ({nRet})");
  169. return false;
  170. }
  171. // ch:探测网络最佳包大小(只对GigE相机有效) | en:Detection network optimal package size(It only works for the GigE camera)
  172. if (device.nTLayerType == MyCamera.MV_GIGE_DEVICE)
  173. {
  174. int nPacketSize = this.device.MV_CC_GetOptimalPacketSize_NET();
  175. if (nPacketSize > 0)
  176. {
  177. nRet = this.device.MV_CC_SetIntValue_NET("GevSCPSPacketSize", (uint)nPacketSize);
  178. //if (nRet != MyCamera.MV_OK)
  179. //{
  180. // ShowErrorMsg("Set Packet Size failed!", nRet);
  181. //}
  182. }
  183. else
  184. {
  185. //ShowErrorMsg("Get Packet Size failed!", nPacketSize);
  186. }
  187. }
  188. }
  189. // ch:注册回调函数 | en:Register image callback //本类不使用回调,使用线程主动循环读取
  190. //ImageCallback = new MyCamera.cbOutputExdelegate(ImageCallbackFunc);
  191. //nRet = device.MV_CC_RegisterImageCallBackEx_NET(ImageCallback, IntPtr.Zero);
  192. //if (MyCamera.MV_OK != nRet)
  193. //{
  194. // WarningEvent?.Invoke(WarningEnum.High, $"Register image callback failed! ({nRet})");
  195. // return false;
  196. //}
  197. init2();
  198. // ch:设置采集[软触发/连续]模式 | en:Set Continues Aquisition Mode
  199. device.MV_CC_SetEnumValue_NET("AcquisitionMode", (uint)MyCamera.MV_CAM_ACQUISITION_MODE.MV_ACQ_MODE_CONTINUOUS);
  200. setMode(false); //软触发
  201. //
  202. getParam();
  203. IsInit = true;
  204. //timer.Elapsed += Timer_Elapsed;
  205. //timer.Interval = 100;
  206. //timer.Enabled = true;
  207. return true;
  208. }
  209. catch (Exception ex)
  210. {
  211. WarningEvent?.Invoke(WarningEnum.High, ex.Message);
  212. return false;
  213. }
  214. }
  215. public void close()
  216. {
  217. if (!IsInit) return;
  218. try
  219. {
  220. IsInit = false;
  221. // ch:取流标志位清零 | en:Reset flow flag bit
  222. if (m_bGrabbing == true)
  223. {
  224. m_bGrabbing = false;
  225. m_hReceiveThread.Join();
  226. }
  227. // ch:关闭设备 | en:Close Device
  228. device.MV_CC_CloseDevice_NET();
  229. device.MV_CC_DestroyDevice_NET();
  230. }
  231. catch { }
  232. }
  233. /// <summary>
  234. ///
  235. /// </summary>
  236. /// <param name="hwnd">显示图像控件句柄</param>
  237. /// <returns></returns>
  238. public bool start(IntPtr preview_Hwnd,string bmp_save_path)
  239. {
  240. if (!IsInit) return false;
  241. if (m_bGrabbing) return true;
  242. this.previewHwnd= preview_Hwnd;
  243. this.bmpSavePath = bmp_save_path;
  244. // ch:标志位置位true | en:Set position bit true
  245. m_bGrabbing = true;
  246. m_hReceiveThread = new Thread(ReceiveThreadProcess);
  247. m_hReceiveThread.Start();
  248. // ch:开始采集 | en:Start Grabbing
  249. int nRet = device.MV_CC_StartGrabbing_NET();
  250. if (MyCamera.MV_OK != nRet)
  251. {
  252. m_bGrabbing = false;
  253. m_hReceiveThread.Join();
  254. WarningEvent?.Invoke(WarningEnum.High, $"Start Grabbing Fail! ({nRet})");
  255. return false;
  256. }
  257. return true;
  258. }
  259. public bool stop()
  260. {
  261. if (!IsInit) return false;
  262. if (!m_bGrabbing) return true;
  263. try
  264. {
  265. // ch:标志位设为false | en:Set flag bit false
  266. m_bGrabbing = false;
  267. m_hReceiveThread.Join();
  268. // ch:停止采集 | en:Stop Grabbing
  269. int nRet = device.MV_CC_StopGrabbing_NET();
  270. if (nRet != MyCamera.MV_OK)
  271. {
  272. WarningEvent?.Invoke(WarningEnum.High, $"Stop Grabbing Fail! ({nRet})");
  273. return false;
  274. }
  275. }
  276. catch
  277. {
  278. return false;
  279. }
  280. return true;
  281. }
  282. #region private
  283. private bool init2()
  284. {
  285. // ch:获取包大小 || en: Get Payload Size
  286. MyCamera.MVCC_INTVALUE stParam = new MyCamera.MVCC_INTVALUE();
  287. //int nRet = device.MV_CC_GetIntValue_NET("PayloadSize", ref stParam);//strKey [IN] 属性键值,如获取宽度信息则为"Width" pstValue[IN][OUT] 返回给调用者有关相机属性结构体指针
  288. //if (MyCamera.MV_OK != nRet)
  289. //{
  290. // System.Windows.Forms.MessageBox.Show("Get PayloadSize Fail");
  291. // return false;
  292. //}
  293. //g_nPayloadSize = stParam.nCurValue;
  294. // ch:获取高 || en: Get Height
  295. int nRet = device.MV_CC_GetIntValue_NET("Height", ref stParam);
  296. if (MyCamera.MV_OK != nRet)
  297. {
  298. System.Windows.Forms.MessageBox.Show("Get Height Fail");
  299. return false;
  300. }
  301. uint nHeight = stParam.nCurValue;
  302. // ch:获取宽 || en: Get Width
  303. nRet = device.MV_CC_GetIntValue_NET("Width", ref stParam);
  304. if (MyCamera.MV_OK != nRet)
  305. {
  306. System.Windows.Forms.MessageBox.Show("Get Width Fail");
  307. return false;
  308. }
  309. uint nWidth = stParam.nCurValue;
  310. this.size = new Size((int)nWidth, (int)nHeight);
  311. // ch:获取步长 || en: Get nRowStep
  312. m_nRowStep = nWidth * nHeight;
  313. return true;
  314. }
  315. /// <summary>
  316. /// false-软触发 true-连续
  317. /// </summary>
  318. /// <param name="type"></param>
  319. public void setMode(bool isContinuous)
  320. {
  321. if (isContinuous)
  322. {
  323. device.MV_CC_SetEnumValue_NET("TriggerMode", (uint)MyCamera.MV_CAM_TRIGGER_MODE.MV_TRIGGER_MODE_OFF);
  324. }
  325. else
  326. {
  327. device.MV_CC_SetEnumValue_NET("TriggerMode", (uint)MyCamera.MV_CAM_TRIGGER_MODE.MV_TRIGGER_MODE_ON);
  328. // ch:触发源选择:0 - Line0;
  329. // 1 - Line1;
  330. // 2 - Line2;
  331. // 3 - Line3;
  332. // 4 - Counter;
  333. // 7 - Software;
  334. //if (cbSoftTrigger.Checked)
  335. device.MV_CC_SetEnumValue_NET("TriggerSource", (uint)MyCamera.MV_CAM_TRIGGER_SOURCE.MV_TRIGGER_SOURCE_SOFTWARE);
  336. //else
  337. // m_MyCamera.MV_CC_SetEnumValue_NET("TriggerSource", (uint)MyCamera.MV_CAM_TRIGGER_SOURCE.MV_TRIGGER_SOURCE_LINE0);
  338. }
  339. isContinuousMode = isContinuous;
  340. }
  341. public void setPreviewWin(IntPtr preview_Hwnd)
  342. {
  343. this.previewHwnd = preview_Hwnd;
  344. }
  345. public void ReceiveThreadProcess()
  346. {
  347. MyCamera.MV_FRAME_OUT stFrameInfo = new MyCamera.MV_FRAME_OUT();
  348. MyCamera.MV_DISPLAY_FRAME_INFO stDisplayInfo = new MyCamera.MV_DISPLAY_FRAME_INFO();
  349. int nRet = MyCamera.MV_OK;
  350. while (m_bGrabbing)
  351. {
  352. bool isOpenWin = IsWindow(this.previewHwnd);
  353. if ((!isContinuousMode && this.scanNum < 1) ||
  354. (isContinuousMode && (this.previewHwnd == IntPtr.Zero || !isOpenWin)))
  355. {
  356. Thread.Sleep(500);
  357. continue;
  358. }
  359. //
  360. nRet = device.MV_CC_GetImageBuffer_NET(ref stFrameInfo, 1000);
  361. if (nRet == MyCamera.MV_OK)
  362. {
  363. if (RemoveCustomPixelFormats(stFrameInfo.stFrameInfo.enPixelType)
  364. || stFrameInfo.stFrameInfo.nFrameLen == 0)
  365. {
  366. device.MV_CC_FreeImageBuffer_NET(ref stFrameInfo);
  367. continue;
  368. }
  369. //pictureBox1显示
  370. if (this.previewHwnd != IntPtr.Zero && isOpenWin)
  371. {
  372. stDisplayInfo.hWnd = this.previewHwnd;// pictureBox1.Handle;
  373. stDisplayInfo.pData = stFrameInfo.pBufAddr;
  374. stDisplayInfo.nDataLen = stFrameInfo.stFrameInfo.nFrameLen;
  375. stDisplayInfo.nWidth = stFrameInfo.stFrameInfo.nWidth;
  376. stDisplayInfo.nHeight = stFrameInfo.stFrameInfo.nHeight;
  377. stDisplayInfo.enPixelType = stFrameInfo.stFrameInfo.enPixelType;
  378. device.MV_CC_DisplayOneFrame_NET(ref stDisplayInfo);
  379. }
  380. //保存
  381. if (this.scanNum>0)//save tag
  382. {
  383. MyCamera.MV_SAVE_IMG_TO_FILE_PARAM stSaveFileParam = new MyCamera.MV_SAVE_IMG_TO_FILE_PARAM();
  384. //lock (BufForDriverLock)
  385. {
  386. if (scannerType == ScannerType.CC)
  387. {
  388. stSaveFileParam.enImageType = MyCamera.MV_SAVE_IAMGE_TYPE.MV_Image_Bmp;
  389. stSaveFileParam.enPixelType = stFrameInfo.stFrameInfo.enPixelType;
  390. stSaveFileParam.pData = stFrameInfo.pBufAddr;//m_BufForDriver;
  391. stSaveFileParam.nDataLen = stFrameInfo.stFrameInfo.nFrameLen;
  392. stSaveFileParam.nHeight = stFrameInfo.stFrameInfo.nHeight;
  393. stSaveFileParam.nWidth = stFrameInfo.stFrameInfo.nWidth;
  394. stSaveFileParam.iMethodValue = 2;
  395. //Save path
  396. //stSaveFileParam.pImagePath = this.bmpSavePath + "\\" + scannerType.ToString() + "_w" + stSaveFileParam.nWidth.ToString() + "_h" + stSaveFileParam.nHeight.ToString() + "_fn" + stFrameInfo.stFrameInfo.nFrameNum.ToString() + ".bmp";
  397. stSaveFileParam.pImagePath = this.bmpSavePath + "\\" + scannerType.ToString() + "_"+DateTime.Now.Ticks + ".bmp";
  398. //m_MyCamera.MV_CC_RegisterImageCallBackEx_NET(new MyCamera.cbOutputExdelegate(ImageCallBack), IntPtr.Zero);
  399. nRet = device.MV_CC_SaveImageToFile_NET(ref stSaveFileParam);
  400. if (MyCamera.MV_OK == nRet)
  401. {
  402. if (ScanEventPath != null)
  403. ScanEventPath?.Invoke(this.scanNum--, stSaveFileParam.pImagePath);
  404. else
  405. ScanEvent?.Invoke(this.scanNum--, read2Bmp(stSaveFileParam.pImagePath));
  406. }
  407. }
  408. else
  409. {
  410. //=== raw byte[]
  411. //byte[] rawFileBuff = toRawImage(stFrameInfo);
  412. //if (rawFileBuff != null)
  413. // ScanEvent?.Invoke(this.scanNum--, rawFileBuff);
  414. //=== bmp
  415. Bitmap bmp = toBitmap(stFrameInfo.pBufAddr, stFrameInfo.stFrameInfo);
  416. //bmp = bytes2bmp(bmp2bytes(bmp));
  417. //string bmpPath = "d:\\" + DateTime.Now.Ticks + ".bmp";
  418. //bmp.Save(bmpPath, ImageFormat.Bmp);
  419. ////bmp = (Bitmap)Bitmap.FromFile(bmpPath);
  420. //bmp = read2Bmp(bmpPath);
  421. if (bmp != null)
  422. ScanEvent?.Invoke(this.scanNum--, bmp);
  423. }
  424. }
  425. }
  426. //free
  427. device.MV_CC_FreeImageBuffer_NET(ref stFrameInfo);
  428. }
  429. else
  430. {
  431. Thread.Sleep(5);
  432. }
  433. }
  434. }
  435. public void AfterGigeUsbImageAcquired_Station1(MyCamera.MV_FRAME_OUT_INFO_EX pFrameInfo, IntPtr pTemp )
  436. {
  437. Bitmap Station1Image1 = new System.Drawing.Bitmap(pFrameInfo.nWidth, pFrameInfo.nHeight, pFrameInfo.nWidth * 1, System.Drawing.Imaging.PixelFormat.Format8bppIndexed, pTemp);
  438. ColorPalette cp = Station1Image1.Palette;
  439. for (int i = 0; i < 256; i++)
  440. {
  441. cp.Entries[i] = Color.FromArgb(i, i, i);
  442. }
  443. Station1Image1.Palette = cp;
  444. }
  445. private byte[] toRawImage(MyCamera.MV_FRAME_OUT stFrameOut)
  446. {
  447. MyCamera.MvGvspPixelType enType = MyCamera.MvGvspPixelType.PixelType_Gvsp_Undefined;
  448. uint nChannelNum = 0;
  449. if (IsColorPixelFormat(stFrameOut.stFrameInfo.enPixelType))
  450. {
  451. enType = MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed;
  452. nChannelNum = 3;
  453. }
  454. else if (IsMonoPixelFormat(stFrameOut.stFrameInfo.enPixelType))
  455. {
  456. enType = MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono8;
  457. nChannelNum = 1;
  458. }
  459. else
  460. return null;
  461. //
  462. if (enType != MyCamera.MvGvspPixelType.PixelType_Gvsp_Undefined)
  463. {
  464. IntPtr pBufForConvert = Marshal.AllocHGlobal((int)(stFrameOut.stFrameInfo.nWidth * stFrameOut.stFrameInfo.nHeight * nChannelNum));
  465. MyCamera.MV_PIXEL_CONVERT_PARAM stConvertPixelParam = new MyCamera.MV_PIXEL_CONVERT_PARAM();
  466. stConvertPixelParam.nWidth = stFrameOut.stFrameInfo.nWidth;
  467. stConvertPixelParam.nHeight = stFrameOut.stFrameInfo.nHeight;
  468. stConvertPixelParam.pSrcData = stFrameOut.pBufAddr;
  469. stConvertPixelParam.nSrcDataLen = stFrameOut.stFrameInfo.nFrameLen;
  470. stConvertPixelParam.enSrcPixelType = stFrameOut.stFrameInfo.enPixelType;
  471. stConvertPixelParam.enDstPixelType = enType;
  472. stConvertPixelParam.pDstBuffer = pBufForConvert;
  473. stConvertPixelParam.nDstBufferSize = (uint)(stFrameOut.stFrameInfo.nWidth * stFrameOut.stFrameInfo.nHeight * nChannelNum);
  474. int nRet = device.MV_CC_ConvertPixelType_NET(ref stConvertPixelParam);
  475. if (MyCamera.MV_OK != nRet)
  476. return null;
  477. // ch:将图像数据保存到本地文件 | en:Save image data to local file
  478. byte[] rawImage = new byte[stConvertPixelParam.nDstLen];
  479. Marshal.Copy(stConvertPixelParam.pDstBuffer, rawImage, 0, (int)stConvertPixelParam.nDstLen);
  480. if (pBufForConvert != IntPtr.Zero)
  481. Marshal.FreeHGlobal(pBufForConvert);
  482. //
  483. return rawImage;
  484. }
  485. return null;
  486. }
  487. private Bitmap toBitmap(IntPtr pData, MyCamera.MV_FRAME_OUT_INFO_EX pFrameInfo)
  488. {
  489. int nRet = MyCamera.MV_OK;
  490. //MyCamera device = m_pMyCamera;
  491. //MyCamera.MV_FRAME_OUT_INFO_EX pFrameInfo = new MyCamera.MV_FRAME_OUT_INFO_EX();
  492. //IntPtr pData = System.Runtime.InteropServices.Marshal.AllocHGlobal((int)g_nPayloadSize);
  493. if (pData == IntPtr.Zero)
  494. return null;
  495. IntPtr pTemp = IntPtr.Zero;
  496. IntPtr pImageBuffer = IntPtr.Zero;
  497. if (IsColorPixelFormat(pFrameInfo.enPixelType)) // 彩色图像处理
  498. {
  499. if (pFrameInfo.enPixelType == MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed)
  500. {
  501. pTemp = pData;
  502. }
  503. else
  504. {
  505. // 其他格式彩色图像转为RGB
  506. nRet = getBmpPtr(device, pData, pFrameInfo.nHeight, pFrameInfo.nWidth, pFrameInfo.enPixelType, pImageBuffer, MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed);
  507. if (MyCamera.MV_OK != nRet)
  508. return null;
  509. pTemp = pImageBuffer;
  510. }
  511. // Packed转Plane
  512. Byte[] byteArrImageData = new Byte[m_nRowStep * 3];
  513. unsafe
  514. {
  515. byte* pBufForSaveImage = (byte*)pTemp;
  516. UInt32 nSupWidth = (pFrameInfo.nWidth + (UInt32)3) & 0xfffffffc;
  517. for (int nRow = 0; nRow < pFrameInfo.nHeight; nRow++)
  518. {
  519. for (int col = 0; col < pFrameInfo.nWidth; col++)
  520. {
  521. byteArrImageData[nRow * nSupWidth + col] = pBufForSaveImage[nRow * pFrameInfo.nWidth * 3 + (3 * col)];
  522. byteArrImageData[pFrameInfo.nWidth * pFrameInfo.nHeight + nRow * nSupWidth + col] = pBufForSaveImage[nRow * pFrameInfo.nWidth * 3 + (3 * col + 1)];
  523. byteArrImageData[pFrameInfo.nWidth * pFrameInfo.nHeight * 2 + nRow * nSupWidth + col] = pBufForSaveImage[nRow * pFrameInfo.nWidth * 3 + (3 * col + 2)];
  524. }
  525. }
  526. pTemp = System.Runtime.InteropServices.Marshal.UnsafeAddrOfPinnedArrayElement(byteArrImageData, 0);
  527. }
  528. //
  529. Bitmap Station1Image1 = new System.Drawing.Bitmap(pFrameInfo.nWidth, pFrameInfo.nHeight, pFrameInfo.nWidth * 3, System.Drawing.Imaging.PixelFormat.Format24bppRgb, pTemp);
  530. //ColorPalette cp = Station1Image1.Palette;
  531. //for (int i = 0; i < 256; i++)
  532. //{
  533. // cp.Entries[i] = Color.FromArgb(i, i, i);
  534. //}
  535. //Station1Image1.Palette = cp;
  536. return Station1Image1;
  537. }
  538. else if (IsMonoPixelFormat(pFrameInfo.enPixelType)) // Mono图像处理
  539. {
  540. if (pFrameInfo.enPixelType == MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono8)
  541. {
  542. pTemp = pData;
  543. }
  544. else
  545. {
  546. // 其他格式Mono转为Mono8
  547. //nRet = ConvertToMono8(device, pData, pImageBuffer, pFrameInfo.nHeight, pFrameInfo.nWidth, pFrameInfo.enPixelType);
  548. nRet = getBmpPtr(device, pData, pFrameInfo.nHeight, pFrameInfo.nWidth, pFrameInfo.enPixelType, pImageBuffer, MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed);
  549. if (MyCamera.MV_OK != nRet)
  550. return null;
  551. pTemp = pImageBuffer;
  552. }
  553. //
  554. Bitmap Station1Image1 = new System.Drawing.Bitmap(pFrameInfo.nWidth, pFrameInfo.nHeight, pFrameInfo.nWidth * 1, System.Drawing.Imaging.PixelFormat.Format8bppIndexed, pTemp);
  555. ColorPalette cp = Station1Image1.Palette;
  556. for (int i = 0; i < 256; i++)
  557. {
  558. cp.Entries[i] = Color.FromArgb(i, i, i);
  559. }
  560. Station1Image1.Palette = cp;
  561. return Station1Image1;
  562. }
  563. else
  564. return null;
  565. }
  566. public Int32 getBmpPtr(MyCamera device, IntPtr pSrc, ushort nHeight, ushort nWidth, MyCamera.MvGvspPixelType nPixelType, IntPtr pDst, MyCamera.MvGvspPixelType type)
  567. {
  568. if (IntPtr.Zero == pSrc || IntPtr.Zero == pDst)
  569. {
  570. return MyCamera.MV_E_PARAMETER;
  571. }
  572. int nRet = MyCamera.MV_OK;
  573. //MyCamera device = obj as MyCamera;
  574. MyCamera.MV_PIXEL_CONVERT_PARAM stPixelConvertParam = new MyCamera.MV_PIXEL_CONVERT_PARAM();
  575. stPixelConvertParam.pSrcData = pSrc;//源数据
  576. if (IntPtr.Zero == stPixelConvertParam.pSrcData)
  577. {
  578. return -1;
  579. }
  580. stPixelConvertParam.nWidth = nWidth;//图像宽度
  581. stPixelConvertParam.nHeight = nHeight;//图像厚度
  582. stPixelConvertParam.enSrcPixelType = nPixelType;//源数据的格式
  583. stPixelConvertParam.nSrcDataLen = (uint)(nWidth * nHeight * ((((uint)nPixelType) >> 16) & 0x00ff) >> 3);
  584. stPixelConvertParam.nDstBufferSize = (uint)(nWidth * nHeight * ((((uint)type) >> 16) & 0x00ff) >> 3);
  585. stPixelConvertParam.pDstBuffer = pDst;//转换后的数据
  586. stPixelConvertParam.enDstPixelType = type;
  587. stPixelConvertParam.nDstBufferSize = (uint)nWidth * nHeight * 3;
  588. nRet = device.MV_CC_ConvertPixelType_NET(ref stPixelConvertParam);//格式转换
  589. if (MyCamera.MV_OK != nRet)
  590. {
  591. return -1;
  592. }
  593. return MyCamera.MV_OK;
  594. }
  595. /// <summary>
  596. /// 其他黑白格式转为Mono8
  597. /// </summary>
  598. /// <param name="obj"></param>
  599. /// <param name="pInData">输出图片数据</param>
  600. /// <param name="pOutData">输出图片数据</param>
  601. /// <param name="nHeight">高</param>
  602. /// <param name="nWidth">宽</param>
  603. /// <param name="nPixelType">像素格式</param>
  604. /// <returns></returns>
  605. public Int32 ConvertToMono8(MyCamera device, IntPtr pInData, IntPtr pOutData, ushort nHeight, ushort nWidth, MyCamera.MvGvspPixelType nPixelType)
  606. {
  607. if (IntPtr.Zero == pInData || IntPtr.Zero == pOutData)
  608. {
  609. return MyCamera.MV_E_PARAMETER;
  610. }
  611. int nRet = MyCamera.MV_OK;
  612. //MyCamera device = obj as MyCamera;
  613. MyCamera.MV_PIXEL_CONVERT_PARAM stPixelConvertParam = new MyCamera.MV_PIXEL_CONVERT_PARAM();
  614. stPixelConvertParam.pSrcData = pInData;//源数据
  615. if (IntPtr.Zero == stPixelConvertParam.pSrcData)
  616. {
  617. return -1;
  618. }
  619. stPixelConvertParam.nWidth = nWidth;//图像宽度
  620. stPixelConvertParam.nHeight = nHeight;//图像厚度
  621. stPixelConvertParam.enSrcPixelType = nPixelType;//源数据的格式
  622. stPixelConvertParam.nSrcDataLen = (uint)(nWidth * nHeight * ((((uint)nPixelType) >> 16) & 0x00ff) >> 3);
  623. stPixelConvertParam.nDstBufferSize = (uint)(nWidth * nHeight * ((((uint)MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed) >> 16) & 0x00ff) >> 3);
  624. stPixelConvertParam.pDstBuffer = pOutData;//转换后的数据
  625. stPixelConvertParam.enDstPixelType = MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono8;
  626. stPixelConvertParam.nDstBufferSize = (uint)(nWidth * nHeight * 3);
  627. nRet = device.MV_CC_ConvertPixelType_NET(ref stPixelConvertParam);//格式转换
  628. if (MyCamera.MV_OK != nRet)
  629. {
  630. return -1;
  631. }
  632. return nRet;
  633. }
  634. /// <summary>
  635. /// 其他彩色格式转为RGB8
  636. /// </summary>
  637. /// <param name="obj"></param>
  638. /// <param name="pSrc"></param>
  639. /// <param name="nHeight"></param>
  640. /// <param name="nWidth"></param>
  641. /// <param name="nPixelType"></param>
  642. /// <param name="pDst"></param>
  643. /// <returns></returns>
  644. public Int32 ConvertToRGB(MyCamera device, IntPtr pSrc, ushort nHeight, ushort nWidth, MyCamera.MvGvspPixelType nPixelType, IntPtr pDst)
  645. {
  646. if (IntPtr.Zero == pSrc || IntPtr.Zero == pDst)
  647. {
  648. return MyCamera.MV_E_PARAMETER;
  649. }
  650. int nRet = MyCamera.MV_OK;
  651. //MyCamera device = obj as MyCamera;
  652. MyCamera.MV_PIXEL_CONVERT_PARAM stPixelConvertParam = new MyCamera.MV_PIXEL_CONVERT_PARAM();
  653. stPixelConvertParam.pSrcData = pSrc;//源数据
  654. if (IntPtr.Zero == stPixelConvertParam.pSrcData)
  655. {
  656. return -1;
  657. }
  658. stPixelConvertParam.nWidth = nWidth;//图像宽度
  659. stPixelConvertParam.nHeight = nHeight;//图像厚度
  660. stPixelConvertParam.enSrcPixelType = nPixelType;//源数据的格式
  661. stPixelConvertParam.nSrcDataLen = (uint)(nWidth * nHeight * ((((uint)nPixelType) >> 16) & 0x00ff) >> 3);
  662. stPixelConvertParam.nDstBufferSize = (uint)(nWidth * nHeight * ((((uint)MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed) >> 16) & 0x00ff) >> 3);
  663. stPixelConvertParam.pDstBuffer = pDst;//转换后的数据
  664. stPixelConvertParam.enDstPixelType = MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed;
  665. stPixelConvertParam.nDstBufferSize = (uint)nWidth * nHeight * 3;
  666. nRet = device.MV_CC_ConvertPixelType_NET(ref stPixelConvertParam);//格式转换
  667. if (MyCamera.MV_OK != nRet)
  668. {
  669. return -1;
  670. }
  671. return MyCamera.MV_OK;
  672. }
  673. private static void ImageCallbackFunc(IntPtr pData, ref MyCamera.MV_FRAME_OUT_INFO_EX pFrameInfo, IntPtr pUser)
  674. {
  675. //Print parse the timestamp information in the frame
  676. Console.WriteLine("ImageCallBack: ExposureTime[" + Convert.ToString(pFrameInfo.fExposureTime)
  677. + "], SecondCount[" + Convert.ToString(pFrameInfo.nSecondCount)
  678. + "], CycleCount[" + Convert.ToString(pFrameInfo.nCycleCount)
  679. + "], CycleOffset[" + Convert.ToString(pFrameInfo.nCycleOffset)
  680. + "], FrameNum[" + Convert.ToString(pFrameInfo.nFrameNum) + "]");
  681. int nStrSize = Marshal.SizeOf(stChunkInfo);
  682. int nUnparsedChunkContent = (int)pFrameInfo.UnparsedChunkList.pUnparsedChunkContent;
  683. for (int i = 0; i < pFrameInfo.nUnparsedChunkNum; i++)
  684. {
  685. stChunkInfo = (MyCamera.MV_CHUNK_DATA_CONTENT)Marshal.PtrToStructure((IntPtr)(nUnparsedChunkContent + i * nStrSize), typeof(MyCamera.MV_CHUNK_DATA_CONTENT));
  686. Console.WriteLine("ChunkInfo:" + "ChunkID[0x{0:x8}],ChunkLen[" + Convert.ToString(stChunkInfo.nChunkLen) + "]", stChunkInfo.nChunkID);
  687. }
  688. Console.WriteLine("************************************");
  689. }
  690. static bool IsMonoPixelFormat(MyCamera.MvGvspPixelType enType)
  691. {
  692. switch (enType)
  693. {
  694. case MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono8:
  695. case MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono10:
  696. case MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono10_Packed:
  697. case MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono12:
  698. case MyCamera.MvGvspPixelType.PixelType_Gvsp_Mono12_Packed:
  699. return true;
  700. default:
  701. return false;
  702. }
  703. }
  704. static bool IsColorPixelFormat(MyCamera.MvGvspPixelType enType)
  705. {
  706. switch (enType)
  707. {
  708. case MyCamera.MvGvspPixelType.PixelType_Gvsp_RGB8_Packed:
  709. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BGR8_Packed:
  710. case MyCamera.MvGvspPixelType.PixelType_Gvsp_YUV422_Packed:
  711. case MyCamera.MvGvspPixelType.PixelType_Gvsp_YUV422_YUYV_Packed:
  712. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGR8:
  713. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerRG8:
  714. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGB8:
  715. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerBG8:
  716. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGB10:
  717. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGB10_Packed:
  718. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerBG10:
  719. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerBG10_Packed:
  720. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerRG10:
  721. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerRG10_Packed:
  722. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGR10:
  723. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGR10_Packed:
  724. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGB12:
  725. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGB12_Packed:
  726. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerBG12:
  727. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerBG12_Packed:
  728. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerRG12:
  729. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerRG12_Packed:
  730. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGR12:
  731. case MyCamera.MvGvspPixelType.PixelType_Gvsp_BayerGR12_Packed:
  732. return true;
  733. default:
  734. return false;
  735. }
  736. }
  737. // ch:去除自定义的像素格式 | en:Remove custom pixel formats
  738. private bool RemoveCustomPixelFormats(MyCamera.MvGvspPixelType enPixelFormat)
  739. {
  740. Int32 nResult = ((int)enPixelFormat) & (unchecked((Int32)0x80000000));
  741. if (0x80000000 == nResult)
  742. {
  743. return true;
  744. }
  745. else
  746. {
  747. return false;
  748. }
  749. }
  750. #endregion
  751. public void getParam()
  752. {
  753. if (!IsInit) return;
  754. MyCamera.MVCC_FLOATVALUE stParam = new MyCamera.MVCC_FLOATVALUE();
  755. int nRet = device.MV_CC_GetFloatValue_NET("ExposureTime", ref stParam);
  756. if (MyCamera.MV_OK == nRet)
  757. {
  758. ExposureTime = stParam.fCurValue;//.ToString("F1");
  759. }
  760. nRet = device.MV_CC_GetFloatValue_NET("Gain", ref stParam);
  761. if (MyCamera.MV_OK == nRet)
  762. {
  763. Gain = stParam.fCurValue;//.ToString("F1");
  764. }
  765. nRet = device.MV_CC_GetFloatValue_NET("ResultingFrameRate", ref stParam);
  766. if (MyCamera.MV_OK == nRet)
  767. {
  768. ResultingFrameRate = stParam.fCurValue;//.ToString("F1");
  769. }
  770. }
  771. /// <summary>
  772. ///
  773. /// </summary>
  774. /// <param name="exposureTime">曝光</param>
  775. /// <param name="gain">增益</param>
  776. /// <param name="resultingFrameRate">帧率</param>
  777. public void setParam(float exposureTime, float gain, float resultingFrameRate)
  778. {
  779. if (!IsInit) return;
  780. bool change = false;
  781. int nRet;
  782. if (exposureTime != ExposureTime)
  783. {
  784. device.MV_CC_SetEnumValue_NET("ExposureAuto", 0);
  785. nRet = device.MV_CC_SetFloatValue_NET("ExposureTime", exposureTime);
  786. if (nRet != MyCamera.MV_OK)
  787. {
  788. WarningEvent?.Invoke(WarningEnum.Normal, $"Scanner Set Exposure Time Fail! ({nRet})");
  789. }
  790. change = true;
  791. }
  792. if (gain != Gain)
  793. {
  794. device.MV_CC_SetEnumValue_NET("GainAuto", 0);
  795. nRet = device.MV_CC_SetFloatValue_NET("Gain", gain);
  796. if (nRet != MyCamera.MV_OK)
  797. {
  798. WarningEvent?.Invoke(WarningEnum.Normal, $"Scanner Set Gain Time Fail! ({nRet})");
  799. }
  800. change = true;
  801. }
  802. if (resultingFrameRate != ResultingFrameRate)
  803. {
  804. nRet = device.MV_CC_SetFloatValue_NET("AcquisitionFrameRate", resultingFrameRate);
  805. if (nRet != MyCamera.MV_OK)
  806. {
  807. WarningEvent?.Invoke(WarningEnum.Normal, $"Scanner Set AcquisitionFrameRate Time Fail! ({nRet})");
  808. }
  809. change = true;
  810. }
  811. //
  812. if(change)
  813. getParam();
  814. }
  815. /// <summary>
  816. /// 软触发拍照
  817. /// </summary>
  818. /// <param name="num">拍照数量</param>
  819. /// <returns></returns>
  820. public bool scan(int num=1)
  821. {
  822. if (!IsInit) return false;
  823. if (!isContinuousMode)
  824. {
  825. // ch:触发命令 | en:Trigger command
  826. int nRet = device.MV_CC_SetCommandValue_NET("TriggerSoftware");
  827. if (MyCamera.MV_OK != nRet)
  828. {
  829. WarningEvent?.Invoke(WarningEnum.Normal, $"Trigger Software Fail! ({nRet})");
  830. return false;
  831. }
  832. }
  833. this.scanNum= num;
  834. return true;
  835. }
  836. public void Dispose()
  837. {
  838. stop();
  839. }
  840. private byte[] bmp2bytes(Bitmap bmp)
  841. {
  842. MemoryStream ms = new MemoryStream();
  843. bmp.Save(ms, System.Drawing.Imaging.ImageFormat.Bmp);
  844. byte[] bytes = ms.GetBuffer(); //byte[] bytes= ms.ToArray(); 这两句都可以,至于区别么,下面有解释
  845. ms.Close();
  846. bmp.Dispose();
  847. return bytes;
  848. }
  849. private Bitmap bytes2bmp(byte[] bytes)
  850. {
  851. MemoryStream ms1 = new MemoryStream(bytes);
  852. Bitmap bm = (Bitmap)Image.FromStream(ms1);
  853. ms1.Close();
  854. return bm;
  855. }
  856. public Bitmap read2Bmp(string path)
  857. {
  858. MemoryStream ms = new System.IO.MemoryStream(File.ReadAllBytes(path));
  859. Bitmap bmp = new Bitmap(ms);
  860. ms.Close();
  861. ms.Dispose();
  862. //FileStream fs = new FileStream(path, FileMode.Open);
  863. //byte[] bmpdata = new byte[fs.Length];
  864. //fs.Read(bmpdata, 0, bmpdata.Length);
  865. //Bitmap bmp = new Bitmap(fs);
  866. //fs.Close();
  867. return bmp;
  868. }
  869. }
  870. }