QQ登录

只需一步,快速开始

登录 | 注册 | 找回密码

三维网

 找回密码
 注册

QQ登录

只需一步,快速开始

展开

通知     

查看: 3785|回复: 2
收起左侧

[已解决] VBA怎样选择一个大圆或者矩形里的所有对象

[复制链接]
发表于 2009-11-9 20:06:52 | 显示全部楼层 |阅读模式 来自: 中国江苏无锡

马上注册,结识高手,享用更多资源,轻松玩转三维网社区。

您需要 登录 才可以下载或查看,没有帐号?注册

x
请教大家,我现在在写一个小程序时碰到要选择一个大圆里的所有小圆,还有一个大矩形里的小圆,请问好实现吗?我百思不得其解,请大家帮忙,谢谢了!
% k! E5 O6 m  l) |) Y功能描述:1.比如我想选一个大圆里所有小圆,当我点按钮“选择所有”,提示我先选择一个大圆,然后我点选大圆之后,所有的小圆被选中。
6 ~& g, G7 @" u3 [7 W- b                   2.我想选择一个斜方向的矩形里的所有小圆,点按钮“选择所有”,提示先选择一个矩形,点矩形后,矩形里所有的小圆被选中。
发表于 2009-11-9 23:34:53 | 显示全部楼层 来自: 中国
ACAD并不支持根据对象选择,只能用"圈围"方法近似地做.当边界为圆时,可以尽可能多地在圆周上取点,用一个近似圆的多边形做边界;对多段线(矩形就是闭合的二维多段线),可以使用多段线的顶点,但不允许自交,如果是开发准备发布的通用工具,就要对可能的自交做检查,以区别对待.而且当二维多段线有凸起(圆弧段)时,也要区别对待.下面的代码仅供参考.
/ o/ o7 ?, c: Q* Y

  1. 2 n! e$ J8 @* Q( R- O. r' f
  2. Private Const N As Long = 100 '声明一个常数,指定从圆周上拾取"圈围"的点数
    $ G4 {( H- v3 ^4 e1 v
  3. 5 @6 x6 ]2 C. ?/ y5 `7 Z. ~
  4. Sub A(); |: A1 x( W' @; ]7 q% I$ b- {6 s
  5.     Dim S1 As AcadSelectionSet '声明第一个选择集,用于从屏幕上选取做为下一步选择集边界的对象0 S2 A9 p4 Y) _% H' a4 D# p: L
  6.     Dim S2 As AcadSelectionSet '声明第二个选择集,用于选取边界内部对象0 ^( d1 S% N& ?8 {( O
  7.     Dim FT(3) As Integer, FD(3) As Variant '声明选择集过滤器
    % V0 [/ d) a/ B, Q$ s7 Q
  8.     Dim OldPICKADD As Long '声明一个长整形变量,用于存放原"pickadd"系统变量. u# z7 n$ b( c# n+ q5 u
  9.     Dim P() As Double '声明一个动态数组,用于存放"圈围"点集的三维坐标) H( V8 `9 s- f3 q  z0 b
  10.     Dim I As Long '循环变量
    9 r- j, g" L- W
  11.     Dim V As Variant '边界圆的圆心或矩形的顶点坐标
    : a; m/ T" R! _# o8 e# s3 `: i
  12.     ' s$ o" `" k5 ?0 W( t* F2 }& @
  13.     On Error Resume Next+ e5 D1 f* I! y& n4 F0 [/ o; c
  14.     ) t- c& k% w8 r  A1 b3 X3 m, _
  15.     FT(0) = -4: FD(0) = "<or" '设置选择集过滤器为选择圆或二维多段线% N3 ^9 D% m6 e; @! v7 f$ h+ l& G
  16.     FT(1) = 0: FD(1) = "Circle": E9 P: x: ^9 f+ k- R  E
  17.     FT(2) = 0: FD(2) = "LWPolyLine"
    8 _* k# U1 J& I2 c! T
  18.     FT(3) = -4: FD(3) = "or>"
    0 c* V! H. ?8 o  A
  19.     With ThisDrawing1 O6 S6 M' |3 |9 |( ], ^
  20.         OldPICKADD = .GetVariable("pickadd" ) '记录原"pickadd"系统变量
    $ Y& a- J& t' t: }9 _; M: y
  21.         .SetVariable "pickadd", 0 '把"pickadd"系统变量临时改为0(用 SHIFT 键添加到选择集),只为方便,不是必要的7 X+ P; G! A2 X. i* m8 G5 @
  22.         Set S1 = .SelectionSets.Add("S1" ) '新建选择集,用于从屏幕上选取边界对象
    2 t; k( P" x& ~& I+ P" x8 j
  23.         S1.SelectOnScreen FT, FD '在屏幕上选取圆或二维多段线
    5 c6 b$ X* p; ?8 @6 t! u3 `* _
  24.         .SetVariable "pickadd", OldPICKADD '把"pickadd"系统变量改回原值
    ) q8 N7 {- g1 e  U1 z4 A( i" ]: ]* G
  25.         If S1.Count > 0 Then '如果在屏幕上有效选取了边界对象% X/ a. `$ f% C: K
  26.             If S1.Item(S1.Count - 1).ObjectName = "AcDbCircle" Then '如果选取的最后一个对象是"圆"3 N7 I$ k+ j: ^3 f" x
  27.                 ReDim P(N * 3 - 1) '按"圈围"点集数量重定义三维坐标数组2 O, K# q7 `/ \" o# i3 {
  28.                 V = S1.Item(S1.Count - 1).Center '提取圆心坐标. O9 A. y; g0 X- U1 N
  29.                 For I = 0 To N - 1 '在圆周上按点集数量均匀取点计算圈围点集坐标# K4 G7 i& \' V0 }4 q* c
  30.                     P(I * 3) = V(0) + S1.Item(S1.Count - 1).Radius * Cos(CDbl(I) / CDbl(N) * 2 * .Utility.AngleToReal(180, acDegrees))) N- ^( I& D  }# u; Q* X
  31.                     P(I * 3 + 1) = V(1) + S1.Item(S1.Count - 1).Radius * Sin(CDbl(I) / CDbl(N) * 2 * .Utility.AngleToReal(180, acDegrees))
    ! ^( e0 c! j0 v6 T$ N; B
  32.                 Next+ y/ L# A- G. m$ R. R% @( E
  33.             Else '如果选取的最后一个对象是二维多段线
    3 u# }- u, L3 i' A
  34.                 V = S1.Item(S1.Count - 1).Coordinates '提取二维多段线顶点坐标(二维)
    ' g+ i3 ]0 N1 J* U7 n
  35.                 ReDim P((UBound(V) \ 2) * 3 + 2) '按多段线顶点数量重定义圈围点集三维坐标数组, j5 K  X; m5 k
  36.                 For I = 0 To UBound(V) \ 2 '把多段线二维坐标写入三维坐标数组
    ; v* o' I. L% R( U2 ]- B) Y
  37.                     P(I * 3) = V(I * 2)
    7 {. X- z$ m/ i4 O9 C, F' H0 _
  38.                     P(I * 3 + 1) = V(I * 2 + 1)& Y- |) \8 _# e5 `5 \: ^  l& J
  39.                 Next* r, w7 W0 p+ v7 s9 z
  40.             End If
    * L( J# y0 a1 p- N
  41.             Set S2 = .SelectionSets.Add("S2" ) '新建选择集,用于选取边界内部对象
    * K3 d0 x6 h/ T5 K) F. K
  42.             S2.SelectByPolygon acSelectionSetWindowPolygon, P '根据点集圈选
    5 o, u5 G5 A' Y
  43.             '自行处理边界内部被选择的对象4 n/ J7 C  k" c
  44.             S2.Delete '删除用过的选择集
    . e0 X, D0 `4 G
  45.         End If6 d2 @8 h& P; p! O0 F
  46.         S1.Delete '删除用过的选择集9 Q8 |4 T( h* _/ M$ q& N5 _/ l% {
  47.     End With
    1 a2 q, o/ A, O# w
  48. End Sub- C/ T+ X5 g+ _+ B+ J( ~4 n) f
复制代码

评分

参与人数 1三维币 +8 收起 理由
wang2003 + 8 应助

查看全部评分

 楼主| 发表于 2009-11-10 18:11:41 | 显示全部楼层 来自: 中国江苏无锡

回复 2# woaishuijia 的帖子

我当初的想法是先选择所有的小圆,再判断小圆的圆心坐标在不在大圆的范围内,还有删除与大圆想交的,这样效率不高,还是老师你的高明。谢谢了!
发表回复
您需要登录后才可以回帖 登录 | 注册

本版积分规则


Licensed Copyright © 2016-2020 http://www.3dportal.cn/ All Rights Reserved 京 ICP备13008828号

小黑屋|手机版|Archiver|三维网 ( 京ICP备2023026364号-1 )

快速回复 返回顶部 返回列表