乐筑天下

搜索
欢迎各位开发者和用户入驻本平台 尊重版权,从我做起,拒绝盗版,拒绝倒卖 签到、发布资源、邀请好友注册,可以获得银币 请注意保管好自己的密码,避免账户资金被盗
查看: 43|回复: 19

[编程交流] Lisp自动绘制rec

[复制链接]

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:25:50 | 显示全部楼层 |阅读模式
大家好,我在一家生产建筑模型的公司工作。
我专门从事绘图领域的工作,在那里我使用AutoCad绘制模型的各个部分。
 
在我们的工作中,我们必须不断地绘制“地图”给那些把这些碎片放在一起形成模型的人
 
我已经开发了几个lisp例程来简化我的工作,现在我正在尝试简化(很多)这些地图的制作。
 
所以我现在需要的是:
1) 一种对点列表中的点进行排序的方法,以便它们相对于基点从最近到最远按顺序排列
2) 积分的数量总是不同的,因此,我如何连续存储这些积分?(如列表中有10个点,则点1为最近点,点10为最远点)
3) 在对点进行排序之前(或之后),删除彼此距离过近的点(例如,如果点2距离点3 0.02个单位,则忽略点3)
 
所有这一切都是为了能够自动检测窗户和门在墙中间的位置,并在地图上标出它们,而不需要手动找到它们
 
这是我到目前为止的代码(它要求模型中使用的材质的E压力、地图中墙的编号以及将墙的编号放置在地图中的文本高度)
 
P、 如果是的话,我是一个非常业余的程序员
  1. (defun c:md ()
  2. (setq layercorrente (getvar 'clayer))
  3. (defun dtr (a) (* Pi (/ a 180.0)))
  4. (defun rtd (b) (* b (/ 180.0 Pi)))
  5. (setvar "OSNAPCOORD" 1)
  6. (setq esp (getdist "\nDigite a espessura da parede <enter = ultima espessura>: "))
  7. (if (= esp nil) (setq esp esptemp))
  8. (setq esptemp esp)
  9. (setq altura_texto (getdist "\nInsira a altura do texto <enter = ultima altura>:"))
  10. (cond
  11.    ((and (= altura_texto nil) (= altura_textotemp nil)) (setq altura_texto esp)
  12.    )
  13.    ((and (= altura_texto nil) (/= altura_textotemp nil)) (setq altura_texto altura_textotemp)
  14.    )
  15. )
  16. (setq numero (getint "Indique o numero da próxima parede <enter = ultimo numero de parede>:"))
  17. (cond
  18.    ((and (= numero nil) (= numero_old nil))
  19.    (setq numero 1)
  20.    )
  21.    ((and (= numero nil) (/= numero_old nil))
  22.    (setq numero numero_old)
  23.    )
  24. )
  25. (setq numero_old numero)
  26. (setq texto (itoa numero))
  27. (command "_-style" "loks" "txt" altura_texto "1" "" "" "" "")
  28. (setq rotation 0)
  29. (setq ponto1 (getpoint "\nInsira o primeiro ponto de inserção da parede levando em conta o sentido anti-horário:"))
  30. (setq ponto2 (getpoint "\nInsira o segundo ponto de inserção:"))
  31. (setq comprimento (distance ponto1 ponto2))
  32. (setq ang1 (angle ponto1 ponto2))
  33. (setq ang2 (rtd ang1))
  34. (setq ang3 (+ ang2 90.0))
  35. (if (>= ang3 360.0) (setq ang3 (- ang3 360)))
  36. (setq ang4 (+ ang2 270.0))
  37. (if (>= ang4 360.0) (setq ang4 (- ang4 360)))
  38. (setq ponto3 (polar ponto2 (dtr ang3) esp))
  39. (setq ponto4 (polar ponto1 (dtr ang3) esp))
  40. (setq pontoref (polar ponto1 ang1 (/ comprimento 2)))
  41. (cond
  42. ((<= 0 ang4 105) (setq pontotxt (polar pontoref (dtr ang4) (* 1.5 altura_texto)))
  43. )
  44. ((and (< 105 ang4 255) (<= 0 numero 9)) (setq pontotxt (polar pontoref (dtr ang4) (* 2 altura_texto)))
  45. )
  46. ((and (< 105 ang4 255) (<= 10 numero 99)) (setq pontotxt (polar pontoref (dtr ang4) (* 2.5 altura_texto)))
  47. )
  48. ((and (< 105 ang4 255) (<= 100 numero)) (setq pontotxt (polar pontoref (dtr ang4) (* 3.4 altura_texto)))
  49. )
  50. ((<= 255 ang4 300) (setq pontotxt (polar pontoref (dtr ang4) (* 2 altura_texto)))
  51. )
  52. ((< 300 ang4 360) (setq pontotxt (polar pontoref (dtr ang4) (* 1.5 altura_texto)))
  53. )
  54. )
  55. (defun c:mapadeparedes ()
  56. (setvar 'clayer layercorrente)
  57. (command "pline" ponto1 ponto2 ponto3 ponto4 "c")
  58. (command "LAYER" "M" "-TEXTO_DECORADO" "C" "42" "" "")
  59. (command "text" pontotxt rotation texto)
  60. (setvar 'clayer layercorrente)
  61. )
  62. (c:mapadeparedes)
  63. (while (/= ponto1 nil)
  64. (setq ponto1 (getpoint "\nInsira o primeiro ponto de inserção da parede levando em conta o sentido anti-horário:"))
  65. (cond
  66.    ((/= ponto1 nil)
  67. (setq ponto2 (getpoint "\nInsira o segundo ponto de inserção:"))
  68. (setq comprimento (distance ponto1 ponto2))
  69. (setq ang1 (angle ponto1 ponto2))
  70. (setq ang2 (rtd ang1))
  71. (setq ang3 (+ ang2 90.0))
  72. (if (>= ang3 360.0) (setq ang3 (- ang3 360)))
  73. (setq ang4 (+ ang2 270.0))
  74. (if (>= ang4 360.0) (setq ang4 (- ang4 360)))
  75. (setq ponto3 (polar ponto2 (dtr ang3) esp))
  76. (setq ponto4 (polar ponto1 (dtr ang3) esp))
  77. (setq pontoref (polar ponto1 ang1 (/ comprimento 2)))
  78. (cond
  79. ((<= 0 ang4 105) (setq pontotxt (polar pontoref (dtr ang4) (* 1.5 altura_texto)))
  80. )
  81. ((and (< 105 ang4 255) (<= 0 numero 9)) (setq pontotxt (polar pontoref (dtr ang4) (* 2 altura_texto)))
  82. )
  83. ((and (< 105 ang4 255) (<= 10 numero 99)) (setq pontotxt (polar pontoref (dtr ang4) (* 2.5 altura_texto)))
  84. )
  85. ((and (< 105 ang4 255) (<= 100 numero)) (setq pontotxt (polar pontoref (dtr ang4) (* 3.4 altura_texto)))
  86. )
  87. ((<= 255 ang4 300) (setq pontotxt (polar pontoref (dtr ang4) (* 2 altura_texto)))
  88. )
  89. ((< 300 ang4 360) (setq pontotxt (polar pontoref (dtr ang4) (* 1.5 altura_texto)))
  90. )
  91. )
  92. (setq numero (+ numero 1))
  93. (setq texto (itoa numero))
  94. (c:mapadeparedes)
  95.    )
  96. )
  97. )
  98. (setq numero_old (+ numero 1))
  99. )
  100. (princ "\nDIGITE MD PARA DESENHAR O MAPA DE PAREDES DE DECORADO MAIS FACILMENTE.")
  101. (princ)
回复

使用道具 举报

63

主题

6297

帖子

6283

银币

后起之秀

Rank: 20Rank: 20Rank: 20Rank: 20

铜币
358
发表于 2022-7-5 15:31:57 | 显示全部楼层
步骤#3的解决方案(Lee Mac的LM:唯一模糊子函数)。
 
无论如何都进行了尝试(没有看他的代码),结果是:
  1. (setq org '(0. 0. 0.)
  2.      pts '((9. 8. 0.) (7. 6. 0.) (1. 2. 0.) (3. 4. 0.))
  3.      )
  4. (vl-sort pts '(lambda (a b) (< (distance org a) (distance org b))))

 
不确定它会更快,因为尾部递归存储“v”,并且不会尝试访问vl remove if中点列表的car。
 
用直接递归做同样的事情——这是李证明的一个相当大的成就。
回复

使用道具 举报

66

主题

1552

帖子

1514

银币

后起之秀

Rank: 20Rank: 20Rank: 20Rank: 20

铜币
325
发表于 2022-7-5 15:33:58 | 显示全部楼层
谢谢Tharwhat和Grrr u的帮助!
最后,我只使用了Tharwhat的解决方案,因为我通过替代方法达到了相同的目标,从而解决了其他问题。
然而,更多的知识永远不够,所以再次感谢=)
 
现在我的例程几乎完成了,我唯一真正需要的是一种更有效的方法来验证指定层是否包含在选择集中
 
现在我正在隔离指定的层,如果选择集为空,这意味着返回nil,否则为T,但它会消耗很多时间,因为这个过程会重复很多次
那么,有没有办法找到在选择集中是否至少有一个指定层的对象?
回复

使用道具 举报

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:37:27 | 显示全部楼层
以下是最直接的方法:
  1. ; _$ (UQpL 0.01 '(1.0 1.1 2.0 2.01 2.2)) >> (1.0 1.1 2.0 2.2)
  2. ; _$ (UQpL 0.15 '(1.0 1.1 2.0 2.01 2.2)) >> (1.0 2.0 2.2)
  3. (defun UQpL ( fuzz pL )
  4. ((lambda (f) (f pL fuzz))
  5.    (lambda (pL fuzz / v)
  6.      (if pL
  7.        (cons
  8.          (setq v (car pL))
  9.          (f ((lambda (pL) (vl-remove-if (function (lambda (x) (equal v x fuzz))) pL)) (cdr pL)) fuzz)
  10.        )
  11.      )
  12.    )
  13. )
  14. ); defun UQpL

 
或者,如果您已经选择了项目并将其转换为搪瓷列表:
  1. (ssget '((8 . "yourlayername")))
回复

使用道具 举报

58

主题

3353

帖子

33

银币

顶梁支柱

Rank: 50Rank: 50

铜币
1761
发表于 2022-7-5 15:39:46 | 显示全部楼层
 
谢谢ronjop!
对于特定情况,第一个代码就足够了,但后一个代码也非常有用!)
 
我会把话题留着,以防再次出现任何问题,但我认为现在我已经准备好了!
 
(另外,我之前已经发布了我的代码,如果有人能分析它来验证我是否“编码正确”,比如……如果有更有效的方法来做我已经在做的事情,代码会有一点扩展,但如果有人有一点空闲时间,我将不胜感激)
回复

使用道具 举报

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:43:29 | 显示全部楼层
大家好,在尝试继续改进我的lisp例程时,我想知道是否有一种方法可以确定要找到的对象的模式,然后在选择集中量化它们。。。喜欢我确定我想找到两条垂直线,在X轴上彼此相距0.1和0.1,然后我选择一堆对象,并希望代码返回给我在这些图案中找到的对象数量,我在前面的选择中指定了这些图案
回复

使用道具 举报

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:46:18 | 显示全部楼层
P、 我想通过选择集而不是数字参数来指定图案
回复

使用道具 举报

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:51:06 | 显示全部楼层
这不可能?
回复

使用道具 举报

1

主题

12

帖子

11

银币

初来乍到

Rank: 1

铜币
5
发表于 2022-7-5 15:54:01 | 显示全部楼层
发布示例选择集的图形。。我不知道你在做什么。
回复

使用道具 举报

58

主题

3353

帖子

33

银币

顶梁支柱

Rank: 50Rank: 50

铜币
1761
发表于 2022-7-5 15:57:01 | 显示全部楼层
嗨,ronjonp,谢谢你的回复!
 

                               
登录/注册后可看大图

 
在这张图中,你可以看到一个有窗口的蓝图片段
我开发了一个例程,可以自动识别图形中的窗口并用矩形标记它们,现在我想要的是指定一个选择模式,就像图像中标记的两个矩形一样,并在一个选区集中跟踪它们,以计算它们出现的次数
 
如果我能做到这一点,我不仅能确定窗口在哪里,还能确定窗口有多少叶子(在这种情况下,它会检测到窗口内的2个选择模式,因此它会知道窗口有3个叶子)
 
我是巴西人,如果我不太全面,请原谅我
回复

使用道具 举报

发表回复

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

  • 微信公众平台

  • 扫描访问手机版

  • 点击图片下载手机App

QQ|关于我们|小黑屋|乐筑天下 繁体中文

GMT+8, 2025-3-14 21:05 , Processed in 0.446956 second(s), 72 queries .

© 2020-2025 乐筑天下

联系客服 关注微信 帮助中心 下载APP 返回顶部 返回列表