ElementRangeCheck.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. /* ==============================================================================
  2. * 功能描述:构件范围检查
  3. * 创 建 者:Garrett
  4. * 创建日期:2018/10/23 15:08:55
  5. * ==============================================================================*/
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Text.RegularExpressions;
  10. using Autodesk.Revit.DB;
  11. using Autodesk.Revit.DB.Mechanical;
  12. using NPOI.SS.UserModel;
  13. using SAGA.DotNetUtils.Extend;
  14. using SAGA.DotNetUtils.Others;
  15. using SAGA.RevitUtils.Extends;
  16. namespace Saga.PlugIn.ModelCheck
  17. {
  18. /// <summary>
  19. ///
  20. /// </summary>
  21. [ParseIndex(Index = 2)]
  22. class ElementRangeCheck : ModeCheckBase
  23. {
  24. public ElementRangeCheck()
  25. {
  26. Name = "构件范围检查";
  27. }
  28. public override bool Check()
  29. {
  30. if (!RBase.IsRight)
  31. {
  32. IsRight = RBase.IsRight;
  33. return IsRight;
  34. }
  35. IsRight = GetCheckResult();
  36. return IsRight;
  37. }
  38. #region CheckMethod
  39. private bool GetCheckResult()
  40. {
  41. bool unitResult = true;
  42. foreach (SagaSignCheckResult signResult in RBase.Results)
  43. {
  44. CheckCurrentFloor(signResult);
  45. }
  46. return Results.All(t => t.IsRight);
  47. }
  48. /// <summary>
  49. /// 检查当前层
  50. /// </summary>
  51. /// <param name="signResult"></param>
  52. private void CheckCurrentFloor(SagaSignCheckResult signResult)
  53. {
  54. var document = signResult.RDocument;
  55. if (SetFloorBaseTopRange(document, signResult))
  56. {
  57. var elements = document.GetAllElements();
  58. foreach (var element in elements)
  59. {
  60. var result = GetCheckResult(signResult, element);
  61. if (result != null)
  62. Results.Add(result);
  63. }
  64. }
  65. CalcRCPassRate(signResult);
  66. }
  67. /// <summary>
  68. /// 计算本层构件范围检查通过率
  69. /// </summary>
  70. private void CalcRCPassRate(SagaSignCheckResult signResult)
  71. {
  72. Func<Func<ModeCheckResultBase, bool>, double> rateFunc = (condition) =>
  73. {
  74. double passRate = 1;
  75. var items = Results.Where(condition);
  76. int count = items.Count();
  77. if (count > 0)
  78. {
  79. var passItems = items.Where(t => t.IsRight);
  80. var passCount = passItems.Count();
  81. passRate = (1.0 * passCount / count);
  82. }
  83. return passRate;
  84. };
  85. signResult.RCPassRate = rateFunc((t) => t.RBase == signResult);
  86. signResult.ColumnWallPassRate = rateFunc((t) => new List<DCElementType>() { DCElementType.Wall, DCElementType.Column }.Contains(((ElementRangeCheckResult)t).RType));
  87. signResult.SpacePassRate = rateFunc((t) => ((ElementRangeCheckResult)t).RType == DCElementType.Space);
  88. signResult.InstPassRate = rateFunc((t) => new List<DCElementType>() { DCElementType.Equipment, DCElementType.EuipmentPart, DCElementType.Beacon }.Contains(((ElementRangeCheckResult)t).RType));
  89. }
  90. private double m_HTop = 0;
  91. private double m_HBase = 0;
  92. private List<double> m_HSandwishs=new List<double>();
  93. /// <summary>
  94. /// 设置当前楼层底部和顶部范围
  95. /// </summary>
  96. /// <param name="doc"></param>
  97. /// <param name="result"></param>
  98. private bool SetFloorBaseTopRange(Document doc, SagaSignCheckResult result)
  99. {
  100. bool rt = true;
  101. try
  102. {
  103. m_HBase = result.HBase;
  104. m_HSandwishs = new List<double>();
  105. //设置楼层底部高度
  106. var curLevelName = result.RPlanName;
  107. var levels = doc.GetLevels();
  108. var mbiLevels = levels.Where(t => Regex.IsMatch(t.Name, $"{ModelCheckConst.IsMBILevel}")).ToList();
  109. Func<string, bool> isSandwichFunc = (str) => { return Regex.IsMatch(str, $"{ModelCheckConst.IsSandwich}");};
  110. //设置楼层顶部高度
  111. if (curLevelName == "RF")
  112. {
  113. m_HTop = double.MaxValue;
  114. }
  115. else
  116. {
  117. var hightThanBaseLevels=mbiLevels.Where(t => t.Elevation.IsThan(m_HBase));
  118. foreach (Level level in hightThanBaseLevels)
  119. {
  120. m_HTop = level.Elevation;
  121. if(isSandwichFunc(level.Name))
  122. m_HSandwishs.Add(level.Elevation);
  123. else
  124. {
  125. break;
  126. }
  127. }
  128. }
  129. }
  130. catch (Exception e)
  131. {
  132. Console.WriteLine(e);
  133. rt = false;
  134. }
  135. return rt;
  136. }
  137. /// <summary>
  138. /// 获取检测结果
  139. /// </summary>
  140. /// <param name="element"></param>
  141. /// <returns></returns>
  142. private ModeCheckResultBase GetCheckResult(SagaSignCheckResult signResult, Element element)
  143. {
  144. var result = new ElementRangeCheckResult();
  145. result.RBase = signResult;
  146. result.FamilyName = element.GetFamily()?.Name;
  147. result.Id = element.Id.ToString();
  148. try
  149. {
  150. double hb = signResult.HBase;
  151. double ht = m_HTop;
  152. //使用时,单位转化为英寸
  153. double w = signResult.Redundant.ToApi();
  154. double zb = 0, zt = 0;
  155. bool rb = false, rt = false;
  156. bool isBoxInst = false;
  157. if (element is Wall wall)
  158. {
  159. isBoxInst = true;
  160. result.RType = DCElementType.Wall;
  161. zb = wall.GetBaseStaticHeight();
  162. zt = wall.GetTopStaticHeight();
  163. }
  164. else if (element is Space space)
  165. {
  166. isBoxInst = true;
  167. result.RType = DCElementType.Space;
  168. zb = space.GetBaseStaticHeight();
  169. zt = space.GetTopStaticHeight();
  170. }
  171. else if(element is FamilyInstance)
  172. {
  173. if (element.IsAllColumn())
  174. {
  175. var fi = element as FamilyInstance;
  176. isBoxInst = true;
  177. result.RType = DCElementType.Column;
  178. zb = fi.GetBaseStaticHeight();
  179. zt = fi.GetTopStaticHeight();
  180. }
  181. else if (element.IsEquipment() || element.IsEquipmentPart() || element.IsBeacon())
  182. {
  183. result.RType = GetRType(element);
  184. zb = element.GetLocationPointMBIXYZ().Z;
  185. rb = zb.IsBetween(hb, ht);
  186. result.IsRight = rb;
  187. result.RMessage = result.IsRight ? "" : "构件范围不满足要求;请检查构件位置";
  188. }
  189. else
  190. {
  191. result = null;
  192. }
  193. }
  194. else
  195. {
  196. result = null;
  197. }
  198. if (isBoxInst)
  199. {
  200. rb = zb.IsBetween(hb - w, hb);
  201. rt = zt.IsBetween(ht - w, ht) || m_HSandwishs.Any(t=>zt.IsBetween(t-w,t));
  202. result.IsRight = rb && rt;
  203. string ttip = rb ? "" : "底部";
  204. string ttop = rt ? "" : rb ? "和顶部" : "顶部";
  205. result.RMessage = result.IsRight ? "" : $"构件范围不满足要求;请检查构件{ttip}{ttop}";
  206. }
  207. }
  208. catch (Exception e)
  209. {
  210. //MessageShowBase.Show(e);
  211. if (result != null)
  212. {
  213. result.IsRight = false;
  214. result.RMessage = result.IsRight ? "" : "构件范围不满足要求;请检查构件底部和顶部";
  215. }
  216. }
  217. return result;
  218. }
  219. /// <summary>
  220. /// 获取构件类型
  221. /// </summary>
  222. /// <param name="fi"></param>
  223. /// <returns></returns>
  224. private DCElementType GetRType(Element fi)
  225. {
  226. DCElementType name = DCElementType.None;
  227. if (fi.IsEquipment())
  228. {
  229. name = DCElementType.Equipment;
  230. }
  231. else if (fi.IsEquipmentPart())
  232. {
  233. name = DCElementType.EuipmentPart;
  234. }
  235. else if (fi.IsBeacon())
  236. {
  237. name = DCElementType.Beacon;
  238. }
  239. return name;
  240. }
  241. #endregion
  242. //[DataCheckProcessAspect]
  243. public override void Export()
  244. {
  245. try
  246. {
  247. IWorkbook book = DCRExport.GetWorkbook();
  248. //ISheet sheet = book.CreateSheet(Name);
  249. ISheet sheet = book.GetSheet(Name);
  250. #region 添加数据
  251. int index = 4;
  252. //添加 共检查XXX条数据,未通过检查的如下 提示
  253. IRow rowTip = sheet.CreateRow(index - 1);
  254. rowTip.AddCell(0, $"总检查{Results.Count}条数据,未通过检查的如下", DataCheckNPOIStyle.Title);
  255. foreach (ElementRangeCheckResult result in Results)
  256. {
  257. SagaSignCheckResult rbase = result.RBase as SagaSignCheckResult;
  258. if (rbase == null)
  259. continue;
  260. //数量过多,只显示有问题的
  261. if (result.IsRight) continue;
  262. index++;
  263. IRow rowN = sheet.CreateRow(index);
  264. DataCheckNPOIStyle style = result.IsRight ? DataCheckNPOIStyle.Content : DataCheckNPOIStyle.Error;
  265. int j = -1;
  266. rowN.AddCell(++j, result.FamilyName, style);
  267. rowN.AddCell(++j, result.RType.GetDescription(), style);
  268. rowN.AddCell(++j, result.Id, style);
  269. string rowN4 = result.IsRight ? "通过" : "不通过";
  270. rowN.AddCell(++j, rowN4, style);
  271. rowN.AddCell(++j, result.RMessage, style);
  272. }
  273. #endregion
  274. }
  275. catch (Exception e)
  276. {
  277. MessageShowBase.Show(e);
  278. }
  279. }
  280. }
  281. /// <summary>
  282. /// UnitCheckResult
  283. /// </summary>
  284. class ElementRangeCheckResult : ModeCheckResultBase
  285. {
  286. public string FamilyName { get; set; }
  287. public string Id { get; set; }
  288. /// <summary>
  289. /// 构件类型
  290. /// </summary>
  291. public DCElementType RType { get; set; }
  292. }
  293. }