revit坐标与屏幕坐标的转换

   class ScreenClientPointUtils
    {
        UIDocument _uiDocument = null;
        UIView _uiView = null;
        Transform _transfrom = Transform.Identity;
        public ScreenClientPointUtils(UIDocument uiDocument)
        {
            System.Diagnostics.Debug.Assert(uiDocument != null);

            _uiDocument = uiDocument;

            var activeView = uiDocument.Document.ActiveView;

            _uiView = uiDocument.GetOpenUIViews().First(o => o.ViewId == activeView.Id);

            _transfrom.Origin = activeView.Origin;
            _transfrom.BasisX = activeView.RightDirection;
            _transfrom.BasisY = activeView.UpDirection;
            _transfrom.BasisZ = activeView.ViewDirection;

            //if (activeView is ViewPlan)
            //    _uiView = uiDocument.GetOpenUIViews().First(o => o.ViewId == activeView.Id);
            //else
            //    throw new ArgumentException(activeView.GetType().FullName);
        }

        /// <summary>
        /// 屏幕坐标到revit平面坐标转换
        /// 
        /// 该接口没有考虑到屏幕DPI,就是说在分辨率不在100%时,可能有问题
        /// 
        /// 参考 ViewPlan2Screen接口
        /// </summary>
        /// <param name="screenPoint"></param>
        /// <returns></returns>
        public XYZ Screen2ViewPlan(System.Drawing.Point screenPoint)
        {
            //屏幕坐标
            var rect = _uiView.GetWindowRectangle();
            //屏幕比例
            double sWidth = rect.Right - rect.Left;
            double sHeight = rect.Bottom - rect.Top;

            double widthScale = (screenPoint.X - rect.Left) / sWidth;
            double heightScale = (rect.Bottom - screenPoint.Y) / sHeight;

            var corners = _uiView.GetZoomCorners();
            XYZ wLeftBottom = corners[0];
            XYZ wRightTop = corners[1];

            double wWidth = wRightTop.X - wLeftBottom.X;
            double wHeight = wRightTop.Y - wLeftBottom.Y;

            double widthDis = wWidth * widthScale;
            double heightDis = wHeight * heightScale;

            return new XYZ(wLeftBottom.X + widthDis, wLeftBottom.Y + heightDis, 0);
        }
        /// <summary>
        /// revit长度转屏幕像素长度
        /// </summary>
        /// <param name="length">单位: 英尺</param>
        /// <returns></returns>
        public double ViewPlanLength2ScreenPixel(double length)
        {
            XYZ wLeftBottom, wRightTop;
            GetWorkspaceRect(out wLeftBottom, out wRightTop);

            var bottomLeft = ViewPlan2Screen(wLeftBottom);
            var topRight = ViewPlan2Screen(wRightTop);

            return length * (topRight.X - bottomLeft.X) / (wRightTop.X - wLeftBottom.X);
        }

        /// <summary>
        /// revit平面坐标到屏幕坐标转化
        /// </summary>
        /// <param name="point"></param>
        /// <returns></returns>
        public System.Drawing.Point ViewPlan2Screen(XYZ point)
        {
            var corners = _uiView.GetZoomCorners();

            XYZ wLeftBottom = _transfrom.Inverse.OfPoint(corners[0]);
            XYZ wRightTop = _transfrom.Inverse.OfPoint(corners[1]);
            var tmpPt = _transfrom.Inverse.OfPoint(point);

            double wWidth = wRightTop.X - wLeftBottom.X;
            double wHeight = wRightTop.Y - wLeftBottom.Y;

            double widthScale = (tmpPt.X - wLeftBottom.X) / wWidth;
            double heightScale = (tmpPt.Y - wLeftBottom.Y) / wHeight;

            //屏幕坐标
            var rect = _uiView.GetWindowRectangle();
            //屏幕比例
            double sWidht = rect.Right - rect.Left;
            double sHeight = rect.Bottom - rect.Top;

            double widthDis = sWidht * widthScale;
            double heightDis = sHeight * heightScale;

            return new System.Drawing.Point(
                (int)((rect.Left + widthDis) * 1 / DPICache.Instance.ScreenRatio),
                (int)((rect.Bottom - heightDis) * 1 / DPICache.Instance.ScreenRatio));
        }
        /// <summary>
        /// 获取revit工作区域的屏幕坐标
        /// </summary>
        /// <param name="bottomLeft"></param>
        /// <param name="topRight"></param>
        public void GetWorkspaceRect(
            out System.Drawing.Point bottomLeft,
            out System.Drawing.Point topRight)
        {
            XYZ wLeftBottom, wRightTop;
            GetWorkspaceRect(out wLeftBottom, out wRightTop);

            bottomLeft = ViewPlan2Screen(wLeftBottom);
            topRight = ViewPlan2Screen(wRightTop);
        }
        void GetWorkspaceRect(out XYZ bottomLeft, out XYZ topRight)
        {
            var corners = _uiView.GetZoomCorners();

            bottomLeft = corners[0];
            topRight = corners[1];
        }
        class DPICache
        {
            DPICache()
            {
                Win32Api.SetProcessDPIAware();

                IntPtr screenDC = Win32Api.GetDC(IntPtr.Zero);
                int dpi_x = Win32Api.GetDeviceCaps(screenDC, /*DeviceCap.*/LOGPIXELSX);
                int dpi_y = Win32Api.GetDeviceCaps(screenDC, /*DeviceCap.*/LOGPIXELSY);
                //_scaleUI.X = dpi_x / 96.0;
                //_scaleUI.Y = dpi_y / 96.0;
                Win32Api.ReleaseDC(IntPtr.Zero, screenDC);
                System.Diagnostics.Debug.Assert(dpi_x == dpi_y);
                if (dpi_x != dpi_y)
                {
                    Framework.Logger.Instance.Error(
                        string.Format("获取屏幕DPI失败, dpi_x: {0} dpi_x: {1} ", dpi_x, dpi_y));
                }

                ScreenRatio = dpi_x / 96.0;
            }
            public static readonly DPICache Instance = new DPICache();
            public double ScreenRatio { get; private set; }

            const int LOGPIXELSX = 88;
            const int LOGPIXELSY = 90;
        }
    }

各大插件公司有个基本功能就是浮动轴符,它的实现原理应该是(当然他们可能有更高明的实现方法,不过我说的这种方法是可行的,作者实现了,效果还行):

      revit工作区域放一个大小一样的透明窗口,然后定时器实时GUI绘线和文本,同时调成窗口大小和位置

这个透明窗口要实时同步revit工作区域,也就是revit工作区域大小变化的时候,透明窗口也要同步变化,这时,就涉及到计算窗口的位置和其边框的大小了,就需要用到上面的帮助类了。

不过该接口还有点问题,就是它没有适配屏幕分辩率不是100%的情况,仅供大家参考,作者有时间会把该问题解决。

其实很多操作都是用上面的技术实现的,如很多功能弹出窗口默认在revit工作区域的左上角;绘制立管。这里特别提下鸿业,他们的管道模块各种功能的绘制,实现的确实高明,效果是相当的顺滑,我尝试过,达不到他们的效果,学习学习。。

  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值