ChartAndClassModif

main
dplimin 2024-02-05 10:21:28 +03:00
parent c118b28235
commit 0e11c73395
10 changed files with 414 additions and 190 deletions

View File

@ -14,11 +14,15 @@ CONFIG += c++17
SOURCES += \
bluetooth1d801.cpp \
chart.cpp \
chartview.cpp \
main.cpp \
mainwindow.cpp
HEADERS += \
bluetooth1d801.h \
chart.h \
chartview.h \
mainwindow.h
FORMS += \

View File

@ -8,13 +8,13 @@ Bluetooth1d801::Bluetooth1d801(QObject *parent):QObject(parent) {
if (localDevice.isValid()) { //при создании экземпляра класса он подключается к свистку
localDevice.powerOn();
localDeviceName = localDevice.name();
qDebug() <<localDeviceName; // пишет "dongle"
qDebug()<< "Итак, имя нашего бт адаптера " <<localDeviceName; // пишет "dongle"
localDevice.setHostMode(QBluetoothLocalDevice::HostDiscoverable);
QList<QBluetoothAddress> remotes;
remotes = localDevice.connectedDevices(); // если например наушники подключены и готовы выплевывает их адрес
// с другими, например сопряженными не работает
qDebug() <<"Подключенные устройства:"<<remotes;
qDebug() <<"если среди них есть 1D801, то что то пошло иначе, чем у разработчика. Попробуйте продолжить работы"<<remotes;
qDebug() <<"если среди них есть 1D801, то что то пошло иначе, чем у разработчика. Попробуйте продолжить работы";
}
@ -35,7 +35,7 @@ void Bluetooth1d801::startDiscovery()
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::finished,
this, &Bluetooth1d801::discoveryFinished);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::canceled,
this, &Bluetooth1d801::discoveryFinished); //странный коннект - слизал с бтчата (всмысле с указателями. Наверное это норм для динамических штук)
this, &Bluetooth1d801::discoveryFinished); //коннект от статика слизал с чата
// начинаем поиск устройств
discoveryAgent->start();
@ -56,14 +56,38 @@ void Bluetooth1d801::stopDiscovery()
}
void Bluetooth1d801::connectTo(const QBluetoothServiceInfo &service)
void Bluetooth1d801::connectTo(QListWidgetItem *item)
{
// туду проверить, может уже подключен??
QBluetoothServiceInfo service = discoveredServices.value(item);
QBluetoothServiceInfo serviceToConnect = service;
qDebug() << "Connecting to service" << service.serviceName() //валим инфу в дебаг
<< "on" << service.device().name();
socket = new QBluetoothSocket(QBluetoothServiceInfo::RfcommProtocol); //создание сокета
qDebug() << "Create socket";
socket->connectToService(service);
connect(socket, &QBluetoothSocket::readyRead, this, &Bluetooth1d801::readSocket);
connect(socket, &QBluetoothSocket::connected, this, QOverload<>::of(&Bluetooth1d801::connected));
connect(socket, &QBluetoothSocket::disconnected, this, &Bluetooth1d801::disconnected);
connect(socket, &QBluetoothSocket::errorOccurred, this, &Bluetooth1d801::onSocketErrorOccurred);
qDebug() << "Мммм.. подключился... ))";
}
void Bluetooth1d801::GetParam()
{
if (!socket){
qDebug()<< "Ты не создал подходящий сокет!!! Я не буду работать!!!";
return;
}
QString message = "Get_Param";
QByteArray text = message.toUtf8() + '\r' + '\n';
socket->write(text);
// взвести коннект
}
@ -109,20 +133,67 @@ void Bluetooth1d801::GetWaveEnv()
void Bluetooth1d801::serviceDiscovered(const QBluetoothServiceInfo &serviceInfo)
{
qDebug() << "Discovered service on"
<< serviceInfo.device().name() << serviceInfo.device().address().toString();
qDebug() << "\tService name:" << serviceInfo.serviceName();
qDebug() << "\tDescription:"
<< serviceInfo.attribute(QBluetoothServiceInfo::ServiceDescription).toString();
qDebug() << "\tProvider:"
<< serviceInfo.attribute(QBluetoothServiceInfo::ServiceProvider).toString();
qDebug() << "\tL2CAP protocol service multiplexer:"
<< serviceInfo.protocolServiceMultiplexer();
qDebug() << "\tRFCOMM server channel:" << serviceInfo.serverChannel();
const QBluetoothAddress address = serviceInfo.device().address();
/* // если находятся много сервисов в одной железке, то названия не дублируются
for (const QBluetoothServiceInfo &info : std::as_const(m_discoveredServices)) {
if (info.device().address() == address)
return; //
}
*/
//! [serviceDiscovered]
QString remoteName;
if (serviceInfo.device().name().isEmpty())
remoteName = address.toString();
else
remoteName = serviceInfo.device().name();
QListWidgetItem *item_p =new QListWidgetItem(QString::fromLatin1("%1 %2").arg(remoteName, serviceInfo.serviceName()));
discoveredServices.insert(item_p, serviceInfo);
emit discoveryResult(item_p);
//qDebug() << "вот это работает в процедуре addItem" << item_p; // хах, так я определял что должно прелететь в слот родительского класса, ахах
//ui->listWidgetDevice->addItem(item);
}
void Bluetooth1d801::readSocket()
{
// return; // удалить 7.02.2024 - до этого подумать откуда он взялся
while (socket->canReadLine()) {
QByteArray line = socket->readLine().trimmed();
qDebug() << line;
emit rawDataRecived(line);
}
}
void Bluetooth1d801::connected()
{
emit connected(socket->peerName());
}
void Bluetooth1d801::onSocketErrorOccurred(QBluetoothSocket::SocketError error)
{
if (error == QBluetoothSocket::SocketError::NoSocketError)
return;
QMetaEnum metaEnum = QMetaEnum::fromType<QBluetoothSocket::SocketError>();
QString errorString = socket->peerName()// + ' '_L1
+ metaEnum.valueToKey(static_cast<int>(error)) + " occurred"; //_L1;
emit socketErrorOccurred(errorString);
}
void Bluetooth1d801::discoveryFinished()
{
//emit discoveryFinishResult(discoveredServices);
}

View File

@ -12,9 +12,16 @@
#include <QBluetoothLocalDevice>
#include <QtCore/qmetaobject.h>
#include <QLatin1StringView>
///прописать статус
/// перечисление статусов
/// сигнал статусИзменен
class Bluetooth1d801 : public QObject
{
@ -27,12 +34,11 @@ public:
public slots:
void startDiscovery(); //запускаем bluetooth поиск
void stopDiscovery(); //прерываем его
void connectTo(QBluetoothServiceInfo); // подключится к сервису
void connectTo(QListWidgetItem *item); // подключится к сервису
//команды для модуля в первородном виде
void GetParam(); //получить параметры измерения
void SetParam(); // запись параметров измерения
void GetMeas(); //Чтение измеренных значений
void GetWave(); //Чтение временного сигнала виброускорения /виброскорости /виброперемещения
void GetSpectrum();//Чтение спектра виброускорения /виброскорости /виброперемещения
@ -41,17 +47,23 @@ public slots:
void GetSpectrumEnv(); //Чтение спектра огибающей виброускорения
void GetWaveEnv();//Чтение сигнала огибающей виброускорения
private slots:
void serviceDiscovered(const QBluetoothServiceInfo &serviceInfo); //отрабатываю сигнал обнаруженного сервиса
void readSocket(); //отработка сигнала прибывших данных
void connected(); // отработка подключения
void onSocketErrorOccurred(QBluetoothSocket::SocketError error);//отработка ошибки
void discoveryFinished();
signals:
void discoveryFinished(QMap<QListWidgetItem *, QBluetoothServiceInfo>); //возвращаем обнаруженные сервисы
void discoveryResult(QListWidgetItem *res_p); //возвращаем обнаруженные сервисы
//void MainWindow::addDiscoveryResult(QListWidgetItem *res_p)
//ui->listWidgetDevice->addItem(res_p);
void rawDataRecived(QByteArray); // прилетели какие то данные!
void disconnected(); // отключилось устройство
void socketErrorOccurred(const QString &errorString);
void connected(const QString &name); // отработка подключения
private:
QBluetoothSocket *socket = nullptr; // объект сокета для порта

42
chart.cpp 100644
View File

@ -0,0 +1,42 @@
// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#include "chart.h"
#include <QGesture>
#include <QGraphicsScene>
#include <QGraphicsView>
Chart::Chart(QGraphicsItem *parent, Qt::WindowFlags wFlags)
: QChart(QChart::ChartTypeCartesian, parent, wFlags)
{
// Seems that QGraphicsView (QChartView) does not grab gestures.
// They can only be grabbed here in the QGraphicsWidget (QChart).
grabGesture(Qt::PanGesture);
grabGesture(Qt::PinchGesture);
}
//![1]
bool Chart::sceneEvent(QEvent *event)
{
if (event->type() == QEvent::Gesture)
return gestureEvent(static_cast<QGestureEvent *>(event));
return QChart::event(event);
}
bool Chart::gestureEvent(QGestureEvent *event)
{
if (QGesture *gesture = event->gesture(Qt::PanGesture)) {
auto pan = static_cast<QPanGesture *>(gesture);
QChart::scroll(-(pan->delta().x()), pan->delta().y());
}
if (QGesture *gesture = event->gesture(Qt::PinchGesture)) {
auto pinch = static_cast<QPinchGesture *>(gesture);
if (pinch->changeFlags() & QPinchGesture::ScaleFactorChanged)
QChart::zoom(pinch->scaleFactor());
}
return true;
}
//![1]

25
chart.h 100644
View File

@ -0,0 +1,25 @@
// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#ifndef CHART_H
#define CHART_H
#include <QChart>
QT_FORWARD_DECLARE_CLASS(QGestureEvent)
//![1]
class Chart : public QChart
//![1]
{
public:
explicit Chart(QGraphicsItem *parent = nullptr, Qt::WindowFlags wFlags = {});
protected:
bool sceneEvent(QEvent *event);
private:
bool gestureEvent(QGestureEvent *event);
};
#endif

83
chartview.cpp 100644
View File

@ -0,0 +1,83 @@
// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#include "chartview.h"
#include <QMouseEvent>
ChartView::ChartView(QChart *chart, QWidget *parent)
: QChartView(chart, parent)
{
setRubberBand(QChartView::RectangleRubberBand);
}
bool ChartView::viewportEvent(QEvent *event)
{
if (event->type() == QEvent::TouchBegin) {
// By default touch events are converted to mouse events. So
// after this event we will get a mouse event also but we want
// to handle touch events as gestures only. So we need this safeguard
// to block mouse events that are actually generated from touch.
m_isTouching = true;
// Turn off animations when handling gestures they
// will only slow us down.
chart()->setAnimationOptions(QChart::NoAnimation);
}
return QChartView::viewportEvent(event);
}
void ChartView::mousePressEvent(QMouseEvent *event)
{
if (m_isTouching)
return;
QChartView::mousePressEvent(event);
}
void ChartView::mouseMoveEvent(QMouseEvent *event)
{
if (m_isTouching)
return;
QChartView::mouseMoveEvent(event);
}
void ChartView::mouseReleaseEvent(QMouseEvent *event)
{
if (m_isTouching)
m_isTouching = false;
// Because we disabled animations when touch event was detected
// we must put them back on.
chart()->setAnimationOptions(QChart::SeriesAnimations);
QChartView::mouseReleaseEvent(event);
}
//![1]
void ChartView::keyPressEvent(QKeyEvent *event)
{
switch (event->key()) {
case Qt::Key_Plus:
chart()->zoomIn();
break;
case Qt::Key_Minus:
chart()->zoomOut();
break;
//![1]
case Qt::Key_Left:
chart()->scroll(-10, 0);
break;
case Qt::Key_Right:
chart()->scroll(10, 0);
break;
case Qt::Key_Up:
chart()->scroll(0, 10);
break;
case Qt::Key_Down:
chart()->scroll(0, -10);
break;
default:
QGraphicsView::keyPressEvent(event);
break;
}
}

30
chartview.h 100644
View File

@ -0,0 +1,30 @@
// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#ifndef CHARTVIEW_H
#define CHARTVIEW_H
#include <QChartView>
#include <QRubberBand>
//![1]
class ChartView : public QChartView
//![1]
{
public:
ChartView(QChart *chart, QWidget *parent = nullptr);
//![2]
protected:
bool viewportEvent(QEvent *event);
void mousePressEvent(QMouseEvent *event);
void mouseMoveEvent(QMouseEvent *event);
void mouseReleaseEvent(QMouseEvent *event);
void keyPressEvent(QKeyEvent *event);
//![2]
private:
bool m_isTouching = false;
};
#endif

View File

@ -1,32 +1,34 @@
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QtCore/qmetaobject.h>
#include <QLatin1StringView>
MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);
/*
if (localDevice.isValid()) { //при создании экземпляра класса он подключается к сввистку
localDevice.powerOn();
localDeviceName = localDevice.name();
qDebug() <<localDeviceName; // пишет "dongle"
localDevice.setHostMode(QBluetoothLocalDevice::HostDiscoverable);
QList<QBluetoothAddress> remotes;
remotes = localDevice.connectedDevices(); // если например наушники подключены и готовы выплевывает их адрес
// с другими, например сопряженными не работает
qDebug() <<remotes;
}
*/
bt = new Bluetooth1d801(this);
//конструктор для графика
auto series = new QLineSeries; //QLineSeries - делает массив точек для графиков. умеет append например
for (int i = 0; i < 500; i++) { //иииии вцикле
QPointF p(i, qSin(M_PI / 50 * i) * 100); // делаем точку c координатами в конструкторе
// p.ry() += QRandomGenerator::global()->bounded(20);
*series << p; // и приздяцим её в массив
} // и так по кругу
chart = new Chart; //новый график
chart->addSeries(series); // закидываем ему массив точек // функция унаследована, есть еще removeAllSeries
//chart->setTitle("Zoom in/out example");
chart->createDefaultAxes();
chart->setAnimationOptions(QChart::SeriesAnimations);
chart->legend()->hide(); // скрыть легенду, в моем случае бесячую точку
auto chartView = new ChartView(chart); // на основе точек создаем обьект
chartView->setRenderHint(QPainter::Antialiasing);
// this->setCentralWidget(chartView);// эта шляпа растягивает chartview на весь окно
ui->horizontalLayout->addWidget(chartView); // добавляем виджет
//конец конструктора для графика
}
MainWindow::~MainWindow()
@ -35,145 +37,67 @@ MainWindow::~MainWindow()
}
void MainWindow::on_pushButton_clicked()
void MainWindow::on_pushButtonSearch_clicked()
{
QBluetoothServiceDiscoveryAgent *discoveryAgent = new QBluetoothServiceDiscoveryAgent(this);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::serviceDiscovered,
this, &MainWindow::serviceDiscovered);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::finished,
this, &MainWindow::discoveryFinished);
connect(discoveryAgent, &QBluetoothServiceDiscoveryAgent::canceled,
this, &MainWindow::discoveryFinished); //странный коннект - слизал с бтчата (всмысле с указателями. Наверное это норм для динамических штук)
// начинаем поиск устройств
discoveryAgent->start();
if (discoveryAgent->isActive())
discoveryAgent->stop();
ui->listWidgetDevice->clear();
// m_discoveryAgent->setUuidFilter(uuid); // подумай? может стоит впиздячить его на релизе
discoveryAgent->start(QBluetoothServiceDiscoveryAgent::FullDiscovery);
ui->labelConnect->setText("Сейчас, я тебе что-нибудь найду");
bt->startDiscovery(); // запускаем поиск
connect( // забираем найденые результаты
bt, &Bluetooth1d801::discoveryResult,
this, &MainWindow::addDiscoveryResult
);
}
void MainWindow::stopDiscovery()
{/*
if (m_discoveryAgent){
m_discoveryAgent->stop();
}
*/
}
void MainWindow::serviceDiscovered(const QBluetoothServiceInfo &serviceInfo)
void MainWindow::on_pushButtonConnect_clicked()
{
qDebug()<< "Нажал кнопочку...)";
qDebug() << "Discovered service on"
<< serviceInfo.device().name() << serviceInfo.device().address().toString();
qDebug() << "\tService name:" << serviceInfo.serviceName();
qDebug() << "\tDescription:"
<< serviceInfo.attribute(QBluetoothServiceInfo::ServiceDescription).toString();
qDebug() << "\tProvider:"
<< serviceInfo.attribute(QBluetoothServiceInfo::ServiceProvider).toString();
qDebug() << "\tL2CAP protocol service multiplexer:"
<< serviceInfo.protocolServiceMultiplexer();
qDebug() << "\tRFCOMM server channel:" << serviceInfo.serverChannel();
const QBluetoothAddress address = serviceInfo.device().address();
/* // если находятся много сервисов в одной железке, то названия не дублируются
for (const QBluetoothServiceInfo &info : std::as_const(m_discoveredServices)) {
if (info.device().address() == address)
return; //
}
*/
//! [serviceDiscovered]
QString remoteName;
if (serviceInfo.device().name().isEmpty())
remoteName = address.toString();
else
remoteName = serviceInfo.device().name();
QListWidgetItem *item =new QListWidgetItem(QString::fromLatin1("%1 %2").arg(remoteName, serviceInfo.serviceName()));
discoveredServices.insert(item, serviceInfo);
ui->listWidgetDevice->addItem(item);
//! [serviceDiscovered]
}
void MainWindow::discoveryFinished()
{
ui->labelConnect->setText("Я сделал всё, что мог");
}
void MainWindow::deviceDiscovered(const QBluetoothDeviceInfo &device)
{
ui->listWidgetDevice->addItem(device.name());
qDebug() << "Found new device:" << device.name() << '(' << device.address().toString() << ')';
}
void MainWindow::on_pushButton_2_clicked()
{
auto items = ui->listWidgetDevice->selectedItems();
if (items.size()) {
QListWidgetItem *item = items[0];
QBluetoothServiceInfo service = discoveredServices.value(item);
qDebug() << "Connecting to service" << service.serviceName() //валим инфу в дебаг
<< "on" << service.device().name();
socket = new QBluetoothSocket(QBluetoothServiceInfo::RfcommProtocol); //создание сокета
qDebug() << "Create socket";
socket->connectToService(service);
connect(socket, &QBluetoothSocket::readyRead, this, &MainWindow::readSocket);
connect(socket, &QBluetoothSocket::connected, this, QOverload<>::of(&MainWindow::connected));
connect(socket, &QBluetoothSocket::disconnected, this, &MainWindow::disconnected);
connect(socket, &QBluetoothSocket::errorOccurred, this, &MainWindow::onSocketErrorOccurred);
qDebug() << "ConnectToService done";
ui->labelConnect->setText("Похоже получилось подключится");
}
}
void MainWindow::readSocket()
{
if (!socket)
return;
while (socket->canReadLine()) {
QByteArray line = socket->readLine().trimmed();
qDebug() << line;
ui->listWidgetResponse->addItem(line.toHex());
emit messageReceived(socket->peerName(),
QString::fromUtf8(line.constData(), line.length()));
}
}
void MainWindow::onSocketErrorOccurred(QBluetoothSocket::SocketError error)
{
if (error == QBluetoothSocket::SocketError::NoSocketError)
return;
QMetaEnum metaEnum = QMetaEnum::fromType<QBluetoothSocket::SocketError>();
QString errorString = socket->peerName()// + ' '_L1
+ metaEnum.valueToKey(static_cast<int>(error)) + " occurred"; //_L1;
emit socketErrorOccurred(errorString);
}
void MainWindow::connected()
{
emit connected(socket->peerName());
}
void MainWindow::on_pushButtonGetParam_clicked()
{
if (!socket){
qDebug()<< "Ты не создал подходящий сокет!!! Я не буду работать!!!";
return;
}
QString message = "Get_Param";
QByteArray text = message.toUtf8() + '\r' + '\n';
socket->write(text);
}
void MainWindow::addDiscoveryResult(QListWidgetItem *res_p)
{
ui->listWidgetDevice->addItem(res_p);
}
void MainWindow::addRawData(QByteArray data)
{
ui->listWidgetResponse->addItem(data.toHex());
}
void MainWindow::on_listWidgetDevice_itemDoubleClicked(QListWidgetItem *item)
{
if(!bt){
qDebug() << "телега пошла вперед лошади, сливай воду";
return;
}
bt->connectTo(item);
connect( // забираем сырые ответы
bt, &Bluetooth1d801::rawDataRecived,
this, &MainWindow::addRawData
);
}
void MainWindow::on_pushButtonClearCharts_clicked()
{
chart->removeAllSeries();
}
void MainWindow::on_pushButton_clicked()
{
auto series = new QLineSeries; //QLineSeries - делает массив точек для графиков. умеет append например
for (int i = 0; i < 500; i++) { //иииии вцикле
QPointF p(i, QRandomGenerator::global()->bounded(100)); // делаем точку c координатами в конструкторе
*series << p; // и приздяцим её в массив
} // и так по кругу
chart->addSeries(series);
}

View File

@ -3,22 +3,20 @@
#include <QMainWindow>
#include <QObject>
#include <QBluetoothDeviceDiscoveryAgent>
#include <QBluetoothServiceDiscoveryAgent>
#include <QBluetoothDeviceInfo>
#include <QBluetoothSocket>
#include <QBluetoothLocalDevice>
#include <QDebug>
#include <QListWidget>
#include <QtCharts>
#include <QLineSeries>
#include <QValueAxis>
#include <QChartView>
#include <QtMath>
#include <QtCore/qobject.h>
#include <QListWidget>
#include <bluetooth1d801.h>
#include "chart.h"
#include "chartview.h"
#include <QLineSeries>
#include <QRandomGenerator>
#include <QtMath>
#include <QValueAxis>
QT_BEGIN_NAMESPACE
@ -38,30 +36,28 @@ public:
~MainWindow();
private slots:
void on_pushButton_clicked();
void deviceDiscovered(const QBluetoothDeviceInfo &device);
void on_pushButton_2_clicked();
void stopDiscovery();
void serviceDiscovered(const QBluetoothServiceInfo &serviceInfo);
void discoveryFinished();
void readSocket();
void connected();
void onSocketErrorOccurred(QBluetoothSocket::SocketError error);
void on_pushButtonSearch_clicked();
void on_pushButtonConnect_clicked();
void on_pushButtonGetParam_clicked();
void addDiscoveryResult(QListWidgetItem *res_p);
void addRawData(QByteArray);
void on_listWidgetDevice_itemDoubleClicked(QListWidgetItem *item);
void on_pushButtonClearCharts_clicked();
void on_pushButton_clicked();
signals:
void messageReceived(const QString &sender, const QString &message);
void connected(const QString &name);
void disconnected();
void socketErrorOccurred(const QString &errorString);
private:
Ui::MainWindow *ui;
QBluetoothSocket *socket = nullptr;
QBluetoothLocalDevice localDevice;
QString localDeviceName;
QMap<QListWidgetItem *, QBluetoothServiceInfo> discoveredServices;
Bluetooth1d801 *bt;
Chart* chart;
QLineSeries* series;
};
#endif // MAINWINDOW_H

View File

@ -488,7 +488,7 @@
<string>MainWindow</string>
</property>
<widget class="QWidget" name="centralwidget">
<widget class="QPushButton" name="pushButton">
<widget class="QPushButton" name="pushButtonSearch">
<property name="geometry">
<rect>
<x>20</x>
@ -501,7 +501,7 @@
<string>Поиск устройств</string>
</property>
</widget>
<widget class="QPushButton" name="pushButton_2">
<widget class="QPushButton" name="pushButtonConnect">
<property name="geometry">
<rect>
<x>150</x>
@ -682,6 +682,43 @@
<string>Ответы</string>
</property>
</widget>
<widget class="QWidget" name="horizontalLayoutWidget">
<property name="geometry">
<rect>
<x>590</x>
<y>20</y>
<width>681</width>
<height>301</height>
</rect>
</property>
<layout class="QHBoxLayout" name="horizontalLayout"/>
</widget>
<widget class="QPushButton" name="pushButtonClearCharts">
<property name="geometry">
<rect>
<x>590</x>
<y>350</y>
<width>151</width>
<height>31</height>
</rect>
</property>
<property name="text">
<string>Очисить график</string>
</property>
</widget>
<widget class="QPushButton" name="pushButton">
<property name="geometry">
<rect>
<x>770</x>
<y>350</y>
<width>201</width>
<height>31</height>
</rect>
</property>
<property name="text">
<string>Постороить случайный</string>
</property>
</widget>
</widget>
<widget class="QMenuBar" name="menubar">
<property name="geometry">