MainWindow.xaml.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. using ICSharpCode.SharpZipLib.Zip.Compression.Streams;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Diagnostics;
  5. using System.Drawing;
  6. using System.Drawing.Imaging;
  7. using System.IO;
  8. using System.IO.Compression;
  9. using System.IO.Ports;
  10. using System.Linq;
  11. using System.Text;
  12. using System.Threading;
  13. using System.Threading.Tasks;
  14. using System.Windows;
  15. using System.Windows.Controls;
  16. using System.Windows.Data;
  17. using System.Windows.Documents;
  18. using System.Windows.Input;
  19. using System.Windows.Interop;
  20. using System.Windows.Media;
  21. using System.Windows.Media.Imaging;
  22. using System.Windows.Navigation;
  23. using System.Windows.Shapes;
  24. using Brush = System.Drawing.Brush;
  25. using Color = System.Drawing.Color;
  26. using Pen = System.Drawing.Pen;
  27. using PixelFormat = System.Drawing.Imaging.PixelFormat;
  28. using Rectangle = System.Drawing.Rectangle;
  29. namespace Air105Camera
  30. {
  31. /// <summary>
  32. /// MainWindow.xaml 的交互逻辑
  33. /// </summary>
  34. [PropertyChanged.AddINotifyPropertyChangedInterface]
  35. public partial class MainWindow : Window
  36. {
  37. public MainWindow()
  38. {
  39. InitializeComponent();
  40. }
  41. //将Bitmap对象转换成bitmapImage对象
  42. private BitmapImage ConvertBitmapToBitmapImage(Bitmap bitmap)
  43. {
  44. MemoryStream stream = new MemoryStream();
  45. bitmap.Save(stream, ImageFormat.Bmp);
  46. BitmapImage image = new BitmapImage();
  47. image.BeginInit();
  48. image.StreamSource = stream;
  49. image.EndInit();
  50. return image;
  51. }
  52. //刷新显示当前图片
  53. private void ShowImage()
  54. {
  55. this.Dispatcher.Invoke(() =>
  56. {
  57. var temp = ConvertBitmapToBitmapImage(image);
  58. CameraImage.Source = temp;
  59. });
  60. }
  61. //缓存图片数据
  62. Bitmap image = new Bitmap(320, 240, PixelFormat.Format16bppRgb565);
  63. Graphics imageGraphic = null;
  64. //串口对象
  65. private SerialPort Uart = new SerialPort();
  66. //总共收到多少包
  67. public int totalPack { get; set; } = 0;
  68. //总共刷了多少张
  69. public int totalPic { get; set; } = 0;
  70. //FPS数
  71. public double fpsNow { get; set; } = 0;
  72. public bool connectEnable { get; set; } = true;
  73. public bool disconnectEnable { get; set; } = true;
  74. private void Window_Loaded(object sender, RoutedEventArgs e)
  75. {
  76. this.DataContext = this;
  77. imageGraphic = Graphics.FromImage(image);
  78. //刷白
  79. imageGraphic.FillRectangle(new SolidBrush(Color.White), 0, 0, image.Width, image.Height);
  80. //图片显示一下
  81. ShowImage();
  82. // 绑定事件监听,用于监听HID设备插拔
  83. (PresentationSource.FromVisual(this) as HwndSource)?.AddHook(WndProc);
  84. RefreshPortList();
  85. //串口数据处理,随便写写
  86. new Thread(Uart2Image).Start();
  87. }
  88. private void ConnectButton_Click(object sender, RoutedEventArgs e)
  89. {
  90. if (Uart.IsOpen)
  91. {
  92. return;
  93. }
  94. if (((string)SerialComboBox.SelectedItem).Length > 0)
  95. {
  96. var portName = (string)SerialComboBox.SelectedItem;
  97. Task.Run(() =>
  98. {
  99. try
  100. {
  101. Uart.PortName = portName;
  102. Uart.Open();
  103. }
  104. catch (Exception ex)
  105. {
  106. MessageBox.Show(ex.Message);
  107. }
  108. });
  109. }
  110. }
  111. private void DisconnectButton_Click(object sender, RoutedEventArgs e)
  112. {
  113. if (!Uart.IsOpen)
  114. {
  115. return;
  116. }
  117. try
  118. {
  119. Uart.Close();
  120. }
  121. catch(Exception ex)
  122. {
  123. MessageBox.Show(ex.Message);
  124. }
  125. }
  126. private static int UsbPluginDeley = 0;
  127. private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
  128. {
  129. if (msg == 0x219)// 监听USB设备插拔消息
  130. {
  131. if (UsbPluginDeley == 0)
  132. {
  133. ++UsbPluginDeley; // Task启动需要准备时间,这里提前对公共变量加一
  134. Task.Run(() =>
  135. {
  136. do Task.Delay(100).Wait();
  137. while (++UsbPluginDeley < 10);
  138. UsbPluginDeley = 0;
  139. RefreshPortList();
  140. });
  141. }
  142. else UsbPluginDeley = 1;
  143. handled = true;
  144. }
  145. return IntPtr.Zero;
  146. }
  147. /// <summary>
  148. /// 刷新设备列表,并在断开恢复后重连
  149. /// </summary>
  150. /// <returns>串口列表是否成功获取</returns>
  151. private bool RefreshPortList()
  152. {
  153. bool result = true;
  154. var list = SerialPort.GetPortNames();
  155. Dispatcher.Invoke(delegate
  156. {
  157. SerialComboBox.ItemsSource = list;
  158. if (SerialComboBox.Items.Count == 0)
  159. {
  160. result = false;
  161. }
  162. else if (SerialComboBox.Items.Contains(Uart.PortName))
  163. {
  164. SerialComboBox.SelectedItem = Uart.PortName;
  165. }
  166. else SerialComboBox.SelectedIndex = 0;
  167. });
  168. return result;
  169. }
  170. bool closed = false;
  171. private void Window_Closing(object sender, System.ComponentModel.CancelEventArgs e)
  172. {
  173. closed = true;
  174. //直接强关软件
  175. Environment.Exit(0);
  176. }
  177. private void RefreshButton_Click(object sender, RoutedEventArgs e)
  178. {
  179. RefreshPortList();
  180. }
  181. /*
  182. 16字节头+N字节数据
  183. 4字节字符串 “VCAM” + 2字节图像宽度 + 2字节图像高度 + 4字节本次数据从哪行开始刷 + 2字节未压缩数据长度 + 2字节压缩后图像数据长度
  184. 如果压缩前和压缩后数据长度是一致的,那就没有压缩;如果不一致,则后面的N字节要用zlib解压
  185. RGB565大端格式
  186. */
  187. //串口数据处理
  188. private void Uart2Image()
  189. {
  190. byte[] temp = new byte[10 * 1024 * 1024];//缓冲区
  191. int p = 0;//上次的位置
  192. Stopwatch sw = new Stopwatch();
  193. sw.Start();
  194. bool getNewData = false;//记一下是不是收到新数据了
  195. while (true)
  196. {
  197. if (closed)
  198. return;
  199. connectEnable = !Uart.IsOpen;
  200. disconnectEnable = Uart.IsOpen;
  201. try
  202. {
  203. if (Uart.IsOpen && Uart.BytesToRead > 0)
  204. {
  205. //读数据
  206. var toread = Uart.BytesToRead;
  207. Uart.Read(temp, p, toread);
  208. p += toread;
  209. getNewData = true;
  210. }
  211. else if (!Uart.IsOpen)
  212. {
  213. fpsNow = 0;
  214. p = 0;
  215. }
  216. if(getNewData)
  217. {
  218. getNewData = false;//收到新数据再刷
  219. for (int i = 3; i < p; i++)
  220. {
  221. //匹配VCAM头
  222. if (temp[i - 3] == 'V' && temp[i - 2] == 'C' && temp[i - 1] == 'A' && temp[i] == 'M'
  223. && p - i > 16)
  224. {
  225. var width = BitConverter.ToUInt16(temp, i + 1);
  226. var height = BitConverter.ToUInt16(temp, i + 3);
  227. var start = BitConverter.ToUInt32(temp, i + 5);
  228. var rawLen = BitConverter.ToUInt16(temp, i + 9);
  229. var comLen = BitConverter.ToUInt16(temp, i + 11);
  230. Debug.WriteLine($"found VCAM: {BitConverter.ToString(temp, i - 3, 16)}\r\n" +
  231. $"{width},{height},{start},{rawLen},{comLen},{i},{p}");
  232. //获取实际数据长度
  233. var realLength = comLen < rawLen ? comLen : rawLen;
  234. if (realLength <= p - i - 13)//一包数据够了
  235. {
  236. totalPack++;
  237. Debug.WriteLine($"{realLength} bytes, start unpack");
  238. //存储rgb数据
  239. byte[] data = null;
  240. if (comLen < rawLen)//包压缩了
  241. {
  242. //Debug.WriteLine(BitConverter.ToString(temp, i + 13, comLen));
  243. using MemoryStream compressed = new MemoryStream(temp, i + 13, comLen);
  244. using MemoryStream decompressed = new MemoryStream();
  245. using InflaterInputStream inputStream = new InflaterInputStream(compressed);
  246. inputStream.CopyTo(decompressed);
  247. data = decompressed.ToArray();
  248. }
  249. else//没压缩
  250. {
  251. Debug.WriteLine($"rawLen,{rawLen},{i + 13 + rawLen}");
  252. data = new byte[rawLen];
  253. for (int n = i + 13; n < i + 13 + rawLen; n++)
  254. data[n - (i + 13)] = temp[n];
  255. }
  256. //直接把rgb565数据写入bitmap,这样快
  257. //先把数据改成小端
  258. for (int n = 0; n < data.Length; n += 2)
  259. {
  260. (data[n], data[n + 1]) = (data[n + 1], data[n]);
  261. }
  262. var rect = new Rectangle(0, (int)start, width, data.Length / 2 / width);
  263. BitmapData bmpData = image.LockBits(rect, ImageLockMode.ReadWrite, image.PixelFormat);
  264. IntPtr ptr = bmpData.Scan0;// Get the address of the first line.
  265. System.Runtime.InteropServices.Marshal.Copy(data, 0, ptr, data.Length);
  266. image.UnlockBits(bmpData);
  267. //刷到最后一行时再显示当前图片
  268. if (start >= height - 16)
  269. {
  270. sw.Stop();
  271. fpsNow = 1000.0 / sw.ElapsedMilliseconds;
  272. sw.Restart();
  273. totalPic++;
  274. ShowImage();
  275. }
  276. //剩下的数据全部左移到开头
  277. for (int n = i + 13 + realLength; n < p; n++)
  278. temp[n - (i + 13 + realLength)] = temp[n];
  279. p -= i + 13 + realLength;
  280. Debug.WriteLine($"done, p now {p}");
  281. if (p > 3)
  282. i = 2;
  283. else
  284. break;
  285. }
  286. }
  287. }
  288. }
  289. Thread.Sleep(1);
  290. }
  291. catch (Exception e)
  292. {
  293. Debug.WriteLine(e.Message);
  294. }
  295. }
  296. }
  297. }
  298. }