© 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". Подскажите, как решить проблему.

Добрый день!

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

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

Для Django рекомендую VDS-хостинг TIMEWEB

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

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! Евгений, огромное спасибо!

Ответы

Только авторизованные пользователи могут отвечать на форуме.
Пожалуйста, Авторизуйтесь или Зарегистрируйтесь
24 сентября 2018 г. 17:42
edorofeeva

C++ - Тест 001. Первая программа и типы данных

  • Результат 100баллов,
  • Очки рейтинга10
24 сентября 2018 г. 17:37
edorofeeva

C++ - Тест 001. Первая программа и типы данных

  • Результат 66баллов,
  • Очки рейтинга-1
23 сентября 2018 г. 14:38
No Names

C++ - Тест 001. Первая программа и типы данных

  • Результат 60баллов,
  • Очки рейтинга-1
Последние комментарии
25 сентября 2018 г. 15:24
pasagir

Qt/C++ - Урок 006. QSqlQueryModel - Таблицы в Qt с помощью SQL-запросов

Это запись метода которая работает параллельно с БД, данные из парсера поступают в БД и в наш метод одновременно
25 сентября 2018 г. 14:56
pasagir

Qt/C++ - Урок 006. QSqlQueryModel - Таблицы в Qt с помощью SQL-запросов

Как можно динамически отображать данные в таблице? На COM-порт непрерывно приходят данные, я их принимаю сохраняю в БД, а после остановка приема/передачи данные отображаются в таблице. В табли...
25 сентября 2018 г. 10:43
Евгений Легоцкой

Qt/C++ - Урок 017. QGraphicsScene или как работать с графикой в Qt

Прямо так не написано. Хотя соглашусь, что в качестве улучшения вызов данного метода здесь к месту.
25 сентября 2018 г. 10:37
reef425

Qt/C++ - Урок 017. QGraphicsScene или как работать с графикой в Qt

В статье написано, что таймер сработает один раз. Но это не так. Было бы хорошо добавить timer->setSingleShot(true); После инициализации таймера.
24 сентября 2018 г. 15:09
Евгений Легоцкой

Qt Linux - Урок 001. Автозапуск Qt приложения под Linux

А вот здесь у меня есть пример использования supervisor. https://evileg.com/ru/post/3/ Вся статья вам там не интересна, интересен только шаг с настройкой supervisor. Он получается ...
Сейчас обсуждают на форуме
25 сентября 2018 г. 15:57
Евгений_Канусовский@1981

Чтение файлов в python

Вот код: import sys from re import matchfrom vira import *from PyQt5 import QtCore, QtGui, QtWidgetsfrom PyQt5.QtWidgets import (QDialog, QFileDialog, QMessageBox, QLineEdit, QProgr...
25 сентября 2018 г. 13:51
DmitrySD

Трансляция видео с помощью VLC по RTP

Спасибо! Данная команда не дала результата. В итоге сделал трансляцию через ffmpeg. ffmpeg.exe -f gdigrab -framerate 30 -i desktop -vcodec libx264 -preset:v veryfast -b:v 4000k -f...
25 сентября 2018 г. 13:39
Arrow

Настройка Qt Creator для Android

Конечно отпишусь.
25 сентября 2018 г. 12:22
avovana

Автозапуск и авторестарт Qt Gui Application в Linux

Не получается... Решил пробовать скрипт, выполняемый при загрузке. В скрипте вечный цикл по старту программы.
Присоединяйтесь к нам в социальных сетях