ImageSplicer.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. using OpenCvSharp;
  2. using OpenCvSharp.Extensions;
  3. using OTSCommon.DBOperate.Model;
  4. using OTSIncAReportApp.OTSRstMgrFunction;
  5. using OTSIncAReportGraph;
  6. using OTSIncAReportGraph.Class;
  7. using OTSIncAReportGraph.OTSIncAReportGraphFuncation;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Data;
  11. using System.Drawing;
  12. using System.Linq;
  13. using System.Text;
  14. using System.Threading.Tasks;
  15. namespace OTSIncAReportApp._1_UI.Control_Graph.Controls
  16. {
  17. class ImageSplicer
  18. {
  19. public double picYmin { get; set; }
  20. public double picXmin { get; set; }
  21. public int imgeH { get; set; }
  22. public int imgeW { get; set; }
  23. /// <summary>
  24. /// 单纯图像拼接无颗粒颜色渲染
  25. /// </summary>
  26. /// <param name="resultFile"></param>
  27. /// <param name="m_list_allDfield"></param>
  28. /// <returns></returns>
  29. public List<DataTable> Mosaics(ResultFile resultFile,List<DisplayRectangle> m_list_allDfield)
  30. {
  31. List<DataTable> list_dt_picdata = new List<DataTable>();
  32. DataTable picDatat = new DataTable();
  33. List<Field> fieldlist = resultFile.List_OTSField;
  34. picDatat.Columns.Add("X", typeof(double));
  35. picDatat.Columns.Add("Y", typeof(double));
  36. foreach (var f in m_list_allDfield)
  37. {
  38. DataRow dr = picDatat.NewRow();
  39. dr["X"] = f.OTSCoordinatePos.X;
  40. dr["Y"] = f.OTSCoordinatePos.Y;
  41. picDatat.Rows.Add(dr);
  42. }
  43. DataTable total_dt_X = picDatat.Copy();
  44. DataView dv_x = total_dt_X.DefaultView;
  45. DataView dv_x_2 = dv_x.ToTable(true, "X").DefaultView;
  46. dv_x_2.Sort = "X ASC";
  47. total_dt_X = dv_x_2.ToTable();
  48. DataTable total_dt_Y = picDatat.Copy();
  49. DataView dv_Y = total_dt_Y.DefaultView;
  50. DataView dv_Y_2 = dv_Y.ToTable(true, "Y").DefaultView;
  51. dv_Y_2.Sort = "Y ASC";
  52. total_dt_Y = dv_Y_2.ToTable();
  53. //循环每一行图片
  54. for (int i = 0; i < total_dt_Y.Rows.Count; i++)
  55. {
  56. NLog.LogManager.GetCurrentClassLogger().Info("Splice line" + i + 1.ToString() + "of" + total_dt_Y.Rows.Count.ToString() + ".....");
  57. DataTable data = new DataTable();
  58. data.Columns.Add("mat", typeof(Bitmap));
  59. data.Columns.Add("X", typeof(double));
  60. //循环每行图片进行拼接
  61. Bitmap bitmap = new Bitmap((int)m_list_allDfield[0].OriginalImage.Width, (int)m_list_allDfield[0].OriginalImage.Height);
  62. for (int a = 0; a < total_dt_X.Rows.Count; a++)
  63. {
  64. DataRow dr2 = data.NewRow();
  65. Graphics g = Graphics.FromImage(bitmap);
  66. g.Clear(Color.White);
  67. g.Dispose();
  68. dr2["mat"] = bitmap;
  69. dr2["X"] = total_dt_X.Rows[a]["X"];
  70. data.Rows.Add(dr2);
  71. }
  72. //循环所有图片
  73. foreach (var f in m_list_allDfield)
  74. {
  75. if (total_dt_Y.Rows[i]["Y"].ToString() == f.OTSCoordinatePos.Y.ToString())
  76. {
  77. //循环图片的行数
  78. for (int c = 0; c < data.Rows.Count; c++)
  79. {
  80. if (data.Rows[c]["X"].ToString() == f.OTSCoordinatePos.X.ToString())
  81. {
  82. Bitmap image = new Bitmap(bitmap.Width, bitmap.Height);
  83. Graphics g2 = Graphics.FromImage(image);
  84. Color color1 = Color.FromArgb(217, 217, 217);
  85. g2.Clear(color1);
  86. g2.DrawImage(image, 0, 0, f.OriginalImage.Width, f.OriginalImage.Height);
  87. Graphics graph_2 = Graphics.FromImage(image);
  88. for (int a = 0; a < fieldlist.Count; a++)
  89. {
  90. if (fieldlist[a].FieldID.ToString() == f.FieldID.ToString())
  91. {
  92. //先获取该Field中的所有Particle
  93. List<Particle> list_particle;
  94. list_particle = fieldlist[a].ParticleList;
  95. //再循环计算所有的Particle对象
  96. foreach (Particle particle in list_particle)
  97. {
  98. List<Segment> list_seg;
  99. list_seg = particle.SegmentList;
  100. //创建颗粒分布图对应的类对象
  101. List<DisplaySegment> list_dsegment = new List<DisplaySegment>();
  102. //再循环取出里面所有的segment
  103. foreach (Segment seg in list_seg)
  104. {
  105. #region 创建DSegment对象,并将STD分析出的化合物颜色保存到DSegment对象中
  106. System.Drawing.Point on_p = new System.Drawing.Point() { X = seg.Start, Y = seg.Height };
  107. System.Drawing.Point off_p = new System.Drawing.Point() { X = seg.Start + seg.Length, Y = seg.Height };
  108. Pen npen = new Pen(Color.FromArgb(255 - particle.AveGray, 0, 0, 0));
  109. graph_2.DrawLine(npen, on_p, off_p);
  110. #endregion
  111. }
  112. }
  113. }
  114. }
  115. data.Rows[c]["mat"] = image;
  116. data.Rows[c]["X"] = f.OTSCoordinatePos.X;
  117. }
  118. }
  119. }
  120. }
  121. DataView dataView1 = data.DefaultView;
  122. dataView1.Sort = "X ASC";
  123. data = dataView1.ToTable();
  124. list_dt_picdata.Add(data);
  125. }
  126. return list_dt_picdata;
  127. }
  128. /// <summary>
  129. /// 图像拼接并进行颗粒渲染颜色
  130. /// </summary>
  131. /// <param name="resultFile"></param>
  132. /// <param name="m_list_allDfield"></param>
  133. /// <returns></returns>
  134. public List<DataTable> ImageStitchingParticleRendering(ResultFile resultFile, List<DisplayRectangle> m_list_allDfield)
  135. {
  136. List<DataTable> list_dt_picdata = new List<DataTable>();
  137. DataTable picDatat = new DataTable();
  138. List<Field> fieldlist = resultFile.List_OTSField;
  139. picDatat.Columns.Add("X", typeof(double));
  140. picDatat.Columns.Add("Y", typeof(double));
  141. foreach (var f in m_list_allDfield)
  142. {
  143. DataRow dr = picDatat.NewRow();
  144. dr["X"] = f.OTSCoordinatePos.X;
  145. dr["Y"] = f.OTSCoordinatePos.Y;
  146. picDatat.Rows.Add(dr);
  147. }
  148. DataTable total_dt_X = picDatat.Copy();
  149. DataView dv_x = total_dt_X.DefaultView;
  150. DataView dv_x_2 = dv_x.ToTable(true, "X").DefaultView;
  151. dv_x_2.Sort = "X ASC";
  152. total_dt_X = dv_x_2.ToTable();
  153. DataTable total_dt_Y = picDatat.Copy();
  154. DataView dv_Y = total_dt_Y.DefaultView;
  155. DataView dv_Y_2 = dv_Y.ToTable(true, "Y").DefaultView;
  156. dv_Y_2.Sort = "Y ASC";
  157. total_dt_Y = dv_Y_2.ToTable();
  158. picYmin = Convert.ToDouble(total_dt_Y.Rows[total_dt_Y.Rows.Count - 1]["Y"]);
  159. picXmin = Convert.ToDouble(total_dt_X.Rows[0]["X"]);
  160. for (int i = 0; i < total_dt_Y.Rows.Count; i++)
  161. {
  162. NLog.LogManager.GetCurrentClassLogger().Info("Splice line" + i + 1.ToString() + "of" + total_dt_Y.Rows.Count.ToString() + ".....");
  163. DataTable data = new DataTable();
  164. data.Columns.Add("mat", typeof(Bitmap));
  165. data.Columns.Add("X", typeof(double));
  166. data.Columns.Add("Y", typeof(double));
  167. imgeH = (int)m_list_allDfield[0].OriginalImage.Height;
  168. imgeW = (int)m_list_allDfield[0].OriginalImage.Width;
  169. Bitmap bitmap = new Bitmap((int)m_list_allDfield[0].OriginalImage.Width, (int)m_list_allDfield[0].OriginalImage.Height);
  170. for (int a = 0; a < total_dt_X.Rows.Count; a++)
  171. {
  172. DataRow dr2 = data.NewRow();
  173. Graphics g = Graphics.FromImage(bitmap);
  174. g.Clear(Color.White);
  175. g.Dispose();
  176. dr2["mat"] = bitmap;
  177. dr2["X"] = total_dt_X.Rows[a]["X"];
  178. data.Rows.Add(dr2);
  179. }
  180. foreach (var f in m_list_allDfield)
  181. {
  182. if (total_dt_Y.Rows[i]["Y"].ToString() == f.OTSCoordinatePos.Y.ToString())
  183. {
  184. for (int c = 0; c < data.Rows.Count; c++)
  185. {
  186. if (data.Rows[c]["X"].ToString() == f.OTSCoordinatePos.X.ToString())
  187. {
  188. Bitmap image = new Bitmap(bitmap.Width, bitmap.Height);
  189. Graphics graph_2 = Graphics.FromImage(image);
  190. Color color1 = Color.FromArgb(217, 217, 217);
  191. graph_2.Clear(color1);
  192. //Color color1 = Color.FromArgb(217, 217, 217);
  193. //graph_2.Clear(color1);
  194. for (int a = 0; a < fieldlist.Count; a++)
  195. {
  196. if (fieldlist[a].FieldID.ToString() == f.FieldID.ToString())
  197. {
  198. //先获取该Field中的所有Particle
  199. List<Particle> list_particle;
  200. list_particle = fieldlist[a].ParticleList;
  201. //再循环计算所有的Particle对象
  202. foreach (Particle particle in list_particle)
  203. {
  204. List<Segment> list_seg;
  205. list_seg = particle.SegmentList;
  206. //创建颗粒分布图对应的类对象
  207. List<DisplaySegment> list_dsegment = new List<DisplaySegment>();
  208. //再循环取出里面所有的segment
  209. foreach (Segment seg in list_seg)
  210. {
  211. #region 创建DSegment对象,并将STD分析出的化合物颜色保存到DSegment对象中
  212. System.Drawing.Point on_p = new System.Drawing.Point() { X = seg.Start, Y = seg.Height };
  213. System.Drawing.Point off_p = new System.Drawing.Point() { X = seg.Start + seg.Length, Y = seg.Height };
  214. var color = DrawFunction.GetColorBySTDTypeIDForBSEAndSorImage(particle.TypeColor, particle.TypeId);
  215. Pen npen = new Pen(color);
  216. graph_2.DrawLine(npen, on_p, off_p);
  217. #endregion
  218. }
  219. }
  220. }
  221. }
  222. data.Rows[c]["mat"] = image;
  223. data.Rows[c]["X"] = f.OTSCoordinatePos.X;
  224. data.Rows[c]["Y"] = f.OTSCoordinatePos.Y;
  225. }
  226. }
  227. }
  228. }
  229. DataView dataView1 = data.DefaultView;
  230. dataView1.Sort = "X ASC";
  231. data = dataView1.ToTable();
  232. list_dt_picdata.Add(data);
  233. }
  234. return list_dt_picdata;
  235. }
  236. public Mat DrawingOfNationalStandardBoxes(DataTable GBDatatableOne,int GridLength, Mat save_pano)
  237. {
  238. // 创建一个HashSet来存储唯一的值
  239. HashSet<int> distinctValuesY = new HashSet<int>();
  240. // 创建一个HashSet来存储唯一的值
  241. HashSet<int> distinctValuesX = new HashSet<int>();
  242. for (int i = 0; i < GBDatatableOne.Rows.Count; i++)
  243. {
  244. distinctValuesY.Add(Convert.ToInt32(GBDatatableOne.Rows[i]["fieldY"]));
  245. distinctValuesX.Add(Convert.ToInt32(GBDatatableOne.Rows[i]["fieldX"]));
  246. }
  247. //OTS坐标向上为正,向右为正
  248. var distinctY = distinctValuesY.ToList();
  249. distinctY.Sort((a, b) => b.CompareTo(a));
  250. var distinctX = distinctValuesX.ToList();
  251. distinctX.Sort();
  252. //取到Y轴图片的最上边的位置
  253. picYmin = picYmin + (imgeH / 2);
  254. int GBY = distinctY.Max() + (GridLength / 2);
  255. int Yzhou = (int)picYmin - GBY;
  256. picXmin = picXmin - (imgeW / 2);
  257. int GBX = distinctX.Min() - (GridLength / 2);
  258. for (int i = 0; i < distinctY.Count; i++)
  259. {
  260. int Xzhou = (int)picXmin - GBX;
  261. DataTable number = CountTheNumberOfX_axes(GBDatatableOne, "fieldY", distinctY[i].ToString());
  262. for (int j = 0; j < number.Rows.Count; j++)
  263. {
  264. // 定义矩形的左上角和右下角坐标
  265. OpenCvSharp.Point topLeft = new OpenCvSharp.Point(Xzhou, Yzhou);
  266. int XzhouDOW = Xzhou + GridLength;
  267. int YzhouDOW = Yzhou + GridLength;
  268. OpenCvSharp.Point textOrg = new OpenCvSharp.Point(Xzhou + 20, Yzhou + 70); // 文字开始的位置
  269. OpenCvSharp.Point bottomRight = new OpenCvSharp.Point(XzhouDOW, YzhouDOW);
  270. Xzhou = Xzhou + GridLength;
  271. // 定义矩形的颜色(BGR)和线宽
  272. Scalar color = new OpenCvSharp.Scalar(0, 0, 255, 255); // 蓝色
  273. int thickness = 2;
  274. // 在图像上绘制矩形
  275. Cv2.Rectangle(save_pano, topLeft, bottomRight, color, thickness);
  276. // 设置要绘制的文字及其位置
  277. string text = number.Rows[j]["data"].ToString();
  278. string input = number.Rows[j]["data"].ToString();
  279. double fontScale = 1; // 字体大小缩放因子
  280. int thickness2 = 1; // 字体粗细
  281. // 获取文字的大小,以便我们可以更好地定位它
  282. OpenCvSharp.Size textSize = OpenCvSharp.Cv2.GetTextSize(input, HersheyFonts.HersheySimplex, fontScale, thickness2, out int baseline);
  283. int chunkSize = GridLength / textSize.Height; // 分割长度
  284. // 使用LINQ的Enumerable.Chunk扩展方法分割字符串
  285. var chunks = Enumerable.Range(0, input.Length / chunkSize + (input.Length % chunkSize > 0 ? 1 : 0))
  286. .Select(x => input.Substring(x * chunkSize, Math.Min(chunkSize, input.Length - x * chunkSize)))
  287. .ToList();
  288. int topgrade = Convert.ToInt32(number.Rows[j]["color"]);
  289. // 设置字体样式和大小
  290. Color color1 = new Color();
  291. switch (topgrade)
  292. {
  293. case 0:
  294. color1 = Color.White;
  295. break;
  296. case 1:
  297. color1 = Color.DarkGray;
  298. break;
  299. case 2:
  300. color1 = Color.Purple;
  301. break;
  302. case 3:
  303. color1 = Color.Blue;
  304. break;
  305. case 4:
  306. color1 = Color.Red;
  307. break;
  308. case 5:
  309. color1 = Color.Red;
  310. break;
  311. default:
  312. color1 = Color.Red;
  313. break;
  314. }
  315. Scalar fontColor = new OpenCvSharp.Scalar(color1.B, color1.G, color1.R, 255); // 文字颜色:黑色
  316. LineTypes lineType = OpenCvSharp.LineTypes.AntiAlias; // 线条类型
  317. foreach (var chunk in chunks)
  318. {
  319. // 使用PutText方法在图像上绘制文字
  320. Cv2.PutText(save_pano, chunk, textOrg, HersheyFonts.HersheySimplex, fontScale, fontColor, thickness, lineType);
  321. textOrg.Y = textOrg.Y + 70;
  322. }
  323. }
  324. Yzhou = Yzhou + GridLength;
  325. }
  326. return save_pano;
  327. }
  328. private DataTable CountTheNumberOfX_axes(DataTable dataTable, string ColumnName, string name)
  329. {
  330. DataTable data = new DataTable();
  331. data.Columns.Add("data");
  332. data.Columns.Add("color");
  333. for (int i = 0; i < dataTable.Rows.Count; i++)
  334. {
  335. if (name == dataTable.Rows[i][ColumnName].ToString())
  336. {
  337. DataRow dr = data.NewRow();
  338. dr["data"] = dataTable.Rows[i]["gradeDetail"].ToString();
  339. dr["color"] = dataTable.Rows[i]["topGrade"].ToString();
  340. data.Rows.Add(dr);
  341. }
  342. }
  343. return data;
  344. }
  345. public static Bitmap ParticleBinaryDiagram(DataTable resultFile, int PICSerialNumber,int SerialNumber)
  346. {
  347. DataTable PICData = resultFile.Clone();
  348. for (int i=0;i< resultFile.Rows.Count;i++)
  349. {
  350. if (Convert.ToInt32(resultFile.Rows[i]["FieldId"]) == PICSerialNumber)
  351. {
  352. PICData.Rows.Add(resultFile.Rows[i].ItemArray);
  353. }
  354. }
  355. DataTable SerialData = PICData.Clone();
  356. for (int i=0;i< PICData.Rows.Count;i++)
  357. {
  358. if (Convert.ToInt32(PICData.Rows[i]["ParticleId"])== SerialNumber)
  359. {
  360. SerialData.Rows.Add(PICData.Rows[i].ItemArray);
  361. }
  362. }
  363. HashSet<int> hsINT = new HashSet<int>();
  364. int left = int.MaxValue;
  365. int wide = 0;
  366. int Ht = int.MaxValue;
  367. List<int> vs = new List<int>();
  368. List<int> HList = new List<int>();
  369. for (int i = 0; i < SerialData.Rows.Count; i++)
  370. {
  371. if (left >Convert.ToInt32(SerialData.Rows[i]["Start"]))
  372. {
  373. left = Convert.ToInt32(SerialData.Rows[i]["Start"]);
  374. }
  375. vs.Add(Convert.ToInt32(SerialData.Rows[i]["Start"]) + Convert.ToInt32(SerialData.Rows[i]["Length"]));
  376. hsINT.Add(Convert.ToInt32(SerialData.Rows[i]["Height"]));
  377. HList.Add(Convert.ToInt32(SerialData.Rows[i]["Height"]));
  378. }
  379. for (int i=0;i< vs.Count;i++)
  380. {
  381. vs[i] = vs[i] - left;
  382. if (wide< vs[i])
  383. {
  384. wide = vs[i];
  385. }
  386. }
  387. for (int i=0;i< HList.Count;i++)
  388. {
  389. if (Ht > HList[i])
  390. {
  391. Ht = HList[i];
  392. }
  393. }
  394. Bitmap bitmap = new Bitmap(wide+20, hsINT.Count+30);
  395. Graphics graphics = Graphics.FromImage(bitmap);
  396. graphics.Clear(Color.White);
  397. //再循环取出里面所有的segment
  398. for (int i=0;i< SerialData.Rows.Count;i++)
  399. {
  400. #region 创建DSegment对象,并将STD分析出的化合物颜色保存到DSegment对象中
  401. Rectangle rectangle = new Rectangle(Convert.ToInt32(SerialData.Rows[i]["Start"]) - left + 10, Convert.ToInt32(SerialData.Rows[i]["Height"]) - Ht + 10, Convert.ToInt32(SerialData.Rows[i]["Length"]),2);
  402. graphics.FillRectangle(Brushes.Black, rectangle);
  403. #endregion
  404. }
  405. return bitmap;
  406. }
  407. }
  408. }