View Full Version : Delete single widget from QLayoutGrid
KeineAhnung
12th May 2014, 13:27
Hi there,
I have been trying to delete a widget from a grid layout all morning without success. Either the widget is still there or the program crashes. Can someone tell me how this should be done? The widgets are added using addWidget(cardVector[k], k/4, k%4);
I have a grid layout with two rows and four columns containing different cards. I would like to delete one of these cards at any position and want that the other cards move up and that an empty "field" is added at the end. How do I do that?
Thanks for taking a look!
anda_skoa
12th May 2014, 14:07
Since you also want to reposition other widgets, the easiest way is to delete the layout and create a new one and only add those widgets you want at the position you want to be.
Cheers,
_
KeineAhnung
12th May 2014, 15:03
I thought about that too but the grid layout was placed in Creator and is part of another layout. If I delete it I do not know how to put a new one back at the same position. Currently I can access the layout using ui->cardLayout but it is the first object of main layout. If I recreate it I need to be able to access as ui->cardLayout again otherwise I need to rewrite most of my code...
anda_skoa
12th May 2014, 16:38
In such a case it is handy to have a widget as a container in the place you currently have the just the layout.
I.e. instead of nesting the grid layout in another, put a plain widget into the outer layout and use a grid on the widget.
Then you can easily just recreate the grid layout for that "container".
Cheers,
_
KeineAhnung
12th May 2014, 21:05
Sorry but I still cannot crack it. I went with your advise and created a Widget in Creator and gave it the size I want the card area to have. In the code I initialize the cards:
cardVector.resize(8);
QGridLayout *grid = new QGridLayout;
grid->setRowMinimumHeight(0,240);
grid->setRowMinimumHeight(1,240);
int k = 0;
for(int i = 0; i < 2; ++i){
for(int j = 0; j < 4; ++j){
cardVector[k] = new Card;
cardVector[k]->index = k;
[...]
grid->addWidget(cardVector[k], i, j);
k++;
}
}
ui->cardContainer->setLayout(grid);
This works fine but redoing it does not work.
cards.removeAt(activeIndex);
cardIndex--;
cardVector.removeAt(activeIndex);
int k = 7;
cardVector.resize(8);
cardVector[k] = new Card;
[...]
// redoing the vector
for(int k = 0; k < fighterIndex; ++k){
card[k].name = card[k].type+" "+QString::number(k+1);
cardVector[k]->setTitle(card[k].name);
updateCard(k);
}
// Redoing the layout
QLayoutItem* item;
while((item = ui->cardContainer->layout()->takeAt(0)) != NULL ){
delete item->widget();
delete item;
}
delete ui->cardContainer->layout();
QGridLayout *grid = new QGridLayout;
grid->setRowMinimumHeight(0,240);
grid->setRowMinimumHeight(1,240);
k = 0;
for(int i = 0; i< 2; ++i){
for(int j = 0; j<4;++j){
cardVector[k]->index = k;
grid->addWidget(cardVector[k], i, j); // <- at this line the program crashes
k++;
}
}
ui->cardContainer->setLayout(grid);
}
I do not see what is different from the first part. Why does the program crash when I add a widget to the layout here? The code is identical to the first one, isn't it? What can I do to find the problem?
Thanks for the help!
anda_skoa
13th May 2014, 09:01
You have a while loop that deletes all widgets.
Then you try to add the now danlging pointers to the layout.
Since you always seem to have 8 cards (even the second code block has cardVector.resize(8)), why delete at all?
My initial suggestion was based on the assmuption that "deleting" means you have one less widget than before.
Cheers,
_
KeineAhnung
13th May 2014, 09:23
I tried to rewrite them without deleting them first. Then I ended up with the "new" cards on top of the old cards.
Actually I only need to update the view. Adjusting the two vectors with the information is easy, but I do not get the view updated. =(
What do you mean with dangling pointers? I thought I have deleted them properly to start from scratch.
anda_skoa
13th May 2014, 10:47
I tried to rewrite them without deleting them first. Then I ended up with the "new" cards on top of the old cards.
If you don't delete you don't have to create new cards, right?
Actually I only need to update the view. Adjusting the two vectors with the information is easy, but I do not get the view updated. =(
Do you call the respective setters on each card so that it is filled with new new data?
What do you mean with dangling pointers? I thought I have deleted them properly to start from scratch.
You delete all widgets in the layout here in a while loop, but you do not clear the cardVector. So all pointers in cardVector are now "dangling" (point to some memory that no longer holds a Card widget).
You then take these pointers and try to add them to the layout again.
As I suggested in comment #2 it would be easier to just deleted the layout and re-add the widgets you want at the position you want, no need to delete or create any widgets.
Obviously filling the existing and layouted widgets with new data is even better, but your initial request didn't say that this was on option.
Cheers,
_
KeineAhnung
13th May 2014, 13:45
Actually it is not an option. Refilling did not work. The connection between the cardVector and the widget is so strong that even after shortening the vector the connection still existed. So after deleting a card and adding a new one the newest one would still be added on the next position and not on top of the last one.
I finally got it to work by destroying the cardVector as well and und redoing it.
Thanks for your help _!
Powered by vBulletin® Version 4.2.5 Copyright © 2024 vBulletin Solutions Inc. All rights reserved.