leven50 发表于 2024-10-17 15:15:37

【快速入门教程二】天工CAD插件开发模板介绍

本帖最后由 天工开发者团队 于 2024-10-17 15:32 编辑

一、整体框架

         通过向导(天工CAD开发资源)得到一个天工CAD插件项目之后,可以得到上述几个核心类,分别有着以下作用。
TGCore:程序初始化入口
TGUIManager:Ribbon菜单UI管理类,负责创建Ribbon菜单
TGCmdManager:Cmd命令管理类,负责命令的注册和分发调度以及对话框的管理
TGEventManager:Event事件管理类,负责响应操作所产生事件的处理
TGHelper:Help类,可以调用一些辅助接口

二、业务流程

         业务流程可以分成三部分:
(一)创建Ribbon菜单:用户根据UI设计菜单,并完成Ribbon命令菜单的创建。
(二)实现业务功能:实现菜单命令,具体包括命令创建、功能实现、命令调度、消息事件处理、命令注册。
(三)处理消息事件:处理命令响应、文档事件等情况。
2.1 创建Ribbon菜单
         打开“当前项目//res//XXXRibbon.xml”文件,对Ribbon菜单进行编辑,具体格式如下:
<ribbon>
    <!-- 在此处添加用户自定义的TabbedContainer -->
    <tab name="天工插件示例">
      <!-- 在此处添加用户自定义的group分组-->
      <group name="Welcome">
            <!--在此处添加用户自定义的命令按钮 -->
            <!--commandId: 自定义命令ID,必须唯一,用于标识命令 -->
            <!--label: 命令显示的名称 -->
            <!--imageId: 命令显示的图标ID -->
            <!--buttonStyle: 命令按钮的样式 -->
            <button commandId="0"label="弹出窗口"imageId="166" buttonStyle="5" />
            <button commandId="1"label="侧边栏"imageId="167" buttonStyle="5" />
            <button commandId="2"label="弹窗消息事件版"imageId="167" buttonStyle="5" />
            <button commandId="3"label="侧边栏消息事件版"imageId="167" buttonStyle="5" />
      </group>
    </tab>
</ribbon>
界面效果图如下所示

命令按钮关键词说明:
commandId:自定义命令ID(唯一)
label:命令显示的名称
imageId:按钮位图资源ID
buttonStyle:按钮风格
“1” seButtonAutomatic:按钮样式自动确定,即UI框架会根据上下文和平台偏好自动选择合适的样式
“2” seButtonCaption:只有按钮的文字标题,没有图标
“3” seButtonIcon:只有按钮图标,没有文字标题
“4” seButtonIconAndCaption:包括按钮图标和文字标题(通常并排显示)
“5” seButtonIconAndCaptionBelow:与“4”类似,但图标位于上方,标题位于下方
“6” seCheckButton:复选按钮,用户可以选中或取消选中
“7” seCheckButtonAndIcon:结合复选按钮和图标的样式,更直观
“8” seRadionButton:单选按钮,用户在一组中选择其中一个
根据UI设计的Menu,创建Ribbon菜单,将所有的命令按钮添加完毕,对应好命令ID,以及按钮要展示的命令名称和图标,并根据需求进行命令分组,菜单创建完成后就可以进行命令的实现了。
2.2 业务功能实现
         业务的实现体现为命令按钮的功能实现,具体步骤:命令创建→功能实现→命令调度→事件处理→命令注册。代码逻辑流程如下图所示。

其中,创建的业务命令类型主要可分为以下三种:
弹框式命令(继承自TGCmdPopup):基于TGCCmdBase的二次封装,支持弹窗对话框形式的业务命令
         命令激活处理接口:Activate()   命令终止处理接口:
         Terminate()   创建并展示对话框:CreateRibbon()
         消息事件处理:MouseClick()、KeyDown()、MouseDrag()、MouseMove()、Filter()等
侧边栏命令(继承自TGCmdEdgebar):基于TGCCmdBase的二次封装,支持侧边栏形式的业务命令
         命令激活处理接口:Activate()
         命令终止处理接口:Terminate()
         创建并展示侧边栏对话框:CreateRibbon()
         消息事件处理:MouseClick()、KeyDown()、MouseDrag()、MouseMove()等
直接操作命令(继承自TGCCmdBase)
         命令激活处理接口:Activate()
         命令终止处理接口:Terminate()
         消息事件处理:raw_MouseDown()、raw_MouseUp()、raw_MouseMove()、raw_MouseClick(、raw_MouseDblClick()、raw_MouseDrag()、raw_Filter()、raw_WindowProc()等)
接下来,以弹框式界面命令实现为例子进行讲解
2.2.1   命令创建
(一)基于TGCmdPopup定义自己的命令类,如TGUserCmdPopup
(二)定义命令Com的别名:typedef ComObject TGCmdUserCmdPopupObj
(三)实现Activate和Terminate接口(若无多余操作,可不重写)
(四)添加需要处理的消息事件
class TGUserCmdPopup : public TGCmdPopup
{
public:
    TGUserCmdPopup();
    virtual ~TGUserCmdPopup();
    STDMETHODIMP Activate();
    STDMETHODIMP Terminate();
    STDMETHODIMP MouseClick(short sButton,    short sShift, double dX, double dY,    double dZ, LPDISPATCH pWindowDispatch,    long lKeyPointType,    LPDISPATCH pGraphicDispatch);
    STDMETHODIMP MouseMove(short sButton, short sShift, double dX, double dY, double dZ, LPDISPATCH pWindowDispatch, long lKeyPointType, LPDISPATCH pGraphicDispatch);
    STDMETHOD(MouseDrag)(short sButton, short sShift, double dX, double dY, double dZ, LPDISPATCH pWindowDispatch, short DragState, long lKeyPointType, LPDISPATCH pGraphicDispatch);
    STDMETHODIMP Filter(LPDISPATCH pGraphicDispatch, VARIANT_BOOL* vbValid);
    STDMETHOD(KeyDown)(short* KeyCode,short Shift);
};
typedef CComObject<TGUserCmdPopup> TGCmdUserCmdPopupObj;
2.2.2 对话框创建
(一)基于CDialogEx,新建MFC类,如TGPopupDialog
(二)通过工具对话框的UI设计完成界面的布局
(三)完成业务逻辑处理
class TGPopupDialog : public CDialogEx
{
    DECLARE_DYNAMIC(TGPopupDialog)
public:
    TGPopupDialog(CWnd* pParent = nullptr);   // 标准构造函数
    virtual ~TGPopupDialog();
// 对话框数据
#ifdef AFX_DESIGN_TIME
    enum { IDD = IDD_POPUPDIALOG };   
#endif
protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持
    DECLARE_MESSAGE_MAP()
public:
// 业务逻辑处理
    afx_msg void OnTestButton1();
    afx_msg void OnTestButton2();
    virtual void OnOK();
    virtual void OnCancel();
};
2.2.3 命令与对话框绑定
(一)修改TGCmdDialogFactory::GetSpecificCmdDialog():添加new TGPopupDialog对象的操作
//此处为用户自定义命令界面类实例生成工厂,请根据自定义command对应的界面实例补充switch
CDialog* TGCmdDialogFactory::GetSpecificCmdDialog()
{
    switch (CommandId)
    {
    case 0:
    {
      TGPopupDialog* pPopupDialog = new TGPopupDialog();
      return dynamic_cast<CDialog*>(pPopupDialog);
      break;
    }
    default:
      return NULL;
      break;
    }
}(二)修改TGCmdDialogFactory::GetSpecificCmdDialogTemplateId():添加相应对话框IDD
//此处为用户自定义界面类ID关联command处,请根据commandid关联所对应的界面
UINT TGCmdDialogFactory::GetSpecificCmdDialogTemplateId()
{
    switch (CommandId)
    {
    case 0:
      return IDD_POPUPDIALOG;
      break;
    default:
      return NULL;
      break;
    }
}2.2.4 消息事件处理
         命令激活时,会涉及到一些消息事件的处理,可以在用户自定义的类中完成对这一部分的处理,比如鼠标事件,此处也可能会涉及到一些SDK的使用,可以参照SDK的帮助文档
HRESULT TGUserCmdPopup::MouseClick(short sButton, short sShift, double dX, double dY, double dZ, LPDISPATCH pWindowDispatch, long lKeyPointType, LPDISPATCH pGraphicDispatch)
{
    HRESULT hr = NOERROR;
    AfxMessageBox(L"MouseClick");
    return hr;
}         在弹框界面中也一样,鼠标点击Button1,实现弹出"Click Button1"的功能
void TGPopupDialog::OnTestButton1()
{
    AfxMessageBox(L"Click Button1");
}
         鼠标点击确认,实现销毁窗口并终止命令的功能
void TGPopupDialog::OnOK()
{
    TGAddinTemplate2App::GetApplication()->AbortCommand(false);
}
2.2.5 命令注册
         在TGCmdMnager命令管理类中完成命令的注册
//此处为用户自定义命令实例化与实例化自定义命令和CommandID关联处
//1.请实例化自定义界面实例
//2.请将实例化自定义界面与关联的CommandID放入Map容器
void TGCmdManager::InitCommandMap()
{
    //声明并创建命令的Com对象,并完成引用计数
    TGCmdPopupObj* PopupCommand = NULL;
    TGCmdPopupObj::CreateInstance(&PopupCommand);
    PopupCommand->AddRef();
    //插入命令集合中
    InsertCommand(0, PopupCommand);//该命令ID(0)要求唯一,且需要和Ribbon菜单中的ID保持一致
}
         如需要多个弹窗式命令按钮,重复上述操作即可。
2.3 事务管理
2.3.1 命令管理的消息事件
         这类事件是在对应的派生类命令中进行处理的,可通过补充这些事件的处理方法来完成。
鼠标事件:
         sButton:鼠标按键,表示那个哪个鼠标按钮被按下。通常,1表示左键,2表示右键,4表示中间键
         sShift:按下鼠标时哪些Shift键(1)、Ctrl键(2)或Alt键(4)被按下。
         dX, dY, dZ:分别表示鼠标在X、Y、Z轴上的位置
         pWindowDispatch:指向触发事件的窗口对象的指针。这可以用来访问窗口对象的方法和属性。
         lKeyPointType:表示关键点类型,可能用于识别特定的交互点类型
         pGraphicDispatch:指向图形对象的指针,可以用来访问与绘图或图形相关的服务和操作
// 鼠标点击事件
STDMETHOD(MouseClick)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }
// 鼠标双击事件
STDMETHOD(MouseDblClick)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }
// 鼠标拖拽事件
STDMETHOD(MouseDrag)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }
// 鼠标按下事件
STDMETHOD(MouseDown)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }
// 鼠标弹起事件
STDMETHOD(MouseUp)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }
// 鼠标移动事件
STDMETHOD(MouseClick)(short sButton,short sShift,double dX,double dY,double dZ,LPDISPATCH pWindowDispatch,long lKeyPointType,LPDISPATCH pGraphicDispatch)
{ }键盘事件:
         KeyAscii:当用户按下键盘上的一个键时,对应的ASCII**被传递给KeyAscii
         KeyCode:表示用户释放的虚拟键码。虚拟键码是一个整数,代表了键盘上的一个键
         Shift:表示在释放按键时哪些键(Shift=1、Ctrl=2、Alt=4)被按下
// 键盘按键事件
STDMETHOD(KeyPress)   (short* KeyAscii){ }
// 键盘按下事件
STDMETHOD(KeyDown)(short* KeyCode,short Shift){ }
// 键盘弹起事件
STDMETHOD(KeyUp)(short* KeyCode,short Shift){ }
2.3.2 EventManage管理的消息事件

         这类事件是在TGEventManager管理类中对应的事件实现里去补充。
命令事件:
基于不同版本会有不同的事件类负责处理,如XAddInEvents、XAddInEventsEx、XAddInEventsEx2。
         nCmdID:绑定的命令标识符,即命令ID。
         hFrameWnd:应用程序主框架窗口的句柄。
         uHelpCommand:请求帮助的类型或命令。
// 相应UI界面中的命令事件,比如菜单按钮的点击
STDMETHOD(raw_OnCommand)(long nCmdID){}
// 相应UI界面中的帮助命令事件
STDMETHOD(raw_OnCommandHelp)(long hFrameWnd,long uHelpCommand,long nCmdID){}
侧边栏事件:
         theDocument:指向文档对象的指针。
         EnvironmentCatID:表示一个环境类别ID,用于分类或描述页面展示环境的一个字符串标识符。
         vbIsPageDisplayable:传出参数,表示页面是否在给定的环境类别下可以显示。
// 添加侧边栏
STDMETHOD(raw_AddPage)(LPDISPATCH theDocument){}
// 删除侧边栏
STDMETHOD(raw_RemovePage)(LPDISPATCH theDocument){}
// 用于判断一个页面是否可以在特定环境下显示
STDMETHOD(raw_IsPageDisplayable)(IDispatch* theDocument,BSTR EnvironmentCatID,VARIANT_BOOL* vbIsPageDisplayable);
{ }文档事件:
// 激活文档切换事件
STDMETHOD(raw_AfterActiveDocumentChange) (LPDISPATCH theDocument){}
// 命令运行之后的事件,命令执行完之后,无论成功失败都会触发该事件
STDMETHOD(raw_AfterCommandRun) (long theCommandID){}
// 文档打开之后的事件
STDMETHOD(raw_AfterDocumentOpen) (LPDISPATCH theDocument){}
// 文档保存完成后的事件处理
STDMETHOD(raw_AfterDocumentSave) (LPDISPATCH theDocument){}
// 新建文档并打开后的事件处理
STDMETHOD(raw_AfterNewDocumentOpen) (LPDISPATCH theDocument){}
// 命令激活之前的事件
STDMETHOD(raw_BeforeCommandRun) (long theCommandID){}
// 文档关闭之前的事件
STDMETHOD(raw_BeforeDocumentClose) (LPDISPATCH theDocument){}
// 文档保存之前的操作
STDMETHOD(raw_BeforeDocumentSave) (LPDISPATCH theDocument){}


页: [1]
查看完整版本: 【快速入门教程二】天工CAD插件开发模板介绍