以梦为马,不负韶华

搜索
查看: 458|回复: 2
收起左侧

[软件交流] 在线CAD二次开发中实现交互绘制的详细步骤

[复制链接]
 楼主| 发表于 1970-1-1 08:00:00 显示全部楼层 |阅读模式
本帖最后由 lihao2014 于 2024-3-5 11:58 编辑

前言
在线CAD的二次开发中对于交互绘制的使用频率非常高,MxCAD的思路是通过调取命令功能和取点对象来实现动态绘制,下面文章我们会详细讲解一下这三个步骤,如果对WEB CAD的二次开发还不熟悉可以参考入门教程

一、什么是命令功能
mxdraw.js库提供了一种可以通过输入一些特定的命令调用绘制方法,并在动态绘制过程中实现参数化绘图的功能,即为命令功能。例如想用命令功能绘制一个圆,可在命令行内调用绘圆方法,并根据用户输入圆的半径参数来确定圆的大小,下面是实现步骤。

1、监听输入
因用户在命令模式下绘图,需要输入命令,系统会即时输出提示消息或相关命令消息,所以需要在画布所在页面添加输入框和文本显示框。
此外还需利用方法Mx.MxFun.setCommandLineInputData()来设置命令行消息数据,利用方法Mx.MxFun.listenForCommandLineInput()来监听命令行消息动态更新的数据、在取点对象中设置的命令等,当监听到用户输入命令并触发enter事件时,自动执行命令。监听代码如下:
  1. <input id="mxCmdText"/>
  2. let event = document.getElementById('mxCmdText')
  3. // 设置命令行消息数据 (在监听input输入框的onKeydown事件的回调函数中调用)
  4. event.onkeydown = function (e) {
  5. Mx.MxFun.setCommandLineInputData(event.value, e.keyCode)
  6. }
复制代码

2、编写绘制函数
在使用命令功能时,编写绘制函数的过程中可以通过取点对象调用setMessage方法给用户提示必要的操作信息以及获取到使用者的输入信息或关键词下面以绘制线段函数为例:
  1. // 线段函数async function BR_Line() {
  2.   const getPoint = new Mx.MrxDbgUiPrPoint();
  3.   // 交互提示
  4.   getPoint.setMessage("\n指定直线起点:");
  5.   getPoint.go(async (status:number)=>{
  6.     if(status != Mx.MrxDbgUiPrBaseReturn.kOk) return
  7.     let pt1 = getPoint.value();
  8.     let line = new Mx.MxDbLine();
  9.     line.pt1 = pt1;
  10.     getPoint.setMessage("\n指定直线终点:");
  11. getPoint.setUserDraw((pt:any,pw:any)=>{
  12.         line.pt2 = pt;
  13.         pw.drawCustomEntity(line)
  14.   });
  15.   line.pt2 = await getPoint.go();
  16. Mx.MxFun.getCurrentDraw().addMxEntity(line);})}
复制代码

3、注册命令
将写好的绘制函数通过Mx.MxFun.addCommand()方法进行注册命令,代码如下:
  1. // 注册命令名 BR_Line:命令名
  2. Mx.MxFun.addCommand("BR_Line", ()=> {
  3.   // 是否正在运行某个命令
  4. if(Mx.MxFun.isRunningCommand()) {
  5.       return
  6.   }
  7.   // 目标绘制函数
  8.   BR_Line()})
复制代码

4、执行命令、输出消息
注册好目标命令后,用户可通过Mx.MxFun.sendStringToExecute()方法手动执行输入的命令。此外,用户还可以通过Mx.MxFun.listenForCommandLineInput()方法按需输出提示消息或命令内容,参考代码如下:
  1. // 手动执行命令
  2. Mx.MxFun.sendStringToExecute("目标命令");
  3. // 监听正在绘制中的命令提示
  4. Mx.MxFun.listenForCommandLineInput(({ msCmdTip, msCmdDisplay, msCmdText }) => {
  5.   console.log(msCmdTip, msCmdDisplay, msCmdText)})
复制代码
命令功能效果完整示例代码如下:
  1. <!DOCTYPE html><html lang="en">
  2. <head>
  3.     <meta charset="UTF-8">
  4.     <meta http-equiv="X-UA-Compatible" content="IE=edge">
  5.     <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6.     <title>MxDbLine 示例</title>
  7.     <script src="https://unpkg.com/mxdraw/dist/mxdraw.umd.js"></script></head>
  8. <body>
  9.     <div style="height: 80vh; overflow: hidden;">
  10.         <canvas id="mxdraw"></canvas>
  11.     </div>
  12.     <div>
  13.         <textarea id="mxCmdArea" rows="5" style="width:100%;overflow: scroll;"
  14.         placeholder="命令:" readonly></textarea>
  15.         <input type="text" id="mxCmdText" />
  16.     </div></body><script type="module">
  17.     Mx.loadCoreCode().then(async () => {
  18.         // 创建控件对象
  19.         Mx.MxFun.createMxObject({
  20.             canvasId: "mxdraw", // canvas元素的id
  21.         });
  22.     })
  23.     let event = document.getElementById('mxCmdText')
  24.     // 设置命令行消息数据 (在监听input输入框的onKeydown事件的回调函数中调用)
  25.     event.onkeydown = function (e) {
  26.         Mx.MxFun.setCommandLineInputData(event.value, e.keyCode)
  27.     }
  28.     // 直线函数
  29.     async function BR_Line() {
  30.         const getPoint = new Mx.MrxDbgUiPrPoint();
  31.         // 交互提示
  32.         getPoint.setMessage("\n指定直线起点:");
  33.         let pt1 = await getPoint.go();
  34.         if (!pt1) return
  35.         let line = new Mx.MxDbLine();
  36.         line.pt1 = pt1;
  37.         getPoint.setMessage("\n指定直线终点:");
  38.         getPoint.setUserDraw((pt, pw) => {
  39.             line.pt2 = pt;
  40.             pw.drawCustomEntity(line)
  41.         });
  42.         const pt2 = await getPoint.go()
  43.         if (!pt2) return
  44.         line.pt2 = pt2;
  45.         Mx.MxFun.getCurrentDraw().addMxEntity(line);
  46.     }
  47.     // 注册命名
  48.     Mx.MxFun.addCommand("BR_Line", () => {
  49.         // 是否正在运行某个命令
  50.         if (Mx.MxFun.isRunningCommand()) {
  51.             return
  52.         }
  53.         BR_Line()
  54.     })
  55.     // 监听正在绘制中的命令提示
  56.     Mx.MxFun.listenForCommandLineInput(({ msCmdTip, msCmdDisplay, msCmdText }) => {
  57.         console.log(msCmdTip, msCmdDisplay, msCmdText)
  58.         document.getElementById('mxCmdArea').innerHTML = msCmdTip
  59.         document.getElementById("mxCmdArea").scrollTop = document.getElementById("mxCmdArea").scrollHeight;
  60.     })
  61. </script>
  62. </html>
复制代码

二、取点对象
在画布中绘制基本图形前,先要学会点对象的绘制(复杂图形都由点、线、面组成),mxdraw.js中提供了API MrxDbgUiPrPoint用于构建一个取点对象,并提供一系列方法属性辅助用户绘图,点击MrxDbgUiPrPoint API查看详细属性和方法说明。


1、基础取点
调用MrxDbgUiPrPoint中的方法go()记录鼠标开始拖动的点位置,用户可利用该方法记录画布中的目标点位,执行go()方法,若取点成功则返回点的位置,默认点值为THREE.Vector3();若取点失败则返回null,示例代码如下:
  1. Mx.loadCoreCode().then(async () => {
  2.     // 创建控件对象
  3.     Mx.MxFun.createMxObject({
  4.         canvasId: "mxdraw", // canvas元素的id
  5.         callback: (mxDrawObject, dom ) => {
  6.             //图纸展示控件创建完成后的回调函数 回调参数mxDraw和dom
  7.             console.log(mxDrawObject, dom)
  8.         },
  9.     });
  10.     const getPoint = new Mx.MxFun.MrxDbgUiPrPoint()
  11.     const pt = await getPoint.go()
  12.     if(!pt) return
  13.     console.log('目标点位' , pt)})
复制代码
此外若用户需要连续取点,则可以结合while函数调用go()方法,代码如下:
  1. const getPoint = new Mx.MxFun.MrxDbgUiPrPoint()while(true){
  2.     const pt = await getPoint.go()
  3.     if(!pt) break}
复制代码

2、辅助命令功能
结合命令功能mxdraw.js中取点对象提供的一系列方法对绘图操作时所进行的操作,提供相应提示和操作选项,以下面两种常用方法为例。
2.1设置提示字符串:setMessage()方法对绘图操作时所进行的操作进行提示,代码如下:
  1. const getPoint = new Mx.MxFun.MrxDbgUiPrPoint()
  2. getPoint.setMessage('设置提示字符串')
  3. const pt = await getPoint.go()
复制代码
2.2关键字:setKeyWords()方法可为当前命令设置关键字选项,keyWordPicked()方法可返回用户选择的关键字,代码如下:
  1. const getPoint = new Mx.MxFun.MrxDbgUiPrPoint()
  2. // 设置关键字列表
  3. getPoint.setKeyWords('[闭合(C)/ 放弃(U)]')
  4. // 获取用户选择关键字
  5. getPoint.isKeyWordPicked('关键字')
复制代码

3.取点对象动态绘制
取点对象中还提供了setUserDraw()方法设置交互过程的动态绘制调用对象,在该过程中所绘制的图像都是临时绘制,不会保存在画布中。该方法提供了两个回调参数:currentPointpWorldDraw
currentPoint为当前光标所在点的点位,pWorldDraw为一个动态绘制回调对象,可根据用户设置动态绘制目标图形,点击动态绘制回调对象查看详细属性和方法说明,示例代码如下:
  1. async function draw_line(){
  2.     const line = new Mx.MxDbLine()
  3.     const getPoint = new Mx.MxFun.MrxDbgUiPrPoint()
  4.     const pt1 = await getPoint.go()
  5.     if(!pt1) return
  6.     line.pt1 = pt1
  7.     // 动态绘制
  8.     getPoint.setUserDraw((currentPoint, pWorldDraw)=>{
  9.         line.pt2 = currentPoint
  10.         pWorldDraw.drawCustomEntity(line)
  11.     })
  12.     const pt2 = await getPoint.go()
  13.     if(!pt2) return
  14.     line.pt2 = pt2
  15.     // 获取控件对象并将线段对象line添加到画布中
  16.     Mx.MxFun.getCurrentDraw().addMxEntity(line);}
复制代码

三、动态绘制


1.基础绘制
mxdraw.js提供了两种实现动态绘制的方式,它们可以在不改变图形类的情况下,改变图形的大小、方向等,帮助用户更快捷地绘制目标图形。这两种动态绘制的方式都是结合取点对象来实现动态绘制图形的功能。

方式一利用API McEdGetPointWorldDrawObject构建一个动态绘制回调对象,再通过取点对象中的setUserDraw()方法调用该对象。点击Mx.McEdGetPointWorldDrawObject API查看详细属性和方法说明,参考代码:
  1. // 实例化取点对象
  2. const getPoint = new Mx.MrxDbgUiPrPoint();
  3. // 实例化线段对象line
  4. let line = new Mx.MxDbLine()
  5. const pt1 = await getPoint.go()
  6. line.setPoint1(pt1)
  7. // 实例化动态绘制对象
  8. const worldDrawComment = new Mx.McEdGetPointWorldDrawObject();
  9. // 设置动态绘制回调函数
  10. worldDrawComment.setDraw((currentPoint) => {
  11.     line.setPoint2(currentPoint);
  12.     worldDrawComment.drawCustomEntity(line)
  13. });
  14. // 使用动态绘制对象
  15. getPoint.setUserDraw(worldDrawComment)
复制代码

方式二直接运用取点对象提供的setUserDraw()方法设置交互过程的动态绘制调用对象。详细描述可参考取点对象动态绘制,示例代码如下:
  1. const getPoint = new Mx.MrxDbgUiPrPoint()
  2. // 实例化线段对象line
  3. let line = new Mx.MxDbLine();
  4. const pt1 = await getPoint.go()
  5. if(!pt1) return
  6. line.pt1 = pt1
  7. // 实例化动态绘制对象
  8. getPoint.setUserDraw((currentPoint, worldDrawComment)=>{
  9.     line.pt2 = currentPoint
  10.     worldDrawComment.drawCustomEntity(line)
  11. });
  12. const pt2 = await getPoint.go()
  13. if(!pt2) return
  14. line.pt2 = pt2;
复制代码
在动态绘制中的绘制的图形都是临时的,意味着这些图形最终是不会保持在画布上的,如果需要将动态绘制的图形保存在画布上请调用取点对象的drawReserve方法

2.图形对象的动态绘制
我们的图形对象是继承自自定义图形对象的,而任何的图形对象都可以通过动态绘制对象实现动态绘制的效果。图形对象具体如何实现动态绘制的请前往自定义图形查看详情,示例代码如下:
  1. // 实例化取点对象
  2. const getPoint = new Mx.MrxDbgUiPrPoint();
  3. // 实例化动态绘制对象
  4. const worldDrawComment = new Mx.McEdGetPointWorldDrawObject();
  5. const pt1 = await getPoint.go()
  6. if(!pt1) return
  7. // 设置动态绘制回调函数
  8. worldDrawComment.setDraw((currentPoint) => {
  9.     // 绘制线段对象
  10.     worldDrawComment.drawLine(pt1, currentPoint);
  11.     // 在绘制过程中还可以一起绘制其他图形 绘制过程结束时,不会保留这些图形,如:
  12.     // 绘制以下图形使用红色
  13.     worldDrawComment.setColor('#ff0000')
  14.     // 绘制半径为6的圆
  15.     worldDrawComment.drawCircle(currentPoint, 6)
  16.     // 绘制字体大小为36的 “文字”  角度为0
  17.     worldDrawComment.drawText("文字" ,36 , 0, currentPoint)
  18.     // 绘制矩形 pt1, currentPoint为矩形两对角
  19.     worldDrawComment.drawRect(pt1, currentPoint)
  20. });
  21. // 使用动态绘制对象
  22. getPoint.setUserDraw(worldDrawComment)
复制代码



[发帖际遇]: 一个袋子砸在了 lihao2014 头上,lihao2014 赚了 2 个 韶华币. 幸运榜 / 衰神榜
发表于 1970-1-1 08:00:00 显示全部楼层
厉害啊,谢谢你的分享
[发帖际遇]: qyf 捡到一块切糕,卖给了小马,赚了 6 个 韶华币. 幸运榜 / 衰神榜
回复 支持 反对

使用道具 举报

发表于 1970-1-1 08:00:00 显示全部楼层
确实厉害,抽空学习一下。
[发帖际遇]: nanbianderen 屌丝逆袭成功,获得白富美女神垂青,赚了 1 个 韶华币. 幸运榜 / 衰神榜
回复 支持 反对

使用道具 举报

懒得打字嘛,点击右侧快捷回复
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|以梦为马,不负韶华

GMT+8, 2025-1-6 15:37

Powered by 以梦为马,不负韶华

© 2024-2099 Meng.Horse

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