К нам в техподдержку часто приходит вопрос: «Как посчитать суммы длин отрезков (участков трубопровода, элементов электрических схем и т.п.) в чертеже?». Существует масса способов решения этой задачи, в сегодняшней публикации мы рассмотрим реализацию приложения на MultiCAD.NET API, суммирующего длины, которое работает в nanoCAD, AutoCAD и ZWCAD. В качестве примера мы возьмем задачу определения суммарной длины труб в схеме водоснабжения и рассмотрим два варианта выбора элементов для подсчета: пользовательский и по созданному фильтру.
Определение суммы длин отрезков, выбранных пользователем
Прежде чем приступить к определению длины отрезка, необходимо определить, что же такое отрезок в MultiCAD.NET. Отрезок является стандартным примитивом наряду с окружностью, текстом, сплайном и др. Для представления отрезка в базе данных чертежа используется класс DbLine из пространства имен всех примитивов Multicad.DatabaseServices.StandardObjects.
Объекты DbLine в качестве свойств содержат начальную и конечную точку, но не содержат информации о длине отрезка. Конечно, координат точек отрезка достаточно для вычисления его длины, но удобнее будет использовать его геометрическое представление — объект класса LineSeg3d (доступ к которому обеспечивает свойство DbLine.Line) и его свойство Length для получения длины:
double length = line.Line.Length;
Итак, рассмотрим первый вариант приложения, когда пользователю предлагается самостоятельно выбрать отрезки для вычисления итогового значения длины. Для реализации пользовательского выбора объекта будет использоваться метод SelectObjects класса менеджера объектов McObjectManager:
public static McObjectId[] SelectObjects(string sPromt);
Метод выводит подсказку в консоль и позволяет пользователю самому выбирать объекты, ID выбранных объектов записываются в массив. Затем производится распознавание элементов массива, и для объектов, которые являются отрезками, получаем длину и инкрементируем результат. Общий вид команды, реализующий эту процедуру:
CommandMethod("LineLengthSum", CommandFlags.NoCheck | CommandFlags.NoPrefix)] public void LineLengthSum() { // Получаем объекты выбором на чертеже McObjectId[] idSelecteds = McObjectManager.SelectObjects("Выберите объекты типа линия") if (idSelecteds.Length == 0) { MessageBox.Show("Объекты не выбраны"); return; } double lengthSum = 0; foreach (McObjectId currID in idSelecteds) { // Получаем объект по ID McObject currObj = currID.GetObject(); // Распознавание типа объекта if (currObj is DbLine) { lengthSum += (currObj as DbLine).Line.Length; } } MessageBox.Show(lengthSum.ToString(), "Длина всех выбранных отрезков:", MessageBoxButtons.OK, MessageBoxIcon.Information); }
Кроме отрезков прямых на чертежах используются полилинии, которые представляют собой совокупность отрезков и/или дуговых элементов. Получить длину полилинии можно аналогично, использую геометрическое представление примитива — класс Polyline3dчерез свойство Polyline:
double length = polyline.Polyline.Length;
Автоматический подсчет суммарной длины линий
На практике, когда чертеж содержит большое число элементов и требуется исключить ошибки пользовательского ввода, объекты могут быть выбраны автоматически, используя фильтр объектов. При создании фильтра указываются необходимые критерии: область выбора объектов (листы, слои, документы, область, и пр.) и типы объектов.
Например, для того, чтобы выбрать все линии на конкретном слое используется фильтр с указанием имени слоя:
ObjectFilter filter = ObjectFilter.Create(true); filter.AddLayer("Холодная вода"); filter.AddType(typeof(DbLine)); List<McObjectId> ids = filter.GetObjects();
Наиболее часто встречающийся на практике пример применения автоматического подсчета суммарной длины линий — формирование отчета по типу труб на схеме водоснабжения.
В нашем примере схема трубопровода организована таким образом, что трубы каждого типа расположены на отдельных слоях: «Контур 1» и «Контур 2».
Следующая команда формирует текстовый отчет с указанием всех типов труб, расположенных на отдельных слоях и их суммарной длины.
[CommandMethod("createReport", CommandFlags.NoCheck | CommandFlags.NoPrefix)] public void createReport() { List<String> reportStrings = getLengthSumByLayer(); if (reportStrings.Count == 0) { MessageBox.Show("Схема не содержит элементов линий и полилиний"); return; } // Отступ для вывода строки отчета int indent = 0; // Заголовок отчета DbText caption = new DbText(); caption.Text = new TextGeom("Длина труб по типу", new Point3d(0, indent, 0), Vector3d.XAxis, "Standard", 10); caption.DbEntity.AddToCurrentDocument(); foreach (String str in reportStrings) { indent -= 10; DbText reportText = new DbText(); reportText.Text = new TextGeom(str, new Point3d(0, indent, 0), Vector3d.XAxis, "Standard", 6); reportText.DbEntity.AddToCurrentDocument(); } }
Подсчет суммарной длины и заполнение строк отчета производится в методе getLengthSumByLayer(), код которого представлен ниже:
public List<String> getLengthSumByLayer() { List<String> reportStrings = new List<String>(); // Получаем все слои на чертеже List<string> layers = McObjectManager.CurrentStyle.GetLayers(); foreach (string layerName in layers) { ObjectFilter filter = ObjectFilter.Create(true).AddType(typeof(DbLine)).AddType(typeof(DbPolyline)).AddLayer(layerName); List<McObjectId> idSelected = filter.GetObjects(); if (idSelected.Count != 0) { double lengthSum = 0; foreach (McObjectId currID in idSelected) { // Получаем объект по ID McObject currObj = currID.GetObject(); // Распознавание типа объекта if (currObj is DbLine) { lengthSum += (currObj as DbLine).Line.Length; } else if (currObj is DbPolyline) { lengthSum += (currObj as DbPolyline).Polyline.Length; } } // Если суммарная длина отрезков и полилиний на слое ненулевая, то добавляем в текст отчета if (lengthSum != 0) { reportStrings.Add(layerName.ToString() + ": " + lengthSum.ToString()); } } } return reportStrings; }
После выполнения данной команды на чертеж будет добавлен отчет вида:
Подробную процедуру загрузки MultiCAD.NET приложений вы можете найти в нашей статье Пошаговый обзор: единое MultiCAD.NET приложение в nanoCAD, AutoCAD, ZWCAD
Обсудить статью можно также и на нашем форуме.
Комментарии