Маркировка пространств на разрезах
Сегодня мы поговорим о маркировке пространств на разрезах.
Для создания марок пространств в Revit API предназначен метод NewSpaceTag класса Autodesk.Revit.Creation.Document. Сигнатура метода следующая:
- public SpaceTag NewSpaceTag(Space space, UV point, View view)
В контексте рассматриваемого в этой статье вопроса нас больше всего интересует второй параметр метода. SDK нам говорит следующее:
Type: Autodesk.Revit.DB.UV
A 2D point that dictates the location on the level of the space.
Т.е. это точка на плоскости уровня, на котором расположено пространство. Действительно, нет никаких проблем создать марку пространства на плане используя следующий код:
- ...
- var spaceLocation = (LocationPoint) space.Location;
- doc.Create
- .NewSpaceTag(space, new UV(spaceLocation.Point.X, spaceLocation.Point.Y), doc.ActiveView);
- ...
Location пространства задается пользователем, на виде в плане он показывается пересечением линий с крестиком. Именно в этом месте и создается марка с помощью кода, показанного выше.
А что же насчет разрезов? А здесь возникают проблемы. Во-первых, за счет того, что метод принимает аргументом точку на плоскости, то в свойстве TagHeadPosition координата Z будет нулевой, и марка в общем случае появится далеко от маркируемого пространства. Во-вторых пространство на разрезе имеет собственный "прицел", который в общем случае не совпадает со Space.Location, а марку хотелось бы привязать именно к пересечению этих линий.
Свойство TagHeadPosition имеет сеттер, т.е. после создания марки мы можем переместить её в нужную позицию. Осталось понять, как получить линии, образующие "прицел".
Для этого нужно получить геометрию текущего вида, в т.ч. невидимую (установив IncludeNonVisibleObjects = true), из этой геометрии получить линии и проверить категорию линии в графическом стиле как показано на этой схеме с помощью утилиты Revit Lookup:
- private static XYZ GetSpaceTagPositionOnSectionView(Space space, ViewSection viewSection)
- {
- var spaceReferenceLines = space.get_Geometry(new Options { View = viewSection, IncludeNonVisibleObjects = true })
- .OfType<Line>()
- .Where(x => IsSpaceReferenceLine(x, space.Document))
- .ToList();
- if (spaceReferenceLines.Count != 2)
- throw new InvalidOperationException("Что-то пошло не так");
- IntersectionResultArray intersectionResult;
- if (spaceReferenceLines[0].Intersect(spaceReferenceLines[1], out intersectionResult) != SetComparisonResult.Overlap)
- throw new InvalidOperationException("Что-то пошло не так");
- return intersectionResult.get_Item(0).XYZPoint;
- }
- private static bool IsSpaceReferenceLine(Line line, Document document)
- {
- var referenceLineStyleCategoryId = new ElementId(BuiltInCategory.OST_MEPSpaceReference);
- var graphicsStyle = document.GetElement(line.GraphicsStyleId) as GraphicsStyle;
- return graphicsStyle?.GraphicsStyleCategory.Id == referenceLineStyleCategoryId;
- }
Теперь осталось только присвоить рассчитанное значение свойству TagHeadPosition после создания марки:
- ...
- var tag = doc.Create
- .NewSpaceTag(space, UV.Zero, doc.ActiveView);
- tag.TagHeadPosition = GetSpaceTagPositionOnSectionView(space, (ViewSection) doc.ActiveView);
- ...
Обсуждение: http://adn-cis.org/forum/index.php?topic=
Опубликовано 12.11.2018