WebPreviewHandler.cs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. using FISLib.LiveVideo;
  2. using System;
  3. using System.Drawing;
  4. using System.IO;
  5. using System.Net;
  6. using System.Net.WebSockets;
  7. using System.Text;
  8. using System.Threading;
  9. using System.Threading.Tasks;
  10. using Vinno.FIS.Sonopost.Managers;
  11. using Vinno.FIS.Sonopost.Managers.Interfaces;
  12. using Vinno.IUS.Common.Log;
  13. namespace Vinno.FIS.Sonopost.Features.Web
  14. {
  15. internal class WebPreviewHandler
  16. {
  17. private const int PREVIEW_WIDTH = 360;
  18. private readonly ILiveVideoManager _liveVideoManager;
  19. private bool _busy = false;
  20. private const int FRAME_IGNORE_COUNT = 2; // 采集24帧/预览12帧 = 2
  21. private int _curFrameSignIndex = 0;
  22. public event EventHandler<byte[]> PreviewImageReceived;
  23. public WebPreviewHandler()
  24. {
  25. _liveVideoManager = AppManager.Instance.GetManager<ILiveVideoManager>();
  26. _liveVideoManager.PreviewImageReceived += OnPreviewImageReceived;
  27. }
  28. public async void HandleRequest(HttpListenerContext context)
  29. {
  30. string sessionId = Guid.NewGuid().ToString("N");
  31. HttpListenerWebSocketContext webSocketContext;
  32. try
  33. {
  34. webSocketContext = await context.AcceptWebSocketAsync(subProtocol: null);
  35. string remoteAddress = context.Request.RemoteEndPoint.Address.ToString();
  36. Logger.WriteLineInfo($"[{nameof(WebPreviewHandler)}] connected. ip: {remoteAddress}");
  37. }
  38. catch (Exception ex)
  39. {
  40. context.Response.StatusCode = 500;
  41. context.Response.Close();
  42. Logger.WriteLineError($"[{nameof(WebPreviewHandler)}] connect failed. ex: {ex}");
  43. return;
  44. }
  45. WebSocket webSocket = webSocketContext.WebSocket;
  46. object loker = new object();
  47. EventHandler<byte[]> action = (object sender, byte[] e) =>
  48. {
  49. if (webSocket.State == WebSocketState.Open)
  50. {
  51. lock (loker)
  52. {
  53. SendMessage(webSocket, e).GetAwaiter().GetResult();
  54. }
  55. }
  56. };
  57. PreviewImageReceived += action;
  58. try
  59. {
  60. byte[] receiveBuffer = new byte[1024];
  61. while (webSocket.State == WebSocketState.Open)
  62. {
  63. WebSocketReceiveResult receiveResult = await webSocket.ReceiveAsync(new ArraySegment<byte>(receiveBuffer), CancellationToken.None);
  64. if (receiveResult.MessageType == WebSocketMessageType.Close)
  65. {
  66. await webSocket.CloseAsync(WebSocketCloseStatus.NormalClosure, string.Empty, CancellationToken.None);
  67. Logger.WriteLineInfo($"[{nameof(WebPreviewHandler)}] remote disconnected.");
  68. }
  69. else
  70. {
  71. byte[] data = Encoding.UTF8.GetBytes("i got u!");
  72. await SendMessage(webSocket, data);
  73. }
  74. }
  75. }
  76. catch (Exception ex)
  77. {
  78. Logger.WriteLineInfo($"[{nameof(WebPreviewHandler)}]{nameof(HandleRequest)} error: ex: {ex}");
  79. }
  80. finally
  81. {
  82. PreviewImageReceived -= action;
  83. if (webSocket != null)
  84. {
  85. webSocket.Dispose();
  86. }
  87. }
  88. }
  89. private unsafe void OnPreviewImageReceived(object sender, FISImageFrameData frame)
  90. {
  91. if (frame.Data == null || frame.Width == 0 || frame.Height == 0)
  92. {
  93. return;
  94. }
  95. var bitmapSize = frame.Width * frame.Height * 4;
  96. if (bitmapSize < frame.Size)
  97. {
  98. return;
  99. }
  100. // 跳过帧处理
  101. _curFrameSignIndex++;
  102. if (_curFrameSignIndex >= FRAME_IGNORE_COUNT)
  103. {
  104. _curFrameSignIndex = 0;
  105. return;
  106. }
  107. if (_busy) return;
  108. _busy = true;
  109. try
  110. {
  111. using (Bitmap source = new Bitmap(frame.Width, frame.Height, System.Drawing.Imaging.PixelFormat.Format32bppRgb))
  112. {
  113. var locker = source.LockBits(new Rectangle(0, 0, frame.Width, frame.Height)
  114. , System.Drawing.Imaging.ImageLockMode.ReadWrite, System.Drawing.Imaging.PixelFormat.Format32bppRgb);
  115. Buffer.MemoryCopy(frame.Data.ToPointer(), locker.Scan0.ToPointer(), frame.Size, frame.Size);
  116. source.UnlockBits(locker);
  117. if (frame.Width - PREVIEW_WIDTH > 200)
  118. {
  119. int height = (int)(1.0f * PREVIEW_WIDTH / frame.Width * frame.Height);
  120. using (Bitmap dest = new Bitmap(source, PREVIEW_WIDTH, height))
  121. {
  122. using (MemoryStream ms = new MemoryStream())
  123. {
  124. dest.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
  125. PreviewImageReceived?.Invoke(null, ms.GetBuffer());
  126. }
  127. }
  128. }
  129. else
  130. {
  131. using (MemoryStream ms = new MemoryStream())
  132. {
  133. source.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
  134. PreviewImageReceived?.Invoke(null, ms.GetBuffer());
  135. }
  136. }
  137. }
  138. }
  139. catch (Exception ex)
  140. {
  141. Logger.WriteLineError($"[{nameof(WebPreviewHandler)}]{nameof(OnPreviewImageReceived)} error:{ex}");
  142. }
  143. finally
  144. {
  145. Thread.Sleep(1);
  146. _busy = false;
  147. }
  148. }
  149. private async Task SendMessage(WebSocket webSocket, byte[] data)
  150. {
  151. try
  152. {
  153. int bufferSize = 10 * 1024;
  154. int seek = 0;
  155. ArraySegment<byte> buffer;
  156. while (seek < data.Length && webSocket.State == WebSocketState.Open)
  157. {
  158. int left = data.Length - seek;
  159. if (left < bufferSize)
  160. {
  161. buffer = new ArraySegment<byte>(data, seek, left);
  162. await webSocket.SendAsync(buffer, WebSocketMessageType.Binary, true, CancellationToken.None);
  163. }
  164. else
  165. {
  166. buffer = new ArraySegment<byte>(data, seek, bufferSize);
  167. await webSocket.SendAsync(buffer, WebSocketMessageType.Binary, false, CancellationToken.None);
  168. }
  169. seek += bufferSize;
  170. }
  171. }
  172. catch (Exception ex)
  173. {
  174. Logger.WriteLineError($"[{nameof(WebPreviewHandler)}]{nameof(SendMessage)} error:{ex}");
  175. }
  176. }
  177. }
  178. }