VTK交互之vtkCommand

本文介绍VTK中利用观察者/命令模式实现交互的两种方法:通过定义回调函数和继承vtkCommand类。通过示例详细展示了如何实现左键点击事件。

观察者/命令模式

  VTK的交互除了可以监听来自鼠标、键盘等外部设备的消息,还可以在渲染场景中生成功能各异的交互部件(Widget),用于控制可化过程的参数,达到用户渲染要求。
   在VTK中要实现与数据的交互,可以基于观察者/命令模式。可以通过两种方法来实现该模式:一种是通过定义回调函数;另外一种是通过从vtkCommand类派生子类。二者的实现过程基本类似,都是通过AddObserver监听感兴趣的时间,然后在回调函数或者vtkCommand::Execute()函数里实现所需的功能。
   观察者/命令模式(Observer/Command)是VTK里用得较多的设计模式。VTK中绝大多数的类派生自vtkObject。查看vtkObject的接口可以找到AddObserver()、RemoveObserver()、GetCommand()等函数。

一、事件回调函数

VTK里使用回调函数实现观察者/命令模式主要分为以下三个步骤。
1、定义回调函数。回调函数的函数签名只能是以下形式:

void func(vtkObject* obj, unsigned long eid, void* clientdata, void* calldata)

其中

  • obj是调用事件的对象(即调用AddObserver()函数的主题对象,例如interactor);
  • eid为所要监听的事件ID,VTK中的事件定义在vtkCommand.h文件中;
  • clientdata是与vtkCalbackCommand实例相关的数据。简单说,是指回调函数里需要访问主程序里的数据时,由主程序向回调函数传递的数据,可以通过vtkCallbackCommand::SetClientData()函数设置。
  • calldata是执行vtkObject::InvokeEvent()函数时,随着回调函数发送的数据,比如当调用ProgressEvent事件时,会自动发送当前的进度值作为calldata。

2、创建一个vtkCallbackCommand对象,并调用vtkCallbackCommand::SetCallback()函数设置所定义的回调函数。

vtkSmartPointer<vtkCallbackCommand> mouseCallback = vtkSmartPointer<vtkCallbackCommand>::New();
mouseCallback->SetCallback(MyCallbackFunc);

3、将vtkCallbackCommand对象添加到主题对象的观察者列表中。

interactor->AddObserver(vtkCommand::LeftButtonPressEvent, mouseCallback);

示例说明

CMakeLists.txt文件代码如下:

CMAKE_MINIMUM_REQUIRED(VERSION 2.8)
PROJECT(CallbackFunction)
FIND_PACKAGE(VTK REQUIRED)
INCLUDE(${VTK_USE_FILE})
ADD_EXECUTABLE(CallbackFunction    CallbackFunction.cpp)
TARGET_LINK_LIBRARIES(CallbackFunction ${VTK_LIBRARIES})

CallbackFunction.cpp文件代码如下:

#include <vtkSmartPointer.h>
#include <vtkPNGReader.h>
#include <vtkImageViewer2.h>
#include <vtkRenderWindowInteractor.h>
#include <vtkCallbackCommand.h>
#include <vtkRenderWindow.h>
#include <vtkRenderer.h>

long pressCounts = 0;

//第一步,定义回调函数。注意回调函数的签名,不能更改。
void MyCallbackFunc(vtkObject*, unsigned long eid, void* clientdata, void *calldata)
{
    std::cout<<"You have clicked: "<<++pressCounts<<" times."<<std::endl;
}

int main()
{
    vtkSmartPointer<vtkPNGReader> reader = vtkSmartPointer<vtkPNGReader>::New();
    reader->SetFileName("E:\\TestData\\VTK-logo.png");

    vtkSmartPointer<vtkImageViewer2> viewer = vtkSmartPointer<vtkImageViewer2>::New();
    viewer->SetInputConnection(reader->GetOutputPort());

    vtkSmartPointer<vtkRenderWindowInteractor> interactor =  vtkSmartPointer<vtkRenderWindowInteractor>::New();
    viewer->SetupInteractor(interactor);
    viewer->Render();

    viewer->GetRenderer()->SetBackground(1.0, 1.0, 1.0);
    viewer->SetSize(640, 480);
    viewer->GetRenderWindow()->SetWindowName("事件回调函数");

    //第二步,设置回调函数。
    vtkSmartPointer<vtkCallbackCommand> mouseCallback = 
        vtkSmartPointer<vtkCallbackCommand>::New();
    mouseCallback->SetCallback ( MyCallbackFunc );

    //第三步,将vtkCallbackCommand对象添加到观察者列表。
    interactor->SetRenderWindow(viewer->GetRenderWindow());
    interactor->AddObserver(vtkCommand::LeftButtonPressEvent, mouseCallback);

    interactor->Initialize();
    interactor->Start();

    return EXIT_SUCCESS;
}

运行结果:
这里写图片描述

二、vtkCommand子类

写vtkCommand的子类来实现观察者/命令模式,分为以下三个步骤。
1、从vtkCommand类派生出子类,并实现vtkCommand::Execute()函数,该函数原型是:

virtual void Exeute(vtkObject* caller, unsigned long eventId, void* callData) = 0;

Execute()是纯虚函数,所以从vtkCommand派生的类都必须实现这个方法。

2、在主程序中实例化一个vtkCommand子类的对象以及调用相关的方法。

3、调用AddObserver()函数监听感兴趣的事件。

示例说明

CMakeLists.txt文件代码如下:

CMAKE_MINIMUM_REQUIRED(VERSION 2.8)
PROJECT(Command)
FIND_PACKAGE(VTK REQUIRED)
INCLUDE(${VTK_USE_FILE})
ADD_EXECUTABLE(Command    Command.cpp)
TARGET_LINK_LIBRARIES(Command ${VTK_LIBRARIES})

Command.cpp文件代码如下:

#include <vtkSmartPointer.h>
#include <vtkConeSource.h>
#include <vtkPolyDataMapper.h>
#include <vtkRenderWindow.h>
#include <vtkRenderWindowInteractor.h>
#include <vtkCamera.h>
#include <vtkActor.h>
#include <vtkRenderer.h>
#include <vtkCommand.h>
#include <vtkBoxWidget.h>
#include <vtkTransform.h>
#include <vtkInteractorStyleTrackballCamera.h>

//第一步
class vtkMyCommand : public vtkCommand
{
public:
    static vtkMyCommand *New()
    {
        return new vtkMyCommand;
    }

    void SetObject(vtkConeSource* cone)
    {
        m_Cone = cone;
    }

    virtual void Execute(vtkObject *caller, unsigned long eventId, void* callData)
    {
        std::cout<<"Left button pressed.\n"
            <<"The Height: "<<m_Cone->GetHeight()<<"\n"
            <<"The Radius: "<<m_Cone->GetRadius()<<std::endl;
    }

private:
    vtkConeSource *m_Cone;
};

int main()
{
    vtkSmartPointer<vtkConeSource> cone = vtkSmartPointer<vtkConeSource>::New();
    cone->SetHeight( 3.0 );
    cone->SetRadius( 1.0 );
    cone->SetResolution( 10 );

    vtkSmartPointer<vtkPolyDataMapper> coneMapper = 
        vtkSmartPointer<vtkPolyDataMapper>::New();
    coneMapper->SetInputConnection( cone->GetOutputPort() );

    vtkSmartPointer<vtkActor> coneActor = vtkSmartPointer<vtkActor>::New();
    coneActor->SetMapper( coneMapper );

    vtkSmartPointer<vtkRenderer> ren1= vtkSmartPointer<vtkRenderer>::New();
    ren1->AddActor( coneActor );
    ren1->SetBackground( 0.1, 0.4, 0.9 );

    vtkSmartPointer<vtkRenderWindow> renWin =
        vtkSmartPointer<vtkRenderWindow>::New();
    renWin->AddRenderer( ren1 );
    renWin->SetSize( 640, 480 );
    renWin->Render();
    renWin->SetWindowName("vtkCommand子类");

    vtkSmartPointer<vtkRenderWindowInteractor> iren = 
        vtkSmartPointer<vtkRenderWindowInteractor>::New();
    iren->SetRenderWindow(renWin);

    vtkSmartPointer<vtkInteractorStyleTrackballCamera> style = 
        vtkSmartPointer<vtkInteractorStyleTrackballCamera>::New();
    iren->SetInteractorStyle(style);

    //第二步
    vtkSmartPointer<vtkMyCommand> callback = vtkSmartPointer<vtkMyCommand>::New();
    callback->SetObject(cone);

    //第三步
    iren->AddObserver(vtkCommand::LeftButtonPressEvent, callback);

    iren->Initialize();
    iren->Start();

    return EXIT_SUCCESS;
}

运行结果:
这里写图片描述

总结:

  在VTK中,用观察者/命令模式实现交互,一般有两种方法:1、实例化vtkCallbackCommand,将自己写的回调函数,设给其;2、从vtkCommand类派生出子类,重写Execute方法。

<think>我们正在处理一个关于在C#中使用VTK时监测不到鼠标松开事件的问题。根据用户提供的参考引用,虽然引用的是Qt相关的问题,但我们可以借鉴类似问题的解决思路。在Qt中,有时在场景中添加QWidget窗口会导致无法捕捉mouseReleaseEvent,这通常是因为事件被上层拦截或处理不当。类似地,在VTK中,鼠标事件的处理也可能因为事件传递机制或交互器设置的问题而无法正确捕获松开事件。 解决思路: 1. 确保交互器正确设置:在VTK中,我们需要确保交互器(Interactor)正确设置,并且启用了鼠标事件的监听。 2. 检查事件回调函数:确认是否正确添加了鼠标松开事件的回调函数,并且函数本身没有错误。 3. 事件传递机制:在VTK中,事件可能被其他事件处理程序拦截,导致无法到达松开事件的处理函数。我们可以尝试调整事件处理的优先级或使用PassDataOn()方法将事件传递下去。 4. 更新交互器:有时,交互器需要更新以响应事件。 具体步骤: 步骤1:确认已经为交互器添加了鼠标松开事件的回调函数。 步骤2:在回调函数中,确保能够正确响应事件。可以尝试在回调函数中加入日志输出,以确认事件是否被触发。 步骤3:如果事件被其他事件处理程序拦截,可以在回调函数中调用vtkRenderWindowInteractor的PassDataOn()方法,以确保事件继续传递。 步骤4:确保在事件处理过程中,没有因为异常导致程序中断。 示例代码: 以下是一个简单的示例,展示如何在C# VTK中为鼠标松开事件添加回调: ```csharp using System; using Kitware.VTK; class Program { static void Main(string[] args) { // 创建渲染窗口、渲染器等(省略部分初始化代码) vtkRenderWindow renderWindow = vtkRenderWindow.New(); vtkRenderWindowInteractor interactor = vtkRenderWindowInteractor.New(); interactor.SetRenderWindow(renderWindow); // 创建回调函数 vtkMyCallback callback = new vtkMyCallback(); // 添加鼠标松开事件的回调 interactor.AddObserver((uint)EventIds.LeftButtonReleaseEvent, callback); // 开始交互 interactor.Start(); } } public class vtkMyCallback : vtkCommand { public override void Execute(vtkObject caller, uint eventId, IntPtr callData) { if (eventId == (uint)EventIds.LeftButtonReleaseEvent) { Console.WriteLine("Mouse Release Event Detected!"); } } } ``` 注意:这里使用了枚举EventIds.LeftButtonReleaseEvent来表示鼠标左键松开事件。如果无法找到这个枚举,可以直接使用数字(VTK中该事件对应的数字是4,但建议使用枚举更安全)。 步骤5:如果以上方法无效,尝试在回调函数中调用事件传递: 在回调函数中,可以尝试获取交互器,并调用其事件传递方法。但是,在VTK的C#版本中,通常不需要手动传递,因为回调本身已经是在事件处理链中。不过,如果发现事件被提前处理了,可以考虑在回调函数中设置事件已处理标志为false,以便其他观察者继续处理。但VTK的回调机制是观察者模式,多个观察者都会收到事件,除非在某个回调中标记事件已处理(但默认是不标记为已处理的)。 步骤6:检查是否是因为窗口焦点问题,导致鼠标事件未被正确捕获。例如,如果鼠标在窗口外释放,那么可能无法捕获事件。确保在窗口内进行测试。 步骤7:更新VTK版本。如果使用的是较旧的VTK版本,可能存在已知的bug,更新到最新版本可能解决问题。 步骤8:检查是否有其他事件处理程序(例如,鼠标按下事件处理程序中)调用了中止事件传播的方法,导致后续事件无法触发。 根据用户之前的问题,我们曾经提到过Qt中类似问题的解决方法,即需要在QWidget中设置属性Qt::WA_NativeWindow,以确保窗口可以正确捕获事件。虽然VTK不是Qt,但我们可以考虑是否是因为VTK渲染窗口的某些属性设置问题。在VTK中,我们可以尝试设置渲染窗口的某些属性,比如启用窗口的鼠标事件捕获,但通常在创建交互器时已经默认设置了。 如果问题依然存在,可能需要深入VTK的事件处理机制,或者考虑使用调试工具跟踪事件流。 相关问题: 1. 在C# VTK中如何添加鼠标按下事件? 2. VTK交互事件处理的基本流程是什么? 3. 如何调试VTK中的事件处理问题? 4. 在VTK中,如何设置事件回调的优先级? 5. 除了鼠标松开事件,VTK还支持哪些交互事件? 请注意,以上代码示例是简化的,实际使用时需要完整的VTK环境。另外,VTK的C#版本与C++版本在事件处理上类似,但要注意C#中的事件枚举和调用方式。 </think> ### C# VTK 鼠标松开事件监测问题解决方案 在 C# 中使用 VTK 时无法监测鼠标松开事件(`MouseReleaseEvent`),通常与事件处理机制或交互器设置相关。以下是逐步解决方案: --- ### 原因分析 1. **事件过滤器问题**:VTK 交互器或父容器可能拦截了事件 2. **回调注册错误**:事件回调未正确绑定到特定事件类型 3. **焦点问题**:渲染窗口未获得焦点时无法接收释放事件 4. **VTK 版本差异**:不同 VTK 版本对事件处理机制有差异 --- ### 解决方案 #### 方法 1:正确注册事件回调 ```csharp // 创建交互vtkRenderWindowInteractor interactor = vtkRenderWindowInteractor.New(); interactor.SetRenderWindow(renderWindow); // 创建自定义回调类 public class MouseCallback : vtkCommand { public override void Execute(vtkObject caller, uint eventId, IntPtr callData) { if (eventId == (uint)EventIds.LeftButtonReleaseEvent) // 鼠标左键释放 { Console.WriteLine("Mouse released!"); } } } // 注册回调(关键步骤) MouseCallback callback = new MouseCallback(); interactor.AddObserver((uint)EventIds.LeftButtonReleaseEvent, callback); ``` #### 方法 2:检查事件过滤器 ```csharp // 确保父控件不会拦截事件 yourParentControl.SetStyle(ControlStyles.EnableNotifyMessage, true); yourParentControl.SetStyle(ControlStyles.UserMouse, true); ``` #### 方法 3:强制焦点设置 ```csharp // 在鼠标按下时强制获取焦点 interactor.AddObserver((uint)EventIds.LeftButtonPressEvent, (caller, eventId, data) => { renderWindow.SetCurrentCursor(VTK_CURSOR_HAND); // 更改光标 yourRenderControl.Focus(); // 强制焦点 }); ``` #### 方法 4:使用 VTK 事件管道(高级方案) ```csharp // 创建交互样式 vtkInteractorStyleTrackballCamera style = vtkInteractorStyleTrackballCamera.New(); interactor.SetInteractorStyle(style); // 重写事件处理 public class CustomStyle : vtkInteractorStyleTrackballCamera { public override void OnLeftButtonUp() { Console.WriteLine("Custom release handling"); base.OnLeftButtonUp(); // 必须调用基类方法 } } ``` --- ### 调试建议 1. **检查事件ID**:确认使用正确的 VTK 事件常量 - `LeftButtonReleaseEvent = 4` - `RightButtonReleaseEvent = 5` 2. **版本兼容性**:VTK 9.x 使用 `AddObserver` 代替旧版 `SetCallback` 3. **最小化测试**:新建空白项目仅包含事件测试代码 4. **日志输出**:在 `MousePress` 和 `MouseRelease` 事件中都添加日志 > 注意:在 Windows 平台需检查 DPI 缩放设置,高DPI环境下可能导致坐标映射错误[^1] --- ### 示例:完整事件处理流程 ```csharp vtkRenderWindow renderWindow = ... // 初始化渲染窗口 vtkRenderWindowInteractor interactor = vtkRenderWindowInteractor.New(); interactor.SetRenderWindow(renderWindow); // 注册事件 interactor.AddObserver((uint)EventIds.LeftButtonPressEvent, new MouseCallback("Pressed")); interactor.AddObserver((uint)EventIds.LeftButtonReleaseEvent, new MouseCallback("Released")); // 回调实现 public class MouseCallback : vtkCommand { private string _msg; public MouseCallback(string msg) => _msg = msg; public override void Execute(vtkObject caller, uint eid, IntPtr data) { Console.WriteLine($"{_msg} at {interactor.GetEventPosition()}"); } } ``` ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值