/*------------------------------------------------------------------------- * 功能描述:RevitUtil * 作者:xulisong * 创建时间: 2019/6/24 9:55:09 * 版本号:v1.0 * -------------------------------------------------------------------------*/ using Autodesk.Revit.DB; using SAGA.RevitUtils.Extends; using SAGA.RevitUtils.Extends.Graphic; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using Autodesk.Revit.DB.Mechanical; using RevitToJBim.Extension; using RevitToJBim.MBI; using SAGA.DotNetUtils; using Parameter = JBIM.Component.Parameter; using SAGA.RevitUtils; using SAGA.DotNetUtils.Extend; using System.Configuration; namespace RevitToJBim.Common { public class RevitUtil { /// /// 获取FamilyInstance顶面轮廓 /// /// /// public static List GetTopPolygon(FamilyInstance fi) { List path = new List(); var faces = fi.GetOriginalFaces(XYZ.BasisZ); var topface = faces.FirstOrDefault(); //传递的path中没有重复点 if (topface != null) { var curves = topface.GetCurves(); for (int i = 0; i < curves.Count; i++) { var current = curves[i]; var points = current.GetPoints(); points.RemoveAt(points.Count - 1); path.AddRange(points); } } return path; } /// /// 获取FamilyInstance box底面轮廓 /// /// /// public static List GetBottomPolygon(FamilyInstance fi) { List path = new List(); try { var box = fi.get_BoundingBox(null); if (box == null) return path; var boxMin = box.Min; var boxMax = box.Max; path.Add(boxMin); path.Add(boxMin.NewX(boxMax.X)); path.Add(boxMin.NewX(boxMax.X).NewY(boxMax.Y)); path.Add(boxMin.NewY(boxMax.Y)); path.Add(boxMin); } catch (Exception e) { Console.WriteLine(e); } return path; } /// /// 获取参数 /// /// /// /// /// public static List GetParameters(Element ele, List parameterNames, bool hasEquipComponentCode = false) { List parameters = new List(); foreach (var parameterName in parameterNames) { var revitParameter = ele.GetParameter(parameterName); if (revitParameter != null) { var parameter = new Parameter(ParameterUtil.FindParameterDefine(parameterName)); parameter.Value = revitParameter.AsString(); parameters.Add(parameter); } } if (hasEquipComponentCode) { var equipComponentCode = ConfigurationManager.AppSettings["EquipComponentCode"]; var typeParameter = ele?.GetElementType()?.Parameters; foreach (Autodesk.Revit.DB.Parameter p in typeParameter) { if (equipComponentCode.Equals(p.Definition.Name)) { if (!p.HasValue || (p.AsString() == null && p.AsString().Length == 0)) continue; var parameter = new Parameter(ParameterUtil.FindParameterDefine(equipComponentCode)); parameter.Value = p.AsString(); parameters.Add(parameter); } } } return parameters; } /// /// 获取设备设施的参数 /// /// /// public static List GetFacilityParameters(FamilyInstance fi) { List parameterNames = ConfigurationManager.AppSettings["FacilityParameters"].Split(',').ToList(); List parameters = GetParameters(fi, parameterNames, true); return parameters; } /// /// 获取空间的参数 /// /// /// public static List GetSpaceParameters(Element space) { List parameterNames = ConfigurationManager.AppSettings["SpaceParameters"].Split(',').ToList(); List parameters = GetParameters(space, parameterNames, false); return parameters; } /// /// 获取房间的参数 /// /// /// public static List GetRoomParameters(Element room) { List parameterNames = ConfigurationManager.AppSettings["RoomParameters"].Split(',').ToList(); List parameters = GetParameters(room, parameterNames, false); return parameters; } /// /// 获取设备设施的参数 /// /// /// public static List GetMEPCurveParameters(MEPCurve curve) { List parameterNames = ConfigurationManager.AppSettings["MEPCurveParameters"].Split(',').ToList(); List parameters = GetParameters(curve, parameterNames, false); return parameters; } /// /// 获取部件所关联的设备 /// /// /// public static Element GetEquipPartParent(Element element) { if (!element.IsEquipmentPart()) return null; string code = element.GetFamilyName()?.Substring(0, 4); ; if (code.IsNullOrEmpty()) return null; //构件所关联的设备 var parentInst = element.Document.GetElements(new ElementIntersectsElementFilter(element)) .FirstOrDefault(t => !t.Id.IsEqual(element.Id) && t.GetFamilyCode() == code); return parentInst; } public static ElementType GetElementType(Element element) { ElementId id = element.GetTypeId(); if (id != null && id != ElementId.InvalidElementId) { Element elementType = element.Document.GetElement(id); return elementType as ElementType; } return null; } /// /// 获取门窗到墙线上的投影线 /// /// /// public static List GetWindowDoorLocation(FamilyInstance fi) { //取几何实体顶点信息,将点到指定方向做投影 List xyzs = GetVertexPoints(fi); XYZ handDirection = (fi.Host as Wall)?.Orientation; if (handDirection != null) { handDirection = handDirection.CrossProduct(XYZ.BasisZ); } else if (handDirection == null) { handDirection = fi.HandOrientation; } var location = fi.GetLocationPoint(); if (location == null) { location = xyzs.FirstOrDefault(); } double min = 0, max = 0; XYZ minXYZ = XYZ.Zero, maxXYZ = XYZ.Zero; for (int i = 0; i < xyzs.Count; i++) { var usePoint = xyzs[i]; var refDirection = usePoint - location; var offset = refDirection.DotProduct(handDirection); var projectPoint = location.OffsetPoint(handDirection, offset); #region 初始化逻辑 if (i == 0) { minXYZ = maxXYZ = projectPoint; min = max = offset; } #endregion #region 判定逻辑 else { if (offset < min) { minXYZ = projectPoint; min = offset; continue; } if (max < offset) { maxXYZ = projectPoint; max = offset; continue; } } #endregion } return new List() { minXYZ, maxXYZ }; } /// /// 获取元素顶点集合 /// /// public static List GetVertexPoints(Element element) { List xyzs = new List(); //string id = element.Id.ToString(); //var solids = Extension.GeometryElementExtension.GetSolids(element, element.Document.GetUseView()); //foreach (var solid in solids) //{ // foreach (Edge edge in solid.Edges) // { // xyzs.AddRange(edge.Tessellate()); // } //} var curves = Extension.GeometryElementExtension.GetGeometryObjects(element, element.Document.GetUseView(), true); foreach (var curve in curves) { xyzs.AddRange(curve.Tessellate()); } return xyzs.Distinct(new XyzEqualComparer()).ToList(); } /// /// 获取墙的轮廓线 /// /// /// public static List> GetWallPolygon(Wall wall) { /* 方案1: * 1找到墙的定位线, * 2将定位线拆分成多段 * 3按照宽度,将多段定位线生成闭合区域 * 方案2: * 直接去墙的顶面,过滤相关face形成轮廓 */ List> polygons = new List>(); #region 过滤墙的顶面 List faces = wall.GetBottomFaces(); #endregion foreach (PlanarFace face in faces) { foreach (CurveLoop curveLoop in face.GetEdgesAsCurveLoops()) { if (curveLoop == null) { continue; } var polygon = curveLoop.GetPolygon(); if (polygon.Any()) { polygons.Add(polygon); } } } return polygons; } } }