如何访问QGridLayout中的小部件坐标

How to access widgets coordinates in QGridLayout

本文关键字:小部 坐标 QGridLayout 何访问 访问      更新时间:2023-10-16

我正在将qlabel放在QGridLayout中,以便在从QWidget派生的类中很好地排列它们。但是,当我尝试访问坐标时,它总是返回(0,0)

下面是我的代码:
class brick : public QWidget
{
Q_OBJECT
public:
    explicit brick(QWidget *parent);
    ...
private:
    QLabel* label;
    QPixmap* brickPic;
}
brick::brick(QWidget *parent) :
QWidget(parent)
{
rect=new QRect();
label=new QLabel;
brickPic=new QPixmap(100,15);
brickPic->fill(QColor(255,0,0));
label->setPixmap(*brickPic);
}
class Grid : public QWidget
{    
QOBJECT
public:
    void fill_grid(QWidget* parent);
    ...
private:
    QGridLayout* grid;
}
void Grid::fill_grid(QWidget* parent)
{
for (int i=0;i<10;i++)
{
    for (int j=0;j<12;j++)
    {
        brick* fillingBrick=new brick(parent);
        grid->addWidget(fillingBrick->getLabel(),j,i);
        qDebug()<<fillingBrick->parentWidget();
        brickVector.push_back(fillingBrick);
    }
}
}

这些将像我之前所说的那样显示在以下从QWidget派生的类中:

class render_area : public QWidget
{
Q_OBJECT
public:
    render_area(QWidget *parent = 0);
    Grid* getGrid() const;
    ...
private:
    Grid* grid;
    QRect* bar;
    ...
}
render_area::render_area(QWidget *parent)
:QWidget(parent)
{
    setFocusPolicy(Qt::StrongFocus);
    setBackgroundRole(QPalette::Base);
    setAutoFillBackground(true);
    bar=new QRect(width()/2,height()+50,150,10);
    grid=new Grid(this);
    grid->fill_grid(this);
    std::cout<<"Grid filled !"<<std::endl;
    qDebug()<<grid->getBrick(5)->pos();
    ...
}

qDebug()<<fillingBrick->parentWidget()返回良好的render_area,而qDebug()<<grid->getBrick(5)->pos()返回不良的QPoint(0,0)

我想知道如何让pos()x()y()返回render_area中的坐标,其中每个砖都放置在其中。我尝试了从QGridLayout的cellRect,从QWidget继承的MapToParent(),迫使render_areashow();,但到目前为止没有运气。

编辑:这就是我说的循环:

void render_area::update_timer()
{
int x1,x2,y1,y2;
bar->getCoords(&x1,&y1,&x2,&y2);
if(is_fixed==false)
{
    //gestion du sol
    if(yc+diametre>height())
    {
        timer.stop();
        QDialog* gameOver=new QDialog;
        gameOver->setLayout(new QGridLayout);
        gameOver->layout()->addWidget(new QPushButton("Ok"));
        gameOver->setGeometry(500,500,300,150);
        gameOver->show();
    }
    if(xc>x1 && xc<x1+150)
    {
        if(yc+diametre>y1)
        {
            vyc=-vyc;
        }
    }
    //plafond
    for (int i=0;i<12;i++)
    {
        for(int j=0;j<10;j++)
        {
            grid->getBrick(i,j)->setRect(grid->getGrid()->cellRect(i,j));
        }
    }
    for(int i=0;i<widgets.size();i++)
    {
            if(widgets.at(i)->getRect()->intersects(ballRectangle))
            {
                std::cout<<i<<std::endl;
                widgets.at(i)->getPixmap()->fill(QColor(255,255,255));
                widgets.at(i)->getLabel()->setPixmap(*(widgets.at(i)->getPixmap()));
                delete widgets.at(i);
                vyc=-vyc;
            }
            //qDebug()<<grid->getBrick(i,j)->getRect();
    }
    //bord droit
    if(xc+diametre>width())
    {
            vxc=-vxc;
    }
    //bord gauche
    if(xc<0)
    {
            vxc=-vxc;
    }
    //integration (Euler explicite)
    xc=xc+dt*vxc;
    yc=yc+dt*vyc;
}
repaint();
}

小部件的位置和大小由布局决定,并在事件循环调用中计算,有时调用窗口小部件上的show()之后,有时该窗口变得可见之前。

既然你想实现一个破砖游戏,有两种方法:

  1. 使用图形视图/场景系统

您基于小部件的方法很好地证明了您的问题的解决方案可以通过而不是检查砖的位置来自然解决。

你将在一个计时器事件中更新球的位置,或者使用动画系统。在任何一种情况下,球小部件都将收到一个具有新位置的moveEvent。在这里,您可以轻松地计算球矩形与每个砖小部件矩形的交点。因为它们是同一个父结点的子结点,所以它们将在同一个坐标系中。到那时,所有的职位都将是当前的。在检测到碰撞后,您将反射球的方向,并真正地delete砖小部件。你也可以开始播放"hit"的音效。

使用QObject属性系统来标记不同的砖部件是相当容易的,这取决于它们的行为特征。所有这些都可以完成,而无需子类化基小部件,例如可以是QFrame