start

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; using WinForm = System.Windows.Forms; using Autodesk.Revit.UI;
using Autodesk.Revit.DB;
using Autodesk.Revit.Attributes; using Autodesk.Revit.DB.Mechanical;
using Autodesk.Revit.UI.Selection;
using Autodesk.Revit.ApplicationServices; using Autodesk.Revit.DB.Structure;
using Autodesk.Revit.DB.ExtensibleStorage;
using Autodesk.Revit.DB.Plumbing;
using Autodesk.Revit.DB.Architecture; using System.Xml;
using SelSet = HongYe.Revit.Public.SelectSet;
using Autodesk.Revit.DB.Electrical;
using System.Diagnostics;
using System.Security.Cryptography;
using System.IO; namespace RevitCodes
{
    /// <summary>
    /// 工具类
    /// </summary>
    public class RevitTool
    {
        /// <summary>
        /// 管件是否连接风管,如果什么也没连接也返回true
        /// </summary>
        /// <returns></returns>
        public static bool IsConnectToDuct(FamilyInstance fi)
        {
            bool bResult = true;
            ConnectorSetIterator csi = fi.MEPModel.ConnectorManager.Connectors.ForwardIterator();
            while (csi.MoveNext())
            {
                Connector conn = csi.Current as Connector;
                if (conn.IsConnected == true)//是否有连接
                {
                    ConnectorSet connectorSet = conn.AllRefs;//找到所有连接器连接的连接器
                    ConnectorSetIterator csiChild = connectorSet.ForwardIterator();
                    while (csiChild.MoveNext())
                    {
                        Connector connected = csiChild.Current as Connector;
                        if (null != connected && connected.Owner.UniqueId != conn.Owner.UniqueId)
                        {
                            //TaskDialog.Show("conn", conn.Width + "|" + connected.Width + ",");
                            // look for physical connections 
                            if (connected.ConnectorType == ConnectorType.End ||
                                connected.ConnectorType == ConnectorType.Curve ||
                                connected.ConnectorType == ConnectorType.Physical)
                            {
                                //判断是不是管件,只要有一头连接的不是风管就返回false
                                if (connected.Owner is Duct)
                                {
                                    Duct duct = connected.Owner as Duct;
                                    Parameter pWidth = duct.get_Parameter(BuiltInParameter.RBS_CURVE_WIDTH_PARAM);
                                    Parameter pHeight = duct.get_Parameter(BuiltInParameter.RBS_CURVE_HEIGHT_PARAM);
                                    //TaskDialog.Show("width", pWidth.AsValueString());
                                }
                                else
                                {
                                    bResult = false;
                                    break;
                                }
                            }
                        }
                    }
                }
            }
            return bResult;
        }
        /// <summary>
        /// 改变风管尺寸
        /// </summary>
        /// <param name="duct"></param>
        /// <param name="sWidth"></param>
        /// <param name="sHeight"></param>
        public static void ChangeDuctSize(Duct duct, string sWidth, string sHeight)
        {
            Parameter pWidth = duct.get_Parameter(BuiltInParameter.RBS_CURVE_WIDTH_PARAM);
            Parameter pHeight = duct.get_Parameter(BuiltInParameter.RBS_CURVE_HEIGHT_PARAM);
            pWidth.SetValueString(sWidth);
            pHeight.SetValueString(sHeight);
        }
        /// <summary>
        /// 改变管件尺寸
        /// </summary>
        /// <param name="fi"></param>
        /// <param name="dWidth"></param>
        /// <param name="dHeight"></param>
        public static void ChangeFittingSize(FamilyInstance fi, double dWidth, double dHeight)
        {
            ConnectorSetIterator csi = fi.MEPModel.ConnectorManager.Connectors.ForwardIterator();
            while (csi.MoveNext())
            {
                Connector conn = csi.Current as Connector;
                conn.Width = dWidth / 304.8;
                conn.Height = dHeight / 304.8;
            }
        }         /// <summary>
        /// 是否需要变换
        /// </summary>
        /// <param name="fi"></param>
        /// <returns></returns>
        public static bool isChange(FamilyInstance fi)
        {
            bool bChange = true;//是否需要变换宽高
            XYZ xyzFace = GetFittingSolid(fi).Normal;
            //正面朝上或者朝下,不需要变换
            if (xyzFace.AngleTo(new XYZ(, , )) < 0.001 || xyzFace.AngleTo(new XYZ(, , -)) < 0.001)
            {
                bChange = false;
            }
            return bChange;
        }
        /// <summary>
        /// 取得管件的几何体的方向面,管件族的几何体,有三个。
        /// </summary>
        /// <param name="fi"></param>
        /// <returns></returns>
        public static PlanarFace GetFittingSolid(FamilyInstance fi)
        {
            //得到管件尺寸
            double dWidth = , dHeight = ;
            ConnectorSetIterator csi = fi.MEPModel.ConnectorManager.Connectors.ForwardIterator();
            while (csi.MoveNext())
            {
                Connector conn = csi.Current as Connector;
                dWidth = conn.Width;
                dHeight = conn.Height;
            }
            //
            PlanarFace resultFace = null;
            Solid resultSolid = null;
            //
            Options opt = new Options();
            opt.ComputeReferences = true;
            opt.DetailLevel = Autodesk.Revit.DB.DetailLevels.Medium;
            //
            GeometryElement e = fi.get_Geometry(opt);
            foreach (GeometryObject obj in e.Objects)
            {
                GeometryInstance gi = obj as GeometryInstance;
                if (gi != null)
                {
                    GeometryElement ge = gi.GetInstanceGeometry();
                    foreach (GeometryObject go in ge.Objects)
                    {
                        Solid solid = go as Solid;
                        if (solid != null && solid.Faces.Size > )
                        {
                            bool isSolid = true;
                            foreach (Face face in solid.Faces)
                            {
                                PlanarFace pf = face as PlanarFace;
                                if (pf != null)
                                {
                                    if (pf.Area < 0.02)//如果有一个面积特别小,这个截面体。角度为1度也大于0.02
                                    {
                                        isSolid = false;
                                    }
                                }
                            }
                            if (isSolid)
                            {
                                resultSolid = solid;
                            }
                        }
                    }
                }
            }
            //找到截面
            double dArea = dWidth * dHeight;
            PlanarFace sectionFace = null;
            foreach (Face face in resultSolid.Faces)
            {
                if (Math.Abs(face.Area - dArea) < 0.001)//面积近似
                {
                    sectionFace = face as PlanarFace;
                    break;
                }
            }
            //找到与截面垂直的管件正面
            foreach (Face face in resultSolid.Faces)
            {
                PlanarFace pFace = face as PlanarFace;
                if (pFace != null)
                {
                    //面积相同的排除
                    if (Math.Abs(face.Area - dArea) < 0.001)//面积近似
                    {
                        continue;
                    }
                    else
                    {
                        double dAngle = pFace.Normal.AngleTo(sectionFace.Normal);
                        if (Math.Abs(dAngle - Math.PI / ) < 0.001)//与截面垂直的面
                        {
                            resultFace = pFace;
                        }
                    }
                }
            }
            //
            return resultFace;
        }         /// <summary>
        /// 从框选元素中过滤出风管
        /// </summary>
        /// <param name="listElement"></param>
        /// <returns></returns>
        public static List<Duct> GetDuctsFromElements(IList<Element> listElement)
        {
            List<Duct> listDuct = new List<Duct>();
            foreach (Element el in listElement)
            {
                if (el is Duct)
                {
                    Duct duct = el as Duct;
                    listDuct.Add(duct);
                }
            }
            return listDuct;
        }
        /// <summary>
        /// 从框选元素中过滤出管件
        /// </summary>
        /// <param name="listElement"></param>
        /// <returns></returns>
        public static List<FamilyInstance> GetFittingsFromElements(IList<Element> listElement)
        {
            List<FamilyInstance> listFitting = new List<FamilyInstance>();
            foreach (Element el in listElement)
            {
                if (el is FamilyInstance)
                {
                    FamilyInstance fi = el as FamilyInstance;
                    listFitting.Add(fi);
                }
            }
            return listFitting;
        }
    }
    //改变尺寸方法一
    [TransactionAttribute(Autodesk.Revit.Attributes.TransactionMode.Manual)]
    public class cmdChangeSize1 : IExternalCommand
    {
        public Result Execute(ExternalCommandData commandData, ref string messages, ElementSet elements)
        {
            UIApplication app = commandData.Application;
            Document doc = app.ActiveUIDocument.Document;
            Selection sel = app.ActiveUIDocument.Selection;             Transaction ts = new Transaction(doc, "hongye");
            //
            double dWidth = ;
            double dHeight = ;
            IList<Element> listElement = sel.PickElementsByRectangle("框选一个弯头及相连的两根风管");
            List<Duct> listDuct = RevitTool.GetDuctsFromElements(listElement);
            List<FamilyInstance> listFi = RevitTool.GetFittingsFromElements(listElement);
            //第一步,改变管件尺寸,改变后,肯定会增加变径。
            ts.Start();
            foreach (FamilyInstance fi in listFi)
            {
                RevitTool.ChangeFittingSize(fi, dWidth, dHeight);
            }
            ts.Commit();
            //第二步,改变风管尺寸,如果不需要变换,变径会自动消失。
            ts.Start();
            foreach (Duct duct in listDuct)
            {
                RevitTool.ChangeDuctSize(duct, dWidth.ToString(), dHeight.ToString());
            }
            ts.Commit();
            //第三步,判断风管与管件之间是否仍然存在变径,如果存在则宽度互换,再次改变风管尺寸。
            ts.Start();
            foreach (FamilyInstance fi in listFi)
            {
                if (!RevitTool.IsConnectToDuct(fi))//如果连接的是变径
                {
                    foreach (Duct duct in listDuct)
                    {
                        RevitTool.ChangeDuctSize(duct, dHeight.ToString(), dWidth.ToString());
                    }
                }
            }
            ts.Commit();
            //////pf.Normal.AngleTo             return Result.Succeeded;
        }
    }     //改变尺寸方法二
    [TransactionAttribute(Autodesk.Revit.Attributes.TransactionMode.Manual)]
    public class cmdChangeSize2 : IExternalCommand
    {
        public Result Execute(ExternalCommandData commandData, ref string messages, ElementSet elements)
        {
            UIApplication app = commandData.Application;
            Document doc = app.ActiveUIDocument.Document;
            Selection sel = app.ActiveUIDocument.Selection;             Transaction ts = new Transaction(doc, "hongye");
            ts.Start();
            //
            double dWidth = ;
            double dHeight = ;
            IList<Element> listElement = sel.PickElementsByRectangle("框选一个弯头及相连的两根风管");
            List<Duct> listDuct = RevitTool.GetDuctsFromElements(listElement);
            List<FamilyInstance> listFi = RevitTool.GetFittingsFromElements(listElement);
            //
            foreach (FamilyInstance fi in listFi)
            {
                RevitTool.ChangeFittingSize(fi, dWidth, dHeight);
                if (RevitTool.isChange(fi))
                {
                    foreach (Duct duct in listDuct)
                    {
                        RevitTool.ChangeDuctSize(duct, dHeight.ToString(), dWidth.ToString());
                    }
                }
                else
                {
                    foreach (Duct duct in listDuct)
                    {
                        RevitTool.ChangeDuctSize(duct, dWidth.ToString(), dHeight.ToString());
                    }
                }
            }
            ts.Commit();             return Result.Succeeded;
        }     }
}

url:http://greatverve.cnblogs.com/p/duct-fitting-change-size.html

Revit API改变风管及管件尺寸的更多相关文章

  1. Revit API移动风管

    移动风管曲线就可以移动风管 , , ));//向上移动3         ts.Commit();         return Result.Succeeded;     } } url:http: ...

  2. Revit API找到风管穿过的墙(当前文档和链接文档)

    start [Transaction(TransactionMode.Manual)] [Regeneration(RegenerationOption.Manual)] public class c ...

  3. 【Revit API】梁构件支座检查算法

    一.前言         应该是第二次写关于Revit API的博文了.虽然在BIM企业中工作,从事桌面BIM软件开发,但是我是不怎么喜欢写Revit API相关的代码.平时更多的是在写界面展示,架构 ...

  4. Revit API 判断一个构件在某个视图中的可见性

    查看 Revit API.发现有Element::IsHidden这个方法.通过UI创建一个element,注意要使得这个element在某些视图可见,但是在另一些视图不可见.运行下面的方法,你会发现 ...

  5. Revit API 操作共享参数和项目参数

    1.获取共享参数 private string GetSharInfo(Autodesk.Revit.ApplicationServices.Application revitApp) { Strin ...

  6. Revit API射线法读取空间中相交的元素

    Revit API提供根据射线来寻找经过的元素.方法是固定模式,没什么好说.关键代码:doc.FindReferencesWithContextByDirection(ptStart, (ptEnd  ...

  7. Revit API 加载族并生成实例图元

    在Revit API中加载族可以使用Doc.LoadFamily方法,传入要加载的族文件路径名,但是这种方式有一种缺点,就是如果族文件在当前工程中没有加载的话则返回成功,如果已经加载过,则返回失败,也 ...

  8. Revit API风管对齐

    start [Transaction(TransactionMode.Manual)] [Regeneration(RegenerationOption.Manual)] ].Origin, list ...

  9. Revit API画垂直于风管的风管

    start /// <summary> /// 选择风管与风管外一点,画与风管垂直的风管. /// </summary> [Transaction(TransactionMod ...

随机推荐

  1. (FFT)A+B Problem

    题目链接:https://cn.vjudge.net/contest/280041#problem/B 题目大意:给你n个数,然后让你找满足a[i] + a[j] = a[k] 的情况总数. 具体思路 ...

  2. CTEX windedt 打开中文tex乱码问题

    % !TEX encoding = System % !TEX program = pdflatex % !TEX encoding = System% !TEX program = pdflatex ...

  3. 在maven 2工程中加入iTextAsian支持(maven添加自定义jar包到本地仓库)

    最近需要在工程中加入JasperReports,其中要用到把报表导出为pdf文件的功能.JasperReports内部使用iText来输出pdf文档,而iText对中文是放在单独的包iTextAsia ...

  4. WPF 中定时器的使用

    DispatcherTimer timer; private void Window_Loaded(object sender, RoutedEventArgs e) { timer = new Di ...

  5. File /data/binlog/mysql-bin.index' not found (Errcode: 13)

    [问题] 需要开启bin-log备份/恢复数据库,但是因为本身bin-log保存的位置存储太小,并且归类性也不好,所以自己新创建了/data/binlog来保存二进制日志 在/etc/my.cnf增加 ...

  6. 错误/异常:java.net.SocketException: Unrecognized Windows Sockets error: 0: JVM_Bind;的解决方法

    1.异常视图 2.解决方法   SocketException(JVM_Bind):套接口异常     说明:服务器端口号被占用 了  解决方法:点击 Window-->Preferences- ...

  7. Java 基本语法---Java方法

    Java 基本语法---Java方法 0.概述 方法:就是用来解决一类问题的代码的有序组合,是一个功能模块: 在方法部分,只使用对象名词调用方法: Scanner sc = new Scanner(S ...

  8. LeetCode(24): 两两交换链表中的节点

    Medium! 题目描述: 给定一个链表,两两交换其中相邻的节点,并返回交换后的链表. 示例: 给定 1->2->3->4, 你应该返回 2->1->4->3. 说 ...

  9. VS Code折腾记 - (4) 常用必备插件推荐【前端】

    前言 这篇文章只要让你做一些基础的配置,把vscode变得更加顺手: 插件的需求不是一成不变,有些插件我已经移除了..在最新的VSCODE 1.9.1中, 部分以前用插件实现的功能已经集成了,那就没有 ...

  10. fhq treap 学习笔记

    序 今天心血来潮,来学习一下fhq treap(其实原因是本校有个OIer名叫fh,当然不是我) 简介 fhq treap 学名好像是"非旋转式treap及可持久化"...听上去怪 ...