自定义的Qt给统计图添加颜色样例控件

本文主要是给博文“自定义的Qt折线图控件”的图表添加颜色样例。之所以分开叙述是因为本文也是自定义控件,它不仅可以给折线图添加,还可以给其他类型的图表(柱状图、饼形图等,如果后续有这类文章的话)添加。本文代码在VS2015和Qt5.9上测试通过。此类较简单,用法是new MColorComment实例,并或然后将MPolyline设为它的父亲,再调用它的add函数向其中添加颜色和对应的文本注释就行了。默认的,该控件显示在MPolyline的上方居中位置。下面是效果图:

上代码,头文件:

class MColorComment : public QWidget
{
    Q_OBJECT

public:
    MColorComment(QWidget* parent = nullptr);
    void setParent(QWidget *parent); /* 隐藏父类方法 */
    void add(QRgb rgb, const QString& text);

private:
    void paintEvent(QPaintEvent *event) override;
    bool eventFilter(QObject *watched, QEvent *event) override;

private:
    QMultiMap<QRgb, QString> comments;
};

CPP文件:

MColorComment::MColorComment(QWidget* parent) : 
    QWidget(parent)
{
    if (parent)
    {
        parent->installEventFilter(this);
    }
}

void MColorComment::setParent(QWidget *parent)
{
    QObject* old = this->parent();
    if (old)
    {
        removeEventFilter(old);
    }
    QWidget::setParent(parent);
    if (parent)
    {
        parent->installEventFilter(this);
    }
}

void MColorComment::add(QRgb rgb, const QString& text)
{
    comments.insert(rgb, text);
    update();
}

void MColorComment::paintEvent(QPaintEvent *event)
{
    const int pad1 = 2; /* 色块和文字之间间距 */
    const int pad2 = 10; /* 文字后的留白 */
    const QSize colorBlock(10, 8);
    QPainter painter(this);
    QFontMetrics fm = painter.fontMetrics();
    int sum = 0;
    for (const auto& str : comments.values())
    {
        QSize strsz = fm.size(0, str);
        sum += colorBlock.width();
        sum += pad1;
        sum += strsz.width();
        sum += pad2;
    }
    sum -= pad2; /* 结尾多加的 */

    int xpos = (width() - sum) / 2;
    for (auto it = comments.begin(); it != comments.end(); it++)
    {
        QRect temp;
        temp.setX(xpos);
        temp.setY((height() - colorBlock.height()) / 2);
        temp.setSize(colorBlock);
        painter.fillRect(temp, QColor(it.key()));
        xpos += colorBlock.width();
        xpos += pad1;
        QSize itsz = fm.size(0, it.value());
        temp.setX(xpos);
        temp.setY((height() - itsz.height()) / 2);
        temp.setSize(itsz);
        painter.drawText(temp, it.value());
        xpos += itsz.width();
        xpos += pad2;
    }
}

bool MColorComment::eventFilter(QObject *watched, QEvent *event)
{
    QWidget* p = dynamic_cast<QWidget*>(watched);
    if (event->type() == QEvent::Resize)
    {
        setFixedSize(p->width(), 20);
    }
    return false;
}

下面给出主窗口构造函数中设置折线图和它的颜色样例的例子。代码中ui.poLines是折线图(MPolyline*):

QtTest::QtTest(QWidget *parent)
    : QMainWindow(parent)
{
    ui.setupUi(this);

    ...

    qsrand((int)time(0));
    MColorComment* comment = new MColorComment();
    comment->setParent(ui.poLines);
    for (int i = 0; i < 2; i++)
    {
        QVector<QPointF> pts;
        for (int j = 0; j < 10; j++)
        {
            pts.push_back(QPointF(qrand() % 200 - 10, qrand() % 100));
        }
        std::sort(pts.begin(), pts.end(), [](QPointF a, QPointF b) { return a.x() < b.x(); });
        std::for_each(pts.begin(), pts.end(), [](QPointF& x) { return x *= 1; });
        QRgb color = ui.poLines->append(pts);
        comment->add(color, QString(u8"颜色%1").arg(i));
    }

    ...

}

 

posted @ 2022-11-12 04:34  兜尼完  阅读(202)  评论(0编辑  收藏  举报