Currently job artifacts in CI/CD pipelines on LRZ GitLab never expire. Starting from Wed 26.1.2022 the default expiration time will be 30 days (GitLab default). Currently existing artifacts in already completed jobs will not be affected by the change. The latest artifacts for all jobs in the latest successful pipelines will be kept. More information: https://gitlab.lrz.de/help/user/admin_area/settings/continuous_integration.html#default-artifacts-expiration

mainwindow.cpp 9.13 KB
Newer Older
1
2
// ================================================================================================
// 
schultezub's avatar
schultezub committed
3
// This file is part of the CAMPVis Software Framework.
4
5
// 
// If not explicitly stated otherwise: Copyright (C) 2012, all rights reserved,
schultezub's avatar
schultezub committed
6
//      Christian Schulte zu Berge <christian.szb@in.tum.de>
7
8
9
//      Chair for Computer Aided Medical Procedures
//      Technische Universitt Mnchen
//      Boltzmannstr. 3, 85748 Garching b. Mnchen, Germany
schultezub's avatar
schultezub committed
10
// For a full list of authors and contributors, please refer to the file "AUTHORS.txt".
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
// 
// The licensing of this softare is not yet resolved. Until then, redistribution in source or
// binary forms outside the CAMP chair is not permitted, unless explicitly stated in legal form.
// However, the names of the original authors and the above copyright notice must retain in its
// original state in any case.
// 
// Legal disclaimer provided by the BSD license:
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY 
// AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR 
// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 
// OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
// 
// ================================================================================================

30
31
32
#include "mainwindow.h"

#include "tgt/assert.h"
schultezub's avatar
schultezub committed
33
#include "application/campvisapplication.h"
34
#include "application/gui/datacontainerinspectorwidget.h"
35
#include "application/gui/datacontainerinspectorcanvas.h"
36
#include "application/gui/qtdatahandle.h"
37
38
#include "core/pipeline/abstractpipeline.h"
#include "core/pipeline/abstractprocessor.h"
39

40
41
#include <QMdiSubWindow>

schultezub's avatar
schultezub committed
42
namespace campvis {
43

44
    MainWindow::MainWindow(CampVisApplication* application)
45
46
        : QMainWindow()
        , _application(application)
47
        , _mdiArea(0)
48
        , _pipelineWidget(0)
49
50
        , _propCollectionWidget(0)
        , _dcInspectorWidget(0)
51
        , _dcInspectorDock(0)
52
53
54
55
        , _btnExecute(0)
        , _btnShowDataContainerInspector(0)
        , _selectedPipeline(0)
        , _selectedProcessor(0)
56
        , _logViewer(0)
57
58
    {
        tgtAssert(_application != 0, "Application must not be 0.");
59
        ui.setupUi(this);
60
61
62
63
        setup();
    }

    MainWindow::~MainWindow() {
schultezub's avatar
schultezub committed
64
        _application->s_PipelinesChanged.disconnect(this);
65
        delete _dcInspectorWidget;
66
67
68
    }

    void MainWindow::setup() {
69
70
        qRegisterMetaType<QtDataHandle>("QtDataHandle");

71
        setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);
72
        setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea);
73
        setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);
74
        setCorner(Qt::TopRightCorner, Qt::RightDockWidgetArea);
75

76
        setTabPosition(Qt::TopDockWidgetArea, QTabWidget::North);
77

78
79
80
81
        _mdiArea = new QMdiArea();
        _mdiArea->tileSubWindows();
        setCentralWidget(_mdiArea);

82
83
        _pipelineWidget = new PipelineTreeWidget();
        ui.pipelineTreeDock->setWidget(_pipelineWidget);
84

85
86
        QWidget* rightWidget = new QWidget();
        ui.pipelinePropertiesDock->setWidget(rightWidget);
87
88
89
90
91
92
93
94
95
96

        QVBoxLayout* rightLayout = new QVBoxLayout();
        rightLayout->setSpacing(4);
        rightWidget->setLayout(rightLayout);

        _btnExecute = new QPushButton("Execute Selected Pipeline/Processor", rightWidget);
        rightLayout->addWidget(_btnExecute);

        _btnShowDataContainerInspector = new QPushButton("Inspect DataContainer of Selected Pipeline", rightWidget);
        rightLayout->addWidget(_btnShowDataContainerInspector);
97

98
        _propCollectionWidget = new PropertyCollectionWidget();
99
        rightLayout->addWidget(_propCollectionWidget);
100

101
        _logViewer = new LogViewerWidget();
102
        ui.logViewerDock->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
103
        ui.logViewerDock->setWidget(_logViewer);
104

105
        _dcInspectorWidget = new DataContainerInspectorWidget();
106
107
108
109
110
111
112
113

        connect(
            this, SIGNAL(updatePipelineWidget(const std::vector<AbstractPipeline*>&)), 
            _pipelineWidget, SLOT(update(const std::vector<AbstractPipeline*>&)));
        connect(
            _pipelineWidget, SIGNAL(clicked(const QModelIndex&)), 
            this, SLOT(onPipelineWidgetItemClicked(const QModelIndex&)));
        connect(
114
115
            this, SIGNAL(updatePropCollectionWidget(HasPropertyCollection*, DataContainer*)),
            _propCollectionWidget, SLOT(updatePropCollection(HasPropertyCollection*, DataContainer*)));
116
117
118
119
120
121
        connect(
            _btnExecute, SIGNAL(clicked()), 
            this, SLOT(onBtnExecuteClicked()));
        connect(
            _btnShowDataContainerInspector, SIGNAL(clicked()), 
            this, SLOT(onBtnShowDataContainerInspectorClicked()));
122
123
124
125
126
127
128
        _application->s_PipelinesChanged.connect(this, &MainWindow::onPipelinesChanged);
    }

    void MainWindow::onPipelinesChanged() {
        emit updatePipelineWidget(_application->_pipelines);
    }

129
130
131
132
133
    void MainWindow::onPipelineWidgetItemClicked(const QModelIndex& index) {
        if (index.isValid()) {
            // Yak, this is so ugly - another reason why GUI programming sucks...
            QVariant item = index.data(Qt::UserRole);
            HasPropertyCollection* ptr = static_cast<HasPropertyCollection*>(item.value<void*>());
134
135
136
137
138

            if (AbstractPipeline* pipeline = dynamic_cast<AbstractPipeline*>(ptr)) {
            	_selectedPipeline = pipeline;
                _selectedProcessor = 0;
            }
schultezub's avatar
schultezub committed
139
140
            else if (AbstractProcessor* processor = dynamic_cast<AbstractProcessor*>(ptr)) {
                _selectedProcessor = processor;
141
142
143
144
145
146

                QVariant parentItem = index.parent().data(Qt::UserRole);
                HasPropertyCollection* pptr = static_cast<HasPropertyCollection*>(parentItem.value<void*>());
                if (AbstractPipeline* pipeline = dynamic_cast<AbstractPipeline*>(pptr)) {
                    _selectedPipeline = pipeline;
                }
schultezub's avatar
schultezub committed
147
            }
148
149

            emit updatePropCollectionWidget(ptr, &_selectedPipeline->getDataContainer());
150
151
        }
        else {
152
            emit updatePropCollectionWidget(0, 0);
153
154
155
        }
    }

156
157
158
159
    QSize MainWindow::sizeHint() const {
        return QSize(800, 450);
    }

160
161
162
163
    void MainWindow::onBtnExecuteClicked() {
        if (_selectedProcessor != 0 && _selectedPipeline != 0) {
            // this is not as trivial as it seems:
            // We need the pipeline, probably an OpenGL context...
164
            _selectedProcessor->invalidate(AbstractProcessor::INVALID_RESULT);
165
166
        }
        else if (_selectedPipeline != 0) {
167
168
169
            for (std::vector<AbstractProcessor*>::const_iterator it = _selectedPipeline->getProcessors().begin(); it != _selectedPipeline->getProcessors().end(); ++it) {
                (*it)->invalidate(AbstractProcessor::INVALID_RESULT);
            }
170
171
172
173
174
        }
    }

    void MainWindow::onBtnShowDataContainerInspectorClicked() {
        if (_selectedPipeline != 0) {
175
176
177
178
179
180
181
182
            if (_dcInspectorDock == 0) {
                _dcInspectorDock = dockPrimaryWidget("Data Container inspector", _dcInspectorWidget);
            } else {
                // Activate the dock's tab
                _dcInspectorDock->setVisible(true);
                _dcInspectorDock->raise();
            }

183
184
185
186
            _dcInspectorWidget->setDataContainer(&(_selectedPipeline->getDataContainer()));
        }
    }

187
    void MainWindow::init() {
188
189
        if (_dcInspectorWidget != 0)
            _dcInspectorWidget->init();
190

191
        _logViewer->init();
192
193
194
    }

    void MainWindow::deinit() {
195
196
        if (_dcInspectorWidget != 0)
            _dcInspectorWidget->deinit();
197

198
        _logViewer->deinit();
199
200
    }

201
    void MainWindow::addVisualizationPipelineWidget(const std::string& name, QWidget* widget) {
202
203
        QMdiSubWindow* subWindow = _mdiArea->addSubWindow(widget);
        subWindow->setWindowTitle(QString::fromStdString(name));
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
    }

    QDockWidget* MainWindow::dockPrimaryWidget(const std::string& name, QWidget* widget) {
        QDockWidget* dockWidget = new QDockWidget(QString::fromStdString(name));
        dockWidget->setWidget(widget);
        dockWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);

        if (_primaryDocks.empty()) {
            addDockWidget(Qt::TopDockWidgetArea, dockWidget);
        } else {
            tabifyDockWidget(_primaryDocks.back(), dockWidget);
            // Activate the dock's tab
            dockWidget->setVisible(true);
            dockWidget->raise();
        }

        _primaryDocks.push_back(dockWidget);
        return dockWidget;
222
223
    }

224
}