2013-08-08 7 views
0

저는 C++로 작성된 작은 시각화 프레임 워크를 가지고 있으며 적절한 GUI를 가지고 시각화를 제어하기 위해 Qt를 사용하려고합니다. 현재 GLUT을 사용하여 창을 만들고보기를 그립니다. 그래서 모든 것은 저에게 모든 것을 해주는 클래스 시각화의 객체를 초기화하는 것입니다 : 모델과 뷰를 유지하는 것. 뷰 자체에는 사용자 입력을 처리하고 모델을 조작하는 컨트롤러가 있습니다. 내 메인 루프는 다음과 같습니다 GLUT와Qt를 미리 작성된 응용 프로그램/프레임 워크에 통합하십시오.

Visualization* vis = new Visualization(); 
vis->createModel("some_file.txt"); 
vis->createView("unknown"); 

// ... 

void demo_app::display() 
{ 
    // clear the framebuffer 
    glClearColor(0.3, 0.3, 0.3, 1.0); 
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); 

    vis.update(); // looks for dirty scenes to update 
    vis.render(); // draws all views which are stored in vis 

    glutSwapBuffers(); 
} 

은 내가 여러 뷰를 정렬하는 하나의 창을 가지고있다. 내가 메인 루프 동안 glCommands를 실행할 수있는 무승부보기를 주문하고 모든 것이 좋습니다.

이제 내 문제는 Qt를 여러 창 (QDialogs + QGLWidgets)과 함께 사용하고 싶습니다. 나는 Qt에 익숙하며 이미 Qt에 프레임 워크를 이식했다. 그러나 문제는 프레임 워크 내에서 너무 많은 Qt 통합을 수행해야한다는 것이고 이것이 내가 원하는 것이 아니란 것입니다. Qt와 GUI 요소를 사용하여 시각화를 제어하려고하지만 뷰의 그리기 호출은 GLUT 예제에 표시된 것처럼 시각화 클래스에서 수행해야합니다. 따라서 상속 된 QGLWidget에 뷰 객체에 대한 포인터를주는 방법이 있어야하며 뷰를 그릴 때마다 위젯은 makeCurrent()을 호출해야합니다. 그러면 glCommands가 컨텍스트에서 그릴 수 있습니다. 내 뷰에서 현재 장면을 칠한 다음 QGLWidget의 paintGL 또는 함수에서 텍스처를 그릴 수 있지만 뷰에 대해 알지 못하면 update()으로 위젯을 어떻게 말할 수 있습니까? QTimer이 0으로 설정된 메인 루프를 에뮬레이션합니다. 내가하고 싶은 것은 프레임 워크 내에서 QGLWidget의 렌더링을 트리거 할 수있는 것입니다. 어떠한 제안? 모래밭? 예?

답변

0

그래서 상속 QGLWidget에게 뷰 객체에 대한 포인터와보기를

당신이 QGLWidget::paintGL 내에서 demo_app::display()를 호출하는 경우

을 그려 져야 할 때마다, 그것이 작동하고 그릴 것을 줄 수있는 방법이 있어야합니다 장면 QGLWidget 컨텍스트에. 그러나 이것은 QGLWidget의 컨텍스트를 사용하므로 모든 초기화 함수를 QGLWidget으로 옮겨야합니다. OpenGL 객체가 컨텍스트에서 공유 될 수 있는지 여부는 기억이 나지 않지만 그렇게 할 수 없다면 실제로 놀라지 않을 것입니다.

하지만 뷰에 대한 지식이없는 경우 어떻게 위젯에 update()를 알릴 수 있습니까?

글쎄, QWidget에 대한 포인터를보기에 추가하고 QWidget->update()을 호출하십시오. 명백해야합니다.

당신이 GLUT 기반의 창 내에서 QT 위젯을 그리려는 경우 ...

즉, Qt 위젯을 GLUT 윈도우에 "포함"하고 싶다면.

모든 Qt 위젯에는 render 메서드가 있습니다. 이 메소드를 사용하여 QPaintDevice에 위젯을 그리거나 QPainter을 사용하여 위젯을 그릴 수 있습니다.

프레임 워크 내에서 위젯을 그리려면 위젯 중 하나를 제공하고 렌더링을 수동으로 호출해야합니다.

또한 프레임 워크에서 마우스 및 키보드 이벤트를 전달하고 Qt 이벤트로 변환해야합니다.

실행 가능해야하지만 구현하는 데 큰 어려움이 있습니다.

EmbeddedDialogs 데모 및 QGraphicsProxyWidget을 참조하십시오.

단순히 과잉 창으로 포함시키지 않고, 창을 과잉뿐만 아니라 GUI 위젯을 사용하려면 ...

랩의 모든 QObject으로 프레임 워크와 과잉 메시지에 관련 넣어 과잉의 메시지 처리 이 물체의 슬롯에 이 슬롯을 0 시간 초과로 실행되는 타이머에 연결하십시오. 내 코드에서

예 (SDL 사용) :

class Game: public QObject{ 
Q_OBJECT 
protected: 
    void processSdlEvents(); 
... 
protected slots: 
    void onGameEnd(); 
    void timerSlot(); 
... 
}; 

Game::Game(/*arguments*/ ...){ 
... 
    gameTimer = new QTimer(this); 
    gameTimer->setSingleShot(false); 
    gameTimer->setInterval(0); 
    connect(gameTimer, SIGNAL(timeout()), this, SLOT(timerSlot())); 
... 
} 

void Game::timerSlot(){ 
    processSdlEvents(); 
    update(); 
    render(); 
} 

void Game::processSdlEvents(){ 
    SDL_Event event; 
    QList<GameEventHandler*> eventHandlers = findChildren<GameEventHandler*>(); 
    /*WARNING: This is NOT an infinite loop. */ 
    while (SDL_PollEvent(&event) != 0){ 
     if (event.type == SDL_QUIT){ 
      emit gameEnded(); 
      continue; 
     } 
     bool processed = false; 
     for (int i = 0; i < eventHandlers.size(); i++){ 
      if (eventHandlers[i]->processEvent(&event)){ 
       processed = true; 
       break; 
      } 
     } 
     if (processed) 
      continue; 
     for (int i = 0; i < joysticks.size(); i++){ 
      if (joysticks[i]->processEvent(&event)){ 
       processed = true; 
       break; 
      } 
     } 
     /*if (processed) 
      continue;*/ 
    } 
} 


int main(int argc, char** argv){ 
    QApplication app(argc, argv); 
    int result = 0; 
    try{ 
     Game game; 
     QObject::connect(&game, SIGNAL(gameEnded()), &app, SLOT(quit())); 
     game.start(); 
     result = app.exec(); 
    } 
    catch(const QString& s){ 
     reportException(s); 
    } 
    catch(std::exception& e){ 
     reportException(e); 
    } 
    catch(...){ 
     reportException(); 
    } 
    return result; 
} 

processSdlEvents 무한 루프 아님을 유의하시기 바랍니다. 그것은 한 번 호출되고 마지막 호출 이후에 수신 된 모든 이벤트를 처리 한 다음 종료됩니다.

이 시나리오에서 프레임 워크 특정 기능은 Qt 메인 루프에서 호출됩니다.

또한 아마 주위에 그것을 다른 방법을하고 (QEventLoop 클래스 또는 QApplication::processEvents()를 사용하여) 자신의 프레임 워크에서 메인 루프에서 Qt는 특정 함수를 호출하지만, 덜 신뢰할 수 있습니다 내가 나 자신 것을 시도하지 않은 수 있습니다.

+0

QGLWidget 내에서 내 demo_app :: display()를 호출하고 싶지 않습니다. 이것은 메인 루프에서 실행되어야하며 뷰는 스스로를 그려야합니다. 나는'viewGL()'이 나의 주요 디스플레이 기능을 트리거하기를 원하지 않는다. 왜냐하면 다른 뷰를 가진 다중'QGLWidgets'이 있어야하기 때문이다. –

+0

@iam_peter : "이것은 메인 루프에서 실행되어야합니다."음, 아니요. 틀렸어요. 메인 루프 내에서 "디스플레이"를 수동으로 호출 할 이유가 없습니다. (** 아마 ** 당신이 전체 화면 어플리케이션을 사용하지 않는다면). Qt는 이런 식으로 작동하지 않습니다. 위젯은 필요할 때'paint' /'paintGL'를 호출합니다. 연속적으로 다시 그리기를 원하면'update '를 계속 호출하십시오. 메인 루프 내에서 여러 위젯을 연속적으로 다시 칠하면 CPU가 낭비되고 유용하지 않습니다. – SigTerm

+0

표시 기능은 일부 모델 변경으로 인해 어떤보기를 다시 그려야하는지 결정합니다. 이 메커니즘은 Qt로 보내져서는 안됩니다. 이것은 필자의 시각화 클래스에 의해 수행되어야하는데, 이는 필자가 메인 루프에서 display/update를 호출하는 이유입니다. 디스플레이 호출 중 뷰가 다시 그려지기 때문에 이것은 Qt에 의해 트리거되어서는 안됩니다. Qt가해야 할 모든 일은 사용자 상호 작용에 대한 내 관점을 알려줌으로써 뷰를 더럽게 설정하는 것입니다. 내 스케줄러 (= 시각화)가 다른 모든 것을 시작해야합니다. –