2013-09-26 06:17:09 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2013 Digia Plc
|
|
|
|
** All rights reserved.
|
|
|
|
** For any questions to Digia, please use contact form at http://qt.digia.com
|
|
|
|
**
|
|
|
|
** This file is part of the QtDataVisualization module.
|
|
|
|
**
|
|
|
|
** Licensees holding valid Qt Enterprise licenses may use this file in
|
|
|
|
** accordance with the Qt Enterprise License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Digia.
|
|
|
|
**
|
|
|
|
** If you have questions regarding the use of this file, please use
|
|
|
|
** contact form at http://qt.digia.com
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define NOMINMAX
|
|
|
|
|
|
|
|
#include "surfacedata.h"
|
|
|
|
#include "QKinectWrapper.h"
|
2014-01-07 11:11:34 +00:00
|
|
|
#include <QtDataVisualization/QValue3DAxis>
|
2013-09-26 10:30:08 +00:00
|
|
|
#include <QScrollBar>
|
|
|
|
#include <QSize>
|
2013-09-27 04:43:40 +00:00
|
|
|
#include <QImage>
|
2013-09-26 06:17:09 +00:00
|
|
|
|
|
|
|
QT_DATAVISUALIZATION_USE_NAMESPACE
|
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
SurfaceData::SurfaceData(Q3DSurface *surface, Q3DScatter *scatter, Q3DBars *bars,
|
|
|
|
QTextEdit *statusArea) :
|
2013-09-27 05:50:03 +00:00
|
|
|
m_surface(surface),
|
2013-09-27 08:24:42 +00:00
|
|
|
m_scatter(scatter),
|
|
|
|
m_bars(bars),
|
2013-09-27 05:50:03 +00:00
|
|
|
m_statusArea(statusArea),
|
|
|
|
m_resize(true),
|
2013-09-27 08:24:42 +00:00
|
|
|
m_resolution(QSize(320, 240)),
|
|
|
|
m_resolutionLevel(0),
|
|
|
|
m_mode(Surface)
|
2013-09-26 06:17:09 +00:00
|
|
|
{
|
|
|
|
// Initialize surface
|
2013-09-26 10:30:08 +00:00
|
|
|
m_surface->setTheme(QDataVis::ThemeIsabelle);
|
2013-09-26 06:17:09 +00:00
|
|
|
QLinearGradient gradient;
|
|
|
|
gradient.setColorAt(0.0, Qt::black);
|
|
|
|
gradient.setColorAt(0.33, Qt::blue);
|
|
|
|
gradient.setColorAt(0.67, Qt::red);
|
|
|
|
gradient.setColorAt(1.0, Qt::yellow);
|
2013-10-29 12:01:42 +00:00
|
|
|
m_surface->setSelectionMode(QDataVis::SelectionNone);
|
2013-09-27 08:24:42 +00:00
|
|
|
m_surface->setGridVisible(false);
|
2013-09-26 06:17:09 +00:00
|
|
|
m_surface->setGradient(gradient);
|
|
|
|
m_surface->axisY()->setMax(255);
|
|
|
|
m_surface->setSurfaceGridEnabled(false);
|
|
|
|
m_surface->setBackgroundVisible(false);
|
2013-12-02 10:59:46 +00:00
|
|
|
m_surface->setFlatShadingEnabled(true);
|
2013-09-26 06:17:09 +00:00
|
|
|
m_surface->setActiveDataProxy(new QHeightMapSurfaceDataProxy());
|
2013-10-01 10:17:27 +00:00
|
|
|
m_surface->setCameraPosition(0.0, 90.0, 105);
|
2013-09-26 06:17:09 +00:00
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
// Initialize scatter
|
|
|
|
m_scatter->setTheme(QDataVis::ThemeStoneMoss);
|
2013-10-29 12:01:42 +00:00
|
|
|
m_scatter->setSelectionMode(QDataVis::SelectionNone);
|
2013-09-27 08:24:42 +00:00
|
|
|
m_scatter->setGridVisible(false);
|
2013-11-28 06:19:37 +00:00
|
|
|
m_scatter->setObjectType(QDataVis::MeshDots, false);
|
2013-10-02 08:42:39 +00:00
|
|
|
m_scatter->setShadowQuality(QDataVis::ShadowQualitySoftLow);
|
2013-09-27 08:24:42 +00:00
|
|
|
m_scatter->setCameraPosition(0.0, 85.0, 110);
|
|
|
|
m_scatter->axisY()->setMin(-128);
|
|
|
|
m_scatter->axisY()->setMax(128);
|
|
|
|
m_scatter->axisX()->setMin(-m_resolution.width() / 2);
|
|
|
|
m_scatter->axisX()->setMax(m_resolution.width() / 2);
|
|
|
|
m_scatter->axisZ()->setMin(-m_resolution.height() / 2);
|
|
|
|
m_scatter->axisZ()->setMax(m_resolution.height() / 2);
|
|
|
|
|
|
|
|
// Initialize bars
|
|
|
|
m_bars->setTheme(QDataVis::ThemeQt);
|
2013-12-12 10:43:04 +00:00
|
|
|
m_bars->setSelectionMode(QDataVis::SelectionNone);
|
2013-09-27 08:24:42 +00:00
|
|
|
m_bars->setGridVisible(false);
|
2013-11-28 06:19:37 +00:00
|
|
|
m_bars->setBarType(QDataVis::MeshBars, true);
|
2013-09-27 08:24:42 +00:00
|
|
|
#if 1
|
2013-10-02 08:42:39 +00:00
|
|
|
m_bars->setShadowQuality(QDataVis::ShadowQualityLow);
|
2013-09-27 08:24:42 +00:00
|
|
|
#else
|
|
|
|
m_bars->setBarSpacing(QSizeF(0.0, 0.0));
|
|
|
|
#endif
|
|
|
|
m_bars->setCameraPosition(0.0, 75.0);
|
|
|
|
m_bars->valueAxis()->setMax(255);
|
2013-09-27 05:50:03 +00:00
|
|
|
|
2013-09-26 10:30:08 +00:00
|
|
|
// Hide scroll bar
|
|
|
|
m_statusArea->verticalScrollBar()->setVisible(false);
|
|
|
|
|
2013-09-26 06:17:09 +00:00
|
|
|
// Connect Kinect signals
|
|
|
|
connect(&m_kinect, &QKinect::QKinectWrapper::dataNotification, this, &SurfaceData::updateData);
|
|
|
|
connect(&m_kinect, &QKinect::QKinectWrapper::statusNotification, this,
|
|
|
|
&SurfaceData::updateStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
SurfaceData::~SurfaceData()
|
2013-09-30 07:59:50 +00:00
|
|
|
{ // HACK: The current context needs to be destroyed last
|
|
|
|
// TODO: Fix properly in datavis code somehow
|
|
|
|
if (m_mode == Surface) {
|
|
|
|
delete m_scatter;
|
|
|
|
delete m_bars;
|
|
|
|
delete m_surface;
|
2013-11-28 06:19:37 +00:00
|
|
|
} else if (m_mode == MeshBars) {
|
2013-09-30 07:59:50 +00:00
|
|
|
delete m_scatter;
|
|
|
|
delete m_surface;
|
|
|
|
delete m_bars;
|
|
|
|
} else {
|
|
|
|
delete m_bars;
|
|
|
|
delete m_surface;
|
|
|
|
delete m_scatter;
|
|
|
|
}
|
2013-09-26 06:17:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::updateData()
|
|
|
|
{
|
|
|
|
QImage depthMap = m_kinect.getDepth();
|
|
|
|
if (m_resize) // Resize for better performance
|
|
|
|
depthMap = depthMap.scaled(m_resolution);
|
2013-09-27 08:24:42 +00:00
|
|
|
if (m_mode != Surface)
|
|
|
|
setData(depthMap);
|
|
|
|
else
|
|
|
|
static_cast<QHeightMapSurfaceDataProxy *>(m_surface->activeDataProxy())->setHeightMap(
|
|
|
|
depthMap);
|
2013-09-26 06:17:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::updateStatus(QKinect::KinectStatus status)
|
|
|
|
{
|
|
|
|
switch (status) {
|
|
|
|
case QKinect::Idle: {
|
2013-09-26 10:30:08 +00:00
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Stopped"));
|
|
|
|
m_statusArea->append(QStringLiteral("<br><b>Ready</b><br>"));
|
2013-09-26 06:17:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case QKinect::Initializing: {
|
2013-09-26 10:30:08 +00:00
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Initializing"));
|
2013-09-26 06:17:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case QKinect::OkRun: {
|
2013-09-26 10:30:08 +00:00
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Running"));
|
|
|
|
m_statusArea->append(QString(QStringLiteral("<i> - resolution: %1 x %2</i>")).arg(
|
|
|
|
m_resolution.width()).arg(m_resolution.height()));
|
2013-09-26 06:17:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
2013-09-26 10:30:08 +00:00
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Error"));
|
2013-09-26 06:17:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::start()
|
|
|
|
{
|
2013-09-26 10:30:08 +00:00
|
|
|
if (m_kinect.isStopped())
|
2013-09-26 06:17:09 +00:00
|
|
|
m_kinect.start();
|
2013-09-26 10:30:08 +00:00
|
|
|
else
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Already running"));
|
2013-09-26 06:17:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::stop()
|
|
|
|
{
|
2013-09-26 10:30:08 +00:00
|
|
|
if (m_kinect.isRunning())
|
2013-09-26 06:17:09 +00:00
|
|
|
m_kinect.stop();
|
2013-09-26 10:30:08 +00:00
|
|
|
else
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Kinect:</b> Already stopped"));
|
2013-09-26 06:17:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::setDistance(int distance)
|
|
|
|
{
|
|
|
|
m_kinect.setMaxDist(unsigned int(distance * 100));
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::setResolution(int selection)
|
|
|
|
{
|
2013-09-27 08:24:42 +00:00
|
|
|
m_resolutionLevel = selection;
|
2013-09-26 06:17:09 +00:00
|
|
|
switch (selection) {
|
|
|
|
case 0: {
|
|
|
|
m_resize = true;
|
|
|
|
m_resolution = QSize(320, 240);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1: {
|
|
|
|
m_resize = true;
|
|
|
|
m_resolution = QSize(384, 288);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: {
|
|
|
|
m_resize = true;
|
|
|
|
m_resolution = QSize(512, 384);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3: {
|
|
|
|
m_resize = false;
|
|
|
|
m_resolution = QSize(640, 480);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
2013-09-27 08:24:42 +00:00
|
|
|
if (m_mode == Scatter) {
|
|
|
|
m_resize = true;
|
|
|
|
m_resolution /= 4;
|
|
|
|
m_scatter->axisX()->setMin(-m_resolution.width() / 2);
|
|
|
|
m_scatter->axisX()->setMax(m_resolution.width() / 2);
|
|
|
|
m_scatter->axisZ()->setMin(-m_resolution.height() / 2);
|
|
|
|
m_scatter->axisZ()->setMax(m_resolution.height() / 2);
|
2013-09-27 10:10:31 +00:00
|
|
|
m_scatterDataArray = new QScatterDataArray;
|
|
|
|
m_scatterDataArray->resize(m_resolution.width() * m_resolution.height());
|
2013-11-28 06:19:37 +00:00
|
|
|
} else if (m_mode == MeshBars) {
|
2013-09-27 08:24:42 +00:00
|
|
|
m_resize = true;
|
|
|
|
m_resolution /= 4;
|
2013-09-27 10:10:31 +00:00
|
|
|
m_barDataArray = new QBarDataArray;
|
|
|
|
m_barDataArray->reserve(m_resolution.height());
|
|
|
|
for (int i = 0; i < m_resolution.height(); i++) {
|
|
|
|
QBarDataRow *newProxyRow = new QBarDataRow(m_resolution.width());
|
|
|
|
m_barDataArray->append(newProxyRow);
|
|
|
|
}
|
2013-09-27 08:24:42 +00:00
|
|
|
}
|
2013-09-27 04:43:40 +00:00
|
|
|
|
2013-09-26 10:30:08 +00:00
|
|
|
m_statusArea->append(QString(QStringLiteral("<b>Resolution:</b> %1 x %2")).arg(
|
|
|
|
m_resolution.width()).arg(m_resolution.height()));
|
|
|
|
if (m_kinect.isStopped())
|
|
|
|
m_statusArea->append(QStringLiteral("<i> - change takes effect once Kinect is running</i>"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::scrollDown()
|
|
|
|
{
|
|
|
|
QScrollBar *scrollbar = m_statusArea->verticalScrollBar();
|
|
|
|
scrollbar->setValue(scrollbar->maximum());
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::useGradientOne()
|
|
|
|
{
|
|
|
|
m_surface->setTheme(QDataVis::ThemeIsabelle);
|
|
|
|
QLinearGradient gradient;
|
|
|
|
gradient.setColorAt(0.0, Qt::black);
|
|
|
|
gradient.setColorAt(0.33, Qt::blue);
|
|
|
|
gradient.setColorAt(0.67, Qt::red);
|
|
|
|
gradient.setColorAt(1.0, Qt::yellow);
|
|
|
|
m_surface->setGradient(gradient);
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Colors:</b> Thermal image imitation"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceData::useGradientTwo()
|
|
|
|
{
|
|
|
|
m_surface->setTheme(QDataVis::ThemeQt);
|
|
|
|
QLinearGradient gradient;
|
|
|
|
gradient.setColorAt(0.0, Qt::white);
|
|
|
|
gradient.setColorAt(0.8, Qt::red);
|
|
|
|
gradient.setColorAt(1.0, Qt::green);
|
|
|
|
m_surface->setGradient(gradient);
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Colors:</b> Highlight foreground"));
|
2013-09-26 06:17:09 +00:00
|
|
|
}
|
2013-09-27 08:24:42 +00:00
|
|
|
|
2013-09-27 04:43:40 +00:00
|
|
|
void SurfaceData::setData(const QImage &image)
|
|
|
|
{
|
|
|
|
QImage heightImage = image;
|
|
|
|
|
|
|
|
uchar *bits = heightImage.bits();
|
|
|
|
|
|
|
|
int imageHeight = heightImage.height();
|
|
|
|
int imageWidth = heightImage.width();
|
|
|
|
int bitCount = imageWidth * 4 * (imageHeight - 1);
|
|
|
|
int widthBits = imageWidth * 4;
|
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
if (m_mode == Scatter) {
|
2013-10-01 07:14:50 +00:00
|
|
|
QScatterDataItem *ptrToDataArray = &m_scatterDataArray->first();
|
2013-09-27 04:43:40 +00:00
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
int limitsX = imageWidth / 2;
|
|
|
|
int limitsZ = imageHeight / 2;
|
|
|
|
qreal height = 0;
|
2013-10-01 07:14:50 +00:00
|
|
|
int count = 0;
|
2013-09-27 04:43:40 +00:00
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
for (int i = -limitsZ; i < limitsZ; i++, bitCount -= widthBits) {
|
|
|
|
for (int j = -limitsX; j < limitsX; j++) {
|
|
|
|
height = qreal(bits[bitCount + ((j + limitsX) * 4)]) - 128.0;
|
|
|
|
if (height > -128) {
|
|
|
|
ptrToDataArray->setPosition(QVector3D(qreal(j), height, qreal(i)));
|
|
|
|
ptrToDataArray++;
|
2013-10-01 07:14:50 +00:00
|
|
|
count++;
|
2013-09-27 08:24:42 +00:00
|
|
|
}
|
2013-09-27 04:43:40 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-27 08:24:42 +00:00
|
|
|
|
2013-10-01 07:14:50 +00:00
|
|
|
QScatterDataArray *dataArray = new QScatterDataArray(m_scatterDataArray->mid(0, count));
|
2013-09-27 08:24:42 +00:00
|
|
|
static_cast<QScatterDataProxy *>(m_scatter->activeDataProxy())->resetArray(dataArray);
|
|
|
|
} else {
|
2013-09-27 10:10:31 +00:00
|
|
|
QBarDataArray *dataArray = m_barDataArray;
|
|
|
|
for (int i = 0; i < imageHeight; i++, bitCount -= widthBits) {
|
|
|
|
QBarDataRow &newRow = *dataArray->at(i);
|
2013-09-27 08:24:42 +00:00
|
|
|
for (int j = 0; j < imageWidth; j++)
|
2013-09-27 10:10:31 +00:00
|
|
|
newRow[j] = qreal(bits[bitCount + (j * 4)]);
|
2013-09-27 08:24:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static_cast<QBarDataProxy *>(m_bars->activeDataProxy())->resetArray(dataArray);
|
2013-09-27 04:43:40 +00:00
|
|
|
}
|
2013-09-27 08:24:42 +00:00
|
|
|
}
|
2013-09-27 04:43:40 +00:00
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
void SurfaceData::changeMode(int mode)
|
|
|
|
{
|
|
|
|
m_mode = VisualizationMode(mode);
|
|
|
|
switch (m_mode) {
|
|
|
|
case Surface: {
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Visualization Type:</b> Surface"));
|
|
|
|
break;
|
2013-09-27 05:50:03 +00:00
|
|
|
}
|
2013-09-27 08:24:42 +00:00
|
|
|
case Scatter: {
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Visualization Type:</b> Scatter"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
m_statusArea->append(QStringLiteral("<b>Visualization Type:</b> Bars"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Reset resolution after mode change
|
|
|
|
setResolution(m_resolutionLevel);
|
|
|
|
}
|
2013-09-27 05:50:03 +00:00
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
ContainerChanger::ContainerChanger(QWidget *surface, QWidget *scatter, QWidget *bars,
|
|
|
|
QWidget *buttonOne, QWidget *buttonTwo)
|
|
|
|
: m_surface(surface),
|
|
|
|
m_scatter(scatter),
|
|
|
|
m_bars(bars),
|
|
|
|
m_button1(buttonOne),
|
|
|
|
m_button2(buttonTwo)
|
|
|
|
{
|
2013-09-27 04:43:40 +00:00
|
|
|
}
|
|
|
|
|
2013-09-27 08:24:42 +00:00
|
|
|
ContainerChanger::~ContainerChanger()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ContainerChanger::changeContainer(int container)
|
|
|
|
{
|
|
|
|
switch (container) {
|
|
|
|
case 0: {
|
|
|
|
m_scatter->setVisible(false);
|
|
|
|
m_bars->setVisible(false);
|
|
|
|
m_surface->setVisible(true);
|
|
|
|
m_button1->setEnabled(true);
|
|
|
|
m_button2->setEnabled(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1: {
|
|
|
|
m_surface->setVisible(false);
|
|
|
|
m_bars->setVisible(false);
|
|
|
|
m_scatter->setVisible(true);
|
|
|
|
m_button1->setEnabled(false);
|
|
|
|
m_button2->setEnabled(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: {
|
|
|
|
m_surface->setVisible(false);
|
|
|
|
m_scatter->setVisible(false);
|
|
|
|
m_bars->setVisible(true);
|
|
|
|
m_button1->setEnabled(false);
|
|
|
|
m_button2->setEnabled(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|