© EVILEG 2015-2018
Рекомендует хостинг
TIMEWEB
22 мая 2018 г. 16:50

Данные из QChartview в QTableWidget

Здравствуйте! Пишу приложение для парсинга текстового файла и вывода данных на график. Столкнулся с проблемой передачи данных от курсора мыши на графике в ячейку таблицы.

mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include <QtCharts/QLineSeries>
#include <QtCharts/QVXYModelMapper>
#include <QtWidgets/QGridLayout>
#include <QtWidgets/QTableView>
#include <QtWidgets/QTableWidget>
#include <QtWidgets/QHeaderView>
#include <QSplitter>
#include <QTextEdit>
#include <QStandardItemModel>
#include <QTextStream>
#include <QFile>
#include <QDebug>
#include <QDateTimeAxis>
#include <QValueAxis>
#include <QDateTime>
#include <QAbstractItemModel>
#include <QLabel>
#include <QMouseEvent>
#include <chart.h>
#include <chartview.h>
QT_CHARTS_USE_NAMESPACE
class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    MainWindow(QWidget *parent = 0);
    ~MainWindow();
public:
    QTableWidget *tabwidget = new QTableWidget;
    QStandardItemModel *model = new QStandardItemModel;
    Chart *chart = new Chart;
    ChartView *chartview = new ChartView(chart);
    bool m_pressedItemState;
    QVector<QDateTime> list;

public slots:
    /*Для улавливания изменения состояния чекбокса использую
    *следующие два метода*/
    void tableItemPressed(QTableWidgetItem * item)
    {
        m_pressedItemState = item->checkState();
    }

    void tableItemChanged(QTableWidgetItem * item)
    {
        if(item->checkState() == Qt::Checked){
            int id = tabwidget->row(item);
            onAddSeries(id+1);
        } else {
            int id = tabwidget->row(item);
            onDelSeries(id+1);
        }
    }
    /*слот добавления серии на график. Номеру строки таблицы i
     *соответствует номер колонки в модели*/
    void onAddSeries(int i);
    /*слот удаления серии из графика. Номеру строки таблицы i
     *соответствует ObjectName серии*/
    void onDelSeries(int i);
    void onAddTableItem();
};
#endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h"
MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    /*инициализируем виджеты, парсим файл (заголовки отправляем в таблицу, данные в модель),
     * создаем сигнально-слотовые соединения для создания/удаления серий графика*/
    chartview->resize(800, 400);
    tabwidget->resize(800, 100);
    QSplitter *splitter = new QSplitter(Qt::Vertical);
    splitter->addWidget(chartview);
    splitter->addWidget(tabwidget);
    this->setCentralWidget(splitter);
    QFile file("test.txt");
        if (file.open(QIODevice::ReadOnly)) {
            int lineindex = 0;
            QTextStream in(&file);
            QString fileLine = in.readLine();
            QStringList lineToken = fileLine.split("\t", QString::SkipEmptyParts);
            tabwidget->insertColumn(0);
            tabwidget->insertColumn(1);
            QStringList s;
            s.append("KKS датчика");
            s.append("Значение");
            tabwidget->setHorizontalHeaderLabels(s);
            for (int j = 0; j < lineToken.size(); j++) {
                tabwidget->insertRow(j);
                QString value = lineToken.at(j);
                QTableWidgetItem *item = new QTableWidgetItem();
                item->data(Qt::CheckStateRole);
                item->setCheckState(Qt::Unchecked);
                item->setText(value);
                tabwidget->setItem(j, lineindex, item);
            }
            while (!in.atEnd()) {
                        QString fileLine = in.readLine();
                        QStringList lineToken = fileLine.split("\t", QString::SkipEmptyParts);
                        for (int j = 0; j < lineToken.size(); j++) {
                            if(j==0){
                                QString dd = lineToken.at(j);
                                QDateTime dt = QDateTime::fromString(dd, "dd-MM-yyyy hh:mm:ss");
                                QStandardItem *item = new QStandardItem(QString::number(dt.toMSecsSinceEpoch()));
                                model->setItem(lineindex, j, item);
                            } else {
                            QString value = lineToken.at(j);
                            QStandardItem *item = new QStandardItem(value);
                            model->setItem(lineindex, j, item);
                            }
                        }
                        lineindex++;
                    }
                    file.close();
        }
            tabwidget->removeRow(0);
            tabwidget->resizeColumnsToContents();
            connect(tabwidget,SIGNAL(itemPressed(QTableWidgetItem *)),this,SLOT(tableItemPressed(QTableWidgetItem *)));
            connect(tabwidget,SIGNAL(itemChanged(QTableWidgetItem *)),this,SLOT(tableItemChanged(QTableWidgetItem *)));
            chartview->setRenderHint(QPainter::Antialiasing);
            chart->setAnimationOptions(QChart::AllAnimations);
}

void MainWindow::onAddSeries(int i){
    /*создаем серию для выбранного датчика i*/
    QLineSeries *series = new QLineSeries(this);
    series->setName("Линия " + QString::number(i));
    series->setObjectName("obj" + QString::number(i));
    QVXYModelMapper *mapper = new QVXYModelMapper(this);
    mapper->setXColumn(0);
    mapper->setYColumn(i);
    mapper->setSeries(series);
    mapper->setModel(model);
    chartview->chart()->addSeries(series);
    if(!chartview->chart()->axisX()){ //если оси времени еще нет, то создаем
    QDateTimeAxis *axisX = new QDateTimeAxis;
    axisX->setObjectName("axiX" + QString::number(i));
    axisX->setTickCount(10);
    axisX->setFormat("dd-MM-yyyy hh:mm:ss");
    chartview->chart()->addAxis(axisX, Qt::AlignBottom);
    series->attachAxis(axisX);
}    
    QValueAxis *axisY = new QValueAxis; //для каждой серии свою ось Y
    axisY->setObjectName("axiY" + QString::number(i));
    axisY->setLabelFormat("%i");
    chartview->chart()->addAxis(axisY, Qt::AlignLeft);
    series->attachAxis(axisY);
    onAddTableItem();
}

void MainWindow::onDelSeries(int i){
    /*при снятии галочки с чекбокса датчика удаляем соответствующую ему серию*/
    chartview->chart()->findChild<QLineSeries *>("obj" + QString::number(i))->deleteLater();
    chartview->chart()->findChild<QValueAxis *>("axiY" + QString::number(i))->deleteLater();
}

void MainWindow::onAddTableItem(){
    QTableWidgetItem *item2 = new QTableWidgetItem;
    item2->setText(chartview->coordX_str); /*как обновлять это значение, которое приходит из chartview->mouseMoveEvent()?*/
    tabwidget->setItem(0, 1, item2);
}

MainWindow::~MainWindow()
{

}
метод получения координат по курсору мыши в классе ChartView:
void ChartView::mouseMoveEvent(QMouseEvent *event)
{
    if (m_isTouching)
        return;
    coordY = chart()->mapToValue(event->pos()).y();
    coordX = chart()->mapToValue(event->pos()).x();
    QDateTime dt = QDateTime::fromMSecsSinceEpoch(chart()->mapToValue(event->pos()).x());
    coordX_str = dt.toString("dd-MM-yyyy hh:mm:ss");
    qDebug()<<coordY;
    qDebug()<<coordX_str;
    QChartView::mouseMoveEvent(event);
}
При вызове слота void MainWindow::onAddTableItem() ячейка таблицы заполняется однажды и значения не обновляются. Как можно добиться обновления данных? По логике напрашивается обновлять значение ячейки прямо из метода void ChartView::mouseMoveEvent(QMouseEvent *event), но это другой класс, который не знает про эту ячейку. Пробовал в chartview.cpp инклудить mainwindow.h чтобы иметь доступ к виджетам, но спотыкаюсь об "invalid use of non-static data member". Подскажите, как решить проблему.
  • #
  • Ответ был помечен как решение
  • 23 мая 2018 г. 6:25

Добрый день!

Добавьте сигнал в ChartView, который будет передавать необходимые для обновления ячейки данные, и подключите данный сигнал к слоту, который будет заниматься обновлением данных.

Например в объявлении класса ChartView
signals:
    void someSignal(int arg1);
А в классе главного окна приложения MainWindow объявить слот, с реализацией
private slots:
    void onSomeSlot(int arg1);
В конструкторе окна подключить этот сигнал от chartview к слоту
connect(cartview, &ChartView::someSignal, this, &MainWindow::onSomeSlot);
Сигнал вызывать в методе mouseMoveEvent. В Слоте уже делать необходимую обработку.

Спасибо за подсказку, но сигнал-слот не работает. При попытке вызвать сигнал

void ChartView::mouseMoveEvent(QMouseEvent *event)
{
...
    emit someSignal(coordX_str);
...
}
получаю ошибку: undefined reference to `ChartView::someSignal(QString)'
Как я понимаю вся проблема в классе ChartView:
#ifndef CHARTVIEW_H
#define CHARTVIEW_H
#include <QtCharts/QChartView>
#include <QtWidgets/QRubberBand>
#include <QDebug>
#include <QDateTime>
QT_CHARTS_USE_NAMESPACE
class ChartView : public QChartView
{
public:
    ChartView(QChart *chart, QWidget *parent = 0);
    QString coordX_str;
    qreal coordY;
    bool m_isTouching;
    qreal coordX;
signals:
    void someSignal(QString arg1);
public slots:
    bool viewportEvent(QEvent *event);
    void mousePressEvent(QMouseEvent *event);
    void mouseMoveEvent(QMouseEvent *event);
    void mouseReleaseEvent(QMouseEvent *event);
    void keyPressEvent(QKeyEvent *event);
    void test();
};
#endif // CHARTVIEW_H
Для сигнально-слотовых соединений необходим макрос Q_OBJECT. При попытке его использования получаю ошибку: undefined reference to `vtable for ChartView'.

Проблема решена, надо было просто перезапустить qmake! Евгений, огромное спасибо!

Ответы

Только авторизованные пользователи могут отвечать на форуме.
Пожалуйста, Авторизуйтесь или Зарегистрируйтесь
25 июня 2018 г. 11:55
lebendig

C++ - Тест 006. Перечисления

  • Результат 100 баллов
  • Очки рейтинга 10
25 июня 2018 г. 11:24
lebendig

C++ - Тест 005. Структуры и Классы

  • Результат 100 баллов
  • Очки рейтинга 10
25 июня 2018 г. 8:48
lebendig

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат 100 баллов
  • Очки рейтинга 10
Последние комментарии
25 июня 2018 г. 9:34
EVILEG

Как написать игру на Qt - Урок 4. Враг - смысл в выживании

Там неправильный подсчёт был по направлению и полный бардак был с поведением, эти 45 градусов исправляли ситуацию. Точную причину уже совсем не помню. А вообще все эти расчёты довольно...
25 июня 2018 г. 9:13
MarkSD

Как написать игру на Qt - Урок 4. Враг - смысл в выживании

Здравствуйте, Подскажите, пжлст, как работает этот код : QLineF lineToTarget(QPointF(0, 0), mapFromItem(target, 0, 0));  // Проводим линию от паука к мухе qreal angl...
25 июня 2018 г. 7:51
EVILEG

PyQt5 - Урок 003. QSystemTrayIcon - Как свернуть приложение в трей

Если не ошибаюсь, можно просто удалить вот эту строку central_widget.setLayout(grid_layout) Там указатель на парента передаётся в само размещение, что автоматически заменяет в размещен...
Сейчас обсуждают на форуме
25 июня 2018 г. 17:38
IscanderChe

Иконка исполняемого файла

Спасибо!
25 июня 2018 г. 13:12
Arrow

QComboBox и База данных

И если можно еще один вопрос. Таблицы во вложении. Если писать: mainModel = new QSqlRelationalTableModel(this);mainModel-&g...;
25 июня 2018 г. 7:49
EVILEG

На чём сделан этот сайт?

Добрый день! На сервере сайта установлена Ubuntu 16.04. В качестве сервера используется VDS. Хостинг-провайдер Timeweb . Сайт написан на Django/Python, для...
19 июня 2018 г. 7:56
EVILEG

как редактировать порядок обхода этементов по нажатию TAB в Qt5 qml

Что-то наподобие такого TextField { Keys.onReturnPressed: nextItemInFocusChain().forceActiveFocus()}

Рекомендуемые страницы