crossplatform.ru

Здравствуйте, гость ( Вход | Регистрация )

> Дополнительный виджет, Плоттер
++Norton++
  опции профиля:
сообщение 4.3.2009, 23:38
Сообщение #1


Студент
*

Группа: Участник
Сообщений: 98
Регистрация: 5.10.2008
Пользователь №: 380

Спасибо сказали: 0 раз(а)




Репутация:   1  


Столкнулся с такой проблемой, нужно написать программу, которая будет строить определенный график, но в Qt не нашел виджета-плоттера. :(
Подскажите пожалуйста, есть ли какая-нибудь возможность решить эту проблему? Как отобразить график? Может есть какие-нибудь дополнительные виджеты?
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
 
Начать новую тему
Ответов
ufna
  опции профиля:
сообщение 24.9.2009, 23:18
Сообщение #2


Активный участник
***

Группа: Участник
Сообщений: 362
Регистрация: 24.5.2008
Из: Курган/СПб
Пользователь №: 182

Спасибо сказали: 29 раз(а)




Репутация:   5  


обычно такая хрень идет по двум причинам:
1. чет не то компилятор наделал, сделай полный ребилд
2. с инклюдами что-то не то (например "зацикливание")
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
AD
  опции профиля:
сообщение 24.9.2009, 23:23
Сообщение #3


Профессионал
*****

Группа: Участник
Сообщений: 2003
Регистрация: 4.2.2008
Из: S-Petersburg
Пользователь №: 84

Спасибо сказали: 70 раз(а)




Репутация:   17  


Думаю, что ребилд должен помочь!


Цитата
2. с инклюдами что-то не то (например "зацикливание")

Нет! Если все скопировать ровно как в примере, то зацикливания не будет. Код проверен и отлажен.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
AD
  опции профиля:
сообщение 27.9.2009, 15:13
Сообщение #4


Профессионал
*****

Группа: Участник
Сообщений: 2003
Регистрация: 4.2.2008
Из: S-Petersburg
Пользователь №: 84

Спасибо сказали: 70 раз(а)




Репутация:   17  


Ну... решил выложить еще раз решение. ПОЛНОСТЬЮ!
GraphicWidget

#ifndef PLOTSETTINGS_H
#define PLOTSETTINGS_H

#include <QPointF>
#include <cmath>

/// Àáñòðàêòíûé êëàññ äëÿ çàäàíèÿ äèàïàçîíà çíà÷åíèé ïî îñÿì x è y
class PlotSettings
{
public:
    double minX;                        ///< ìèíèìàëüíîå çíà÷åíèå ïî îñè àáñöèññ
    double maxX;                        ///< ìàêñèìàëüíîå çíà÷åíèå ïî îñè àáñöèññ
    int numXTicks;                        ///< êîëè÷åñòâî äåëåíèé íà îñè àáñöèññ
    double minY;                        ///< ìèíèìàëüíîå çíà÷åíèå ïî îñè îðäèíàò
    double maxY;                        ///< ìàêñèìàëüíîå çíà÷åíèå ïî îñè îðäèíàò
    int numYTicks;                        ///< êîëè÷åñòâî äåëåíèé íà îñè îðäèíàò

protected:
    void adjustAxis(double& min, double& max, int& numTicks);

public:
    PlotSettings();
    void scroll(double dx, double dy);
    void scale(double delta_x, double delta_y);
    void adjust();
    void adjust(QPointF& point);
    double spanX() const { return fabs(maxX - minX); }
    double spanY() const { return fabs(maxY - minY); }
};

#endif


#include "PlotSettings.h"

PlotSettings::PlotSettings(): minX(-50.), minY(-4.), maxX(50.), maxY(45.)
{
    numXTicks = 8;
    numYTicks = 8;
}

/// Óâåëè÷åíèå/óìåíüøåíèå çíà÷åíèÿ minX, maxX, minY, maxY íà èíòåðâàë ìåæäó 2-ìÿ îòìåòêàìè
void PlotSettings::scroll(double dx, double dy)
{
    double stepX = spanX() / numXTicks;
    minX += dx * stepX;
    maxX += dx * stepX;

    double stepY = spanY() / numYTicks;
    minY += dy * stepY;
    maxY += dy * stepY;

    adjust();
}

/// Óâåëè÷åíèå/óìåíüøåíèå çíà÷åíèÿ minX, maxX, minY, maxY íà èíòåðâàë ìåæäó 2-ìÿ îòìåòêàìè
void PlotSettings::scale(double delta_x, double delta_y)
{
    if((minX == maxX || minY == maxY) && delta_x < 0 && delta_y < 0) return;

    double stepX = spanX() / numXTicks;
    minX -= delta_x * stepX;
    maxX += delta_x * stepX;

    double stepY = spanY() / numYTicks;
    minY -= delta_y * stepY;
    maxY += delta_y * stepY;
}

/// Îêðóãëåíèå çíà÷åíèé minX, minY, maxX, maxY
void PlotSettings::adjust()
{
    adjustAxis(minX, maxX, numXTicks);
    adjustAxis(minY, maxY, numYTicks);
}

/// Îêðóãëåíèå çíà÷åíèé çàäàííîé òî÷êè point
void PlotSettings::adjust(QPointF& point)
{
    double mn_x = minX, mn_y = minY;
    int ticks_x = numXTicks, ticks_y = numYTicks;
    double mx_x = point.x(), mx_y = point.y();

    adjustAxis(mn_x, mx_x, ticks_x);
    adjustAxis(mn_y, mx_y, ticks_y);

    point.setX(mx_x);
    point.setY(mx_y);
}

/// Ïðåîáðàçîâàíèå ïàðàìåòðîâ â óäîáíûå çíà÷åíèÿ
void PlotSettings::adjustAxis(double& min, double&  max, int& numTicks)
{
    const int MinTicks = 5;
    double grossStep = (max - min) / MinTicks;
    double step = pow(10.0, floor(log10(grossStep)));

    if(5 * step < grossStep)
        step *= 5;
    else if(2 * step < grossStep)
        step *= 2;

    numTicks = int(ceil(max / step) - floor(min / step));        
    if(numTicks < MinTicks)
        numTicks = MinTicks;

    min = floor(min / step) * step;
    max = ceil(max / step) * step;
}


#ifndef GRAPHICWIDGET_H
#define GRAPHICWIDGET_H

#include "ui_GraphicWidget.h"
#include "PlotSettings.h"

#include <QtGui/QWidget>
#include <QPoint>

class QPaintEvent;
class QPainter;
class QRubberBand;
class QKeyEvent;
class QResizeEvent;
class QCloseEvent;
class QShowEvent;
class QWheelEvent;
class QMouseEvent;

/// Âèäæåò, ãäå áóäåò îòðèñîâûâàòüñÿ ãðàôèê
class GraphicWidget: public QWidget, public Ui::GraphicWidgetClass
{
    Q_OBJECT

private:
    PlotSettings settings;        ///< íàñòðîéêà äëÿ ðàçëè÷íûõ ìàñøòàáîâ
    QRubberBand* rubber;        ///< "ðåçèíîâàÿ ëåíòà"
    bool rubberBandIsShown;        ///< ôëàæîê ïîïàäàíèÿ êóðñîðà â "ðåçèíîâóþ ëåíòó"
    QPoint origin;                ///< íà÷àëüíûå êîîðäèíàòû âûäåëÿåìîé îáëàñòè
    QVector<QPointF> curve_vec;    ///< âåêòîð òî÷åê êðèâîé

private:
    void drawGrid(QPainter *painter);
    void drawCurves(QPainter* painter);
    QPointF initXY(double sx, double sy);
    void initCurves();

protected:
    void paintEvent(QPaintEvent* events);
    void keyPressEvent(QKeyEvent* events);
    void wheelEvent(QWheelEvent* events);
    void mousePressEvent(QMouseEvent* events);
    void mouseMoveEvent(QMouseEvent* events);
    void mouseReleaseEvent(QMouseEvent* events);
    void resizeEvent(QResizeEvent* events) { QWidget::resizeEvent(events); update(); }
    void closeEvent(QCloseEvent* events) { QWidget::closeEvent(events); }
    void showEvent(QShowEvent* events) { QWidget::showEvent(events); }

public:
    GraphicWidget(QWidget *parent = 0, Qt::WFlags flags = 0);
    ~GraphicWidget();
    void setPlotSettings(const PlotSettings& sts) { settings = sts; settings.adjust(); update(); }
    void zoom(double delta) { settings.scale(delta, delta); settings.adjust(); update(); }
};

#endif // GRAPHICWIDGET_H


#include "GraphicWidget.h"

#include <QPainter>
#include <QRect>
#include <QString>
#include <QRubberBand>
#include <QKeyEvent>
#include <QWheelEvent>
#include <QMouseEvent>
#include <algorithm>
using namespace std;

GraphicWidget::GraphicWidget(QWidget *parent, Qt::WFlags flags): QWidget(parent, flags), rubberBandIsShown(false),
                rubber(new QRubberBand(QRubberBand::Rectangle, this))
{
    setupUi(this);

    /// Èíèöèàëèçàöèÿ íåîáõîäèìûõ ïàðàìåòðîâ
    setPlotSettings(PlotSettings());

    initCurves();
}

GraphicWidget::~GraphicWidget()
{}

/// Çàïîëíåíèÿ âåêòîðà ñ êðèâîé (êðèâàÿ y(x) = x)
void GraphicWidget::initCurves()
{
    curve_vec.append(QPointF(-100., -100.));
    curve_vec.append(QPointF(-10., -10.));
    curve_vec.append(QPointF(-1., -1.));
    curve_vec.append(QPointF(0., 0.));
    curve_vec.append(QPointF(1., 1.));
    curve_vec.append(QPointF(1.5, 1.5));
    curve_vec.append(QPointF(5., 5.));
    curve_vec.append(QPointF(10., 10.));
    curve_vec.append(QPointF(100., 100.));
}

/// Îòðèñîâêà ñåòêè
void GraphicWidget::drawGrid(QPainter *painter)
{
    QRect rect(graphWidget -> rect());
    if(!rect.isValid()) return;
    
    QRect boundString;    
    double great_max = max(settings.numXTicks, settings.numYTicks) + 1;
    for(register int i=0, j=0, k=0; i<=great_max; ++i, ++j, ++k)
    {
        if(j <= settings.numXTicks)                ///< îòðèñîâêà âäîëü îñè X
        {
            int x = rect.left() + (j * (rect.width() - 1) / settings.numXTicks);
            double label = settings.minX + (j * settings.spanX() / settings.numXTicks);
            QString s_label(QString::number(label, 'f', 3));

            painter -> setPen(Qt::black);
            painter -> drawLine(x, rect.top(), x, rect.bottom());

            int flags = Qt::AlignHCenter | Qt::AlignTop;
            boundString = painter -> boundingRect(boundString, flags, s_label);    
            painter -> drawText(x - (boundString.width() + 5), rect.bottom() - (boundString.height() + 5),
                            boundString.width(), boundString.height(), flags, s_label);
        }
        if(k <= settings.numYTicks)                ///< îòðèñîâêà âäîëü îñè Y
        {
            int y = rect.bottom() - (k * (rect.height() - 1) / settings.numYTicks);
            double label = settings.minY + (k * settings.spanY() / settings.numYTicks);
            QString s_label(QString::number(label, 'f', 3));

            painter -> setPen(Qt::black);
            painter -> drawLine(rect.left(), y, rect.right(), y);

            int flags = Qt::AlignRight | Qt::AlignTop;
            boundString = painter -> boundingRect(boundString, flags, s_label);    
            painter -> drawText(rect.left() + 7, y - boundString.height(),
                                boundString.width(), boundString.height(), flags, s_label);
        }
    }
    painter -> drawRect(rect.adjusted(0, 0, -1, -1));
}

/// Èíèöèàëèçàöèÿ êîîðäèíàò - ïðåîáðàçîâàíèå èç êîîðäèíàò ãðàôèêà (sx,sy) â ýêðàííûå (x,y)
QPointF GraphicWidget::initXY(double sx, double sy)
{
    QRect rect(graphWidget -> rect());
    double dx, dy;

    /// Âû÷èñëåíèå ñìåùåíèé âäîëü îñåé
    dx = sx - settings.maxX;
    dy = sy - settings.minY;

    /// Âû÷èñëåíèå ýêðàííûõ êîîðäèíàò
    double x = rect.right() + (dx * (rect.width() - 1) / settings.spanX());
    double y = rect.bottom() - (dy * (rect.height() - 1) / settings.spanY());

    return QPointF(x, y);
}

/// Îòðèñîâêà ãðàôèêà
void GraphicWidget::drawCurves(QPainter* painter)
{
    painter -> setPen(Qt::blue);
    QPolygonF polyline(curve_vec.size());
    for(register int i=0; i<curve_vec.size(); ++i)
        polyline[i] = initXY(curve_vec[i].x(), curve_vec[i].y());
    painter -> drawPolyline(polyline);
}

/// Îòðèñîâêà ãðàôèêà
void GraphicWidget::paintEvent(QPaintEvent* events)
{
    QPainter painter(this);
    drawGrid(&painter);
    drawCurves(&painter);

    QWidget::paintEvent(events);
}

/// Íàæàòèå íà êíîïêè êëàâèàòóðû
void GraphicWidget::keyPressEvent(QKeyEvent* events)
{
    switch(events -> key())
    {
    case Qt::Key_Plus:
        zoom(-1.0);
    break;
    case Qt::Key_Minus:
        zoom(1.0);
    break;
    case Qt::Key_Left:
        settings.scroll(-1.0, 0.0);
        update();
    break;
    case Qt::Key_Right:
        settings.scroll(1.0, 0.0);
        update();
    break;
    case Qt::Key_Up:
        settings.scroll(0.0, 1.0);
        update();
    break;
    case Qt::Key_Down:
        settings.scroll(0.0, -1.0);
        update();
    break;
    case Qt::Key_X:
        if(events -> modifiers() & Qt::ALT) close();
    break;
    case Qt::Key_Return:
        if(events -> modifiers() & Qt::ALT)
        {
            if(!isMaximized()) showMaximized();
            else showNormal();
        }
    break;
    default:
        QWidget::keyPressEvent(events);
    }
}

/// Èçìåíåíèå ìàñøòàáà ïðè äâèæåíèè êîëåñèêà
void GraphicWidget::wheelEvent(QWheelEvent* events)
{
    int numDegrees = events -> delta() / 8;
    double numTicks = numDegrees / 15.0;
    
    zoom(numTicks);
    update();
}

/// Íàæàòèå íà êíîïêó - èçìåíåíèå ìàñøòàáà
void GraphicWidget::mousePressEvent(QMouseEvent* events)
{
    QWidget::mousePressEvent(events);

    QRect r;
    switch(events -> button())
    {
    case Qt::LeftButton:                ///< åñëè íàæàòà ëåâàÿ êíîïêà ìûøè
        origin = events -> pos();
        rubberBandIsShown = true;
        setCursor(Qt::CrossCursor);
        r = QRect(origin, QSize());
        rubber -> setGeometry(r);
        rubber -> show();
    break;
    case Qt::RightButton:                ///< åñëè íàæàòà ïðàâàÿ êíîïêà ìûøè
    break;
    default:
    break;
    }
}

/// Ïåðåîïðåäåëåíèå ôóíêöèè ïåðåäâèæåíèÿ ìûøè
void GraphicWidget::mouseMoveEvent(QMouseEvent* events)
{
    if(rubberBandIsShown)
    {
        rubber -> setWindowOpacity(0.0);
        rubber -> setGeometry(QRect(origin, events -> pos()).normalized());
    }

    update();
}

/// Âîçâðàùåíèå ïðåæíåãî âèäà êóðñîðó è èçìåíåíèå ìàñøòàáà
void GraphicWidget::mouseReleaseEvent(QMouseEvent* events)
{
    if(events -> button() == Qt::LeftButton && rubberBandIsShown)
    {
        rubberBandIsShown = false;
        unsetCursor();
        
        QRect rect = rubber -> geometry().normalized();
        if(rect.width() < 10 || rect.height() < 10)
        {
            double dx = rect.width() / settings.spanX();
            double dy = rect.height() / settings.spanY();
            settings.scroll(dx, dy);

            update();
            return;
        }
        PlotSettings prevSettings = settings;

        double dx = prevSettings.spanX() / width();
        double dy = prevSettings.spanY() / height();
        settings.minX = prevSettings.minX + dx * rect.left();
        settings.maxX = prevSettings.minX + dx * rect.right();
        settings.minY = prevSettings.maxY - dy * rect.bottom();
        settings.maxY = prevSettings.maxY - dy * rect.top();
        settings.adjust();

        rubber -> hide();
        update();
    }
}

Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение

Сообщений в этой теме
- ++Norton++   Дополнительный виджет   4.3.2009, 23:38
- - SABROG   Это? http://qwt.sourceforge.net/class_qwt_plot.htm...   5.3.2009, 0:04
- - AD   Цитата(++Norton++ @ 4.3.2...   5.3.2009, 0:19
- - ++Norton++   SABROG, спасибо, пробовал, скачивал, только не пол...   5.3.2009, 0:34
|- - AD   Цитата(++Norton++ @ 5.3.2...   5.3.2009, 0:35
|- - ++Norton++   Конечно, все отлично, буду ждать!   5.3.2009, 0:50
- - Litkevich Yuriy   ++Norton++, про Qwt есть отдельная тема, почитай т...   5.3.2009, 8:16
- - kwisp   Цитата(++Norton++ @ 5.3.2...   5.3.2009, 9:02
- - Kagami   Я тоже свой виджет для построения графиков делал   5.3.2009, 9:45
|- - AD   Первое. Желательно создать новый проект. Основной ...   5.3.2009, 11:52
- - ++Norton++   Спасибо, все понятно кроме некоторых деталей: 1) ...   5.3.2009, 12:21
|- - AD   Цитата(++Norton++ @ 5.3.2...   5.3.2009, 12:32
- - ++Norton++   Вопрос, конечно ламерский, но все же А как можно...   5.3.2009, 12:47
|- - AD   Пофигу где. Главное, чтобы ты смог открыть окно, а...   5.3.2009, 12:55
- - ++Norton++   Цитата(AD @ 5.3.2009, 13:55) нужная форма...   5.3.2009, 13:05
|- - AD   Обведенный в овальчик!   5.3.2009, 13:19
- - ++Norton++   Огромное спасибо! Отлично, все получилось...   6.3.2009, 11:15
- - ++Norton++   И всетаки, как же строить графики?   7.3.2009, 11:02
|- - AD   Цитата(++Norton++ @ 7.3.2...   7.3.2009, 13:11
|- - AD   Добавить в класс 3 функции: private: void dra...   7.3.2009, 14:00
- - Kagami   Напоминает пример из книжки Жасмин Бланшет   7.3.2009, 14:26
|- - AD   Цитата(Kagami @ 7.3.2009, 14:26) Напомина...   7.3.2009, 14:51
- - Litkevich Yuriy   AD, Оборачивай длинные портянки кода в тэг экспанд   7.3.2009, 14:48
- - ++Norton++   Вроде все сделал так, но при компиляции выдается о...   7.3.2009, 16:15
|- - AD   Цитата(++Norton++ @ 7.3.2...   7.3.2009, 16:20
- - ++Norton++   Огромнейшее спасибо! Теперь все отлично работа...   7.3.2009, 20:40
- - ++Norton++   Да, хотел только еще один вопрос задать... Когда н...   7.3.2009, 22:38
|- - AD   ЦитатаДа, хотел только еще один вопрос задать... К...   7.3.2009, 22:56
|- - BRE   Цитата(++Norton++ @ 7.3.2...   7.3.2009, 23:08
|- - AD   GraphicWidget::GraphicWidget(QWidget *...   7.3.2009, 23:15
- - ++Norton++   Спасибо! Теперь все работает! Только лента...   8.3.2009, 0:46
|- - AD   Цитата(++Norton++ @ 8.3.2...   10.3.2009, 12:14
|- - AD   Цитата(++Norton++ @ 8.3.2...   6.11.2009, 15:49
- - asdf   а подскажите по этому примеру в чём косяк. при доб...   24.9.2009, 11:17
|- - AD   Цитата(asdf @ 24.9.2009, 12:17) а подскаж...   24.9.2009, 18:06
|- - asdf   Цитата(AD @ 24.9.2009, 18:06) Прошу приве...   24.9.2009, 23:10
- - ufna   обычно такая хрень идет по двум причинам: 1. чет н...   24.9.2009, 23:18
|- - AD   Думаю, что ребилд должен помочь! Цитата2. с...   24.9.2009, 23:23
|- - AD   Ну... решил выложить еще раз решение. ПОЛНОСТЬЮ...   27.9.2009, 15:13
- - Litkevich Yuriy   AD, чёто с русскими коментариями бяка какая-то   27.9.2009, 17:01
|- - AD   Цитата(Litkevich Yuriy @ 27.9.2009, 18:01...   27.9.2009, 17:10
- - Гость   Уважаемые, поясните пожалуйста что за файл подключ...   24.12.2009, 21:20
|- - AD   Цитата(Гость @ 24.12.2009, 21:20) Уважаем...   24.12.2009, 22:33
|- - Fitz   Цитата(Гость @ 24.12.2009, 21:20) Уважаем...   2.12.2010, 14:26
- - Litkevich Yuriy   Цитата(Гость_Гость_* @ 25.12.2009, 0:20) ...   24.12.2009, 22:35
- - AlmNeft   AD, ++Norton++, выложите пожалуйста папку с прогой...   17.4.2010, 22:43
|- - AD   Цитата(AlmNeft @ 17.4.2010, 23:43) AD, ++...   17.4.2010, 23:28
- - Fitz   /// Инициализация координат - преобразование из ко...   4.2.2011, 15:02


Быстрый ответОтветить в данную темуНачать новую тему
Теги
Нет тегов для показа


4 чел. читают эту тему (гостей: 4, скрытых пользователей: 0)
Пользователей: 0




RSS Текстовая версия Сейчас: 29.11.2024, 15:45