使用QWidget画圆环图的几种方法

今天无意看到了别人画3d饼图,无意中发现了画圆环图的一些方法;由此,做一个分享。

1、使用QRegion做出一个圆环区域,使用QPainterPath添加这个区域,然后fillPath。

    QPainter painter(this);
    painter.save();
    //设置反锯齿
    painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform | QPainter::Qt4CompatiblePainting);
    QRect drawRect = event->rect();
    QRegion region(drawRect.adjusted(10,10,-10,-10),QRegion::Ellipse);
    drawRect.setSize(QSize(drawRect.width()/2,drawRect.height()/2));
    drawRect.moveTopLeft(QPoint((event->rect().width() - drawRect.width())/2,(event->rect().height() - drawRect.height())/2));
    QRegion region2(drawRect,QRegion::Ellipse);
    QPainterPath painterPath;
    painterPath.addRegion(region.subtracted(region2));
    painter.fillPath(painterPath,Qt::red);
    event->accept();
    painter.restore();

2、使用裁剪,使得只有裁剪区域才能被绘制

void Widget::paintEvent(QPaintEvent *event)
{
    QPainter painter(this);
    painter.save();
    //设置反锯齿
    painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform | QPainter::Qt4CompatiblePainting);
    QRect drawRect = event->rect();
    QRegion region(drawRect.adjusted(10,10,-10,-10),QRegion::Ellipse);
    drawRect.setSize(QSize(drawRect.width()/2,drawRect.height()/2));
    drawRect.moveTopLeft(QPoint((event->rect().width() - drawRect.width())/2,(event->rect().height() - drawRect.height())/2));
    QRegion region2(drawRect,QRegion::Ellipse);
    painter.setClipRegion(region.subtracted(region2));
    painter.fillRect(region.boundingRect(),Qt::yellow);
    event->accept();
    painter.restore();
}

3、在圆的中心画一个与背景颜色相同的圆

void Widget::paintEvent(QPaintEvent *event)
{
    QPainter painter(this);
    painter.save();
    //设置反锯齿
    painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform | QPainter::Qt4CompatiblePainting);
    QRect drawRect = event->rect();
    painter.setBrush(Qt::yellow);
    painter.drawEllipse(drawRect.adjusted(10,10,-10,-10));
    painter.setBrush(painter.background());
    painter.drawEllipse(drawRect.adjusted(drawRect.width()/4,drawRect.height()/4,-drawRect.width()/4,-drawRect.height()/4));
    event->accept();
    painter.restore();
}

4、采用线型渐变,在中心位置到内环半径位置为透明,环部分为颜色值。这里的半径位置在线形渐变中用比例表示,这种只能画圆环,椭圆环不行

void Widget::paintEvent(QPaintEvent *event)
{
    QPainter painter(this);
    painter.save();
    //设置反锯齿
    painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform | QPainter::Qt4CompatiblePainting);
    QRect drawRect = event->rect();
    QRadialGradient rg(drawRect.center(),drawRect.width()/2,drawRect.center());
    rg.setColorAt(0,Qt::transparent);
    rg.setColorAt(0.6,Qt::transparent);
    rg.setColorAt(0.61,Qt::yellow);
    rg.setColorAt(1,Qt::yellow);
    painter.setBrush(rg);
    painter.drawEllipse(drawRect);
    event->accept();
    painter.restore();
}


第一种、第二种有略微锯齿,因为在做区域交叉算法时有取舍,painter设置的反锯齿是给画图操作函数用才有效。

第三种效果较好,可以是圆环图或椭圆环图,需要背景色一致,但这个背景色在painter里面可以直接得到

第四种效果也很好,但只能是圆环,但他可以是其它效果,比如将线形渐变的焦点不设置在圆心,中间不是透明

void Widget::paintEvent(QPaintEvent *event)
{
    QPainter painter(this);
    painter.save();
    //设置反锯齿
    painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform | QPainter::Qt4CompatiblePainting);
    QRect drawRect = event->rect();
    QRadialGradient rg(drawRect.center(),drawRect.width()/2,drawRect.center() -= QPoint(30,80));
    rg.setColorAt(0,QColor(Qt::red).darker(400));
    rg.setColorAt(0.6,QColor(Qt::red).darker(150));
    rg.setColorAt(0.61,QColor(Qt::yellow).darker(150));
    rg.setColorAt(1,QColor(Qt::yellow).lighter(200));
    painter.setBrush(rg);
    painter.drawEllipse(drawRect);
    event->accept();
    painter.restore();
}





评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值