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 8.64 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

schultezub's avatar
schultezub committed
40
namespace campvis {
41

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

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

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

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

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

77
78
        _pipelineWidget = new PipelineTreeWidget();
        ui.pipelineTreeDock->setWidget(_pipelineWidget);
79

80
81
        QWidget* rightWidget = new QWidget();
        ui.pipelinePropertiesDock->setWidget(rightWidget);
82
83
84
85
86
87
88
89
90
91

        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);
92
93

        _propCollectionWidget = new PropertyCollectionWidget(_centralWidget);
94
        rightLayout->addWidget(_propCollectionWidget);
95

96
97
        _logViewer = new QTextEdit();
        _logViewer->setReadOnly(true);
98
        ui.logViewerDock->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
99
        ui.logViewerDock->setWidget(_logViewer);
100

101
102
        _dcInspectorWidget = new DataContainerInspectorWidget();

103
104
105
106
107
108
109
        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(
schultezub's avatar
schultezub committed
110
111
            this, SIGNAL(updatePropCollectionWidget(HasPropertyCollection*)),
            _propCollectionWidget, SLOT(updatePropCollection(HasPropertyCollection*)));
112
113
114
115
116
117
        connect(
            _btnExecute, SIGNAL(clicked()), 
            this, SLOT(onBtnExecuteClicked()));
        connect(
            _btnShowDataContainerInspector, SIGNAL(clicked()), 
            this, SLOT(onBtnShowDataContainerInspectorClicked()));
118
119
120
121
122
123
124
        _application->s_PipelinesChanged.connect(this, &MainWindow::onPipelinesChanged);
    }

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

125
126
127
128
129
    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*>());
130
            emit updatePropCollectionWidget(ptr);
131
132
133
134
135

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

145
146
147
148
    QSize MainWindow::sizeHint() const {
        return QSize(800, 450);
    }

149
150
151
152
    void MainWindow::onBtnExecuteClicked() {
        if (_selectedProcessor != 0 && _selectedPipeline != 0) {
            // this is not as trivial as it seems:
            // We need the pipeline, probably an OpenGL context...
153
            _selectedProcessor->invalidate(AbstractProcessor::INVALID_RESULT);
154
155
        }
        else if (_selectedPipeline != 0) {
156
157
158
            for (std::vector<AbstractProcessor*>::const_iterator it = _selectedPipeline->getProcessors().begin(); it != _selectedPipeline->getProcessors().end(); ++it) {
                (*it)->invalidate(AbstractProcessor::INVALID_RESULT);
            }
159
160
161
162
163
        }
    }

    void MainWindow::onBtnShowDataContainerInspectorClicked() {
        if (_selectedPipeline != 0) {
164
165
166
167
168
169
170
171
            if (_dcInspectorDock == 0) {
                _dcInspectorDock = dockPrimaryWidget("Data Container inspector", _dcInspectorWidget);
            } else {
                // Activate the dock's tab
                _dcInspectorDock->setVisible(true);
                _dcInspectorDock->raise();
            }

172
173
174
175
            _dcInspectorWidget->setDataContainer(&(_selectedPipeline->getDataContainer()));
        }
    }

176
    void MainWindow::init() {
177
178
        if (_dcInspectorWidget != 0)
            _dcInspectorWidget->init();
179
180
181
182

        _log = new QTextEditLog(_logViewer);
        _log->addCat("", true);
        LogMgr.addLog(_log);
183
184
185
    }

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

        LogMgr.removeLog(_log);
190
191
    }

192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
    void MainWindow::addVisualizationPipelineWidget(const std::string& name, QWidget* widget) {
        dockPrimaryWidget(name, widget);
    }

    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;
    }

214
}