Qt Forum

    • Login
    • Search
    • Categories
    • Recent
    • Tags
    • Popular
    • Users
    • Groups
    • Search
    • Unsolved

    Update: Forum Guidelines & Code of Conduct


    Qt World Summit: Early-Bird Tickets

    Unsolved Layout, widgets not inside ???

    General and Desktop
    9
    87
    1217
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • JoeCFD
      JoeCFD @SPlatten last edited by JoeCFD

      @SPlatten You created a verticalLayoutWidget inside a group box which has 4 children:
      verticalLayout
      radio1
      radio2
      radio3

      verticalLayoutWidget is not needed. the parent of verticalLayoutWidget is groupBox.

      SPlatten 1 Reply Last reply Reply Quote 0
      • SPlatten
        SPlatten @JoeCFD last edited by SPlatten

        @JoeCFD , to repeat what I did:

        • Dragged Group Box from Containers palette and dropped onto centralWidget
        • Dragged Vertical Layout from Layouts palette and dropped onto Group Box
        • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
        • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
        • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
        • Edited text attribute for each control

        Thats it, one layout not two. Take a look at the XML I posted, its pretty clear and its the actual elements that are generated from this XML that I'm questioning.

        Kind Regards,
        Sy

        1 Reply Last reply Reply Quote 0
        • JoeCFD
          JoeCFD @JoeCFD last edited by

          @JoeCFD
          if there is no title for the group, set top margin to 0
          QString group_style_sheet = QString("QGroupBox{border: 2px solid gray;border-radius: 8px;margin-top: 0px;}" );

          SPlatten 1 Reply Last reply Reply Quote 0
          • SPlatten
            SPlatten @JoeCFD last edited by

            @JoeCFD , but why? What I'm trying to understand is why if done in Qt Creator WYSIWYG, no fiddles are required and it works, but the code generated from the XML just doesn't add up.

            Kind Regards,
            Sy

            JoeCFD 1 Reply Last reply Reply Quote 0
            • JoeCFD
              JoeCFD @SPlatten last edited by JoeCFD

              @SPlatten Could be some global settings in your project which cause this problem. Since all my widgets are customized, I do not complain about this. BTW, border-radius setting is bad sometimes when screen resolution is not high enough. Often, I have to set it to 0.

              SPlatten 1 Reply Last reply Reply Quote 0
              • SPlatten
                SPlatten @JoeCFD last edited by

                @JoeCFD , please see latest edit....

                Kind Regards,
                Sy

                JoeCFD 1 Reply Last reply Reply Quote 0
                • JoeCFD
                  JoeCFD @SPlatten last edited by JoeCFD

                  @SPlatten
                  Dragged Group Box from Containers palette and dropped onto centralWidget
                  Dragged Radio Button from Buttons palette
                  Dragged Radio Button from Buttons palette
                  Dragged Radio Button from Buttons palette
                  right click inside qgroupbox and select layout->Lay out Vertically ---> size policy is added here automatically.
                  Edited text attribute for each control

                  do this for each major layout.

                  SPlatten 1 Reply Last reply Reply Quote 0
                  • SPlatten
                    SPlatten @JoeCFD last edited by

                    @JoeCFD , This was just an exercise using Qt Creator to see what it does and why the works compared to coding it by hand....

                    Kind Regards,
                    Sy

                    JoeCFD 1 Reply Last reply Reply Quote 0
                    • JoeCFD
                      JoeCFD @SPlatten last edited by JoeCFD

                      @SPlatten
                      group layout

                      SPlatten 1 Reply Last reply Reply Quote 0
                      • SPlatten
                        SPlatten @JoeCFD last edited by

                        @JoeCFD , can you not paste the source or post what you did?

                        Kind Regards,
                        Sy

                        1 Reply Last reply Reply Quote 0
                        • JoeCFD
                          JoeCFD last edited by

                          link added

                          SPlatten 1 Reply Last reply Reply Quote 0
                          • SPlatten
                            SPlatten @JoeCFD last edited by

                            @JoeCFD , thank you, I'm still trying to see why I cannot replicate in code what the WYSIWYG editor does.

                            Kind Regards,
                            Sy

                            1 Reply Last reply Reply Quote 0
                            • SPlatten
                              SPlatten last edited by

                              I thought I'd hit on something, I found another post online and added:

                              pobjLayout->addItem(new QWidgetItem(pobjWidget));
                              

                              No joy, still doesn't work! Looking at the layout in the debugger, after executing this line, there are no additions to the children, same with addWidget...so what do I have to do?

                              Kind Regards,
                              Sy

                              1 Reply Last reply Reply Quote 0
                              • SGaist
                                SGaist Lifetime Qt Champion last edited by

                                Provide a minimal compilable example of that specific part of your code so that people can experiment with it.

                                Interested in AI ? www.idiap.ch
                                Please read the Qt Code of Conduct - https://forum.qt.io/topic/113070/qt-code-of-conduct

                                SPlatten 1 Reply Last reply Reply Quote 2
                                • SPlatten
                                  SPlatten @SGaist last edited by SPlatten

                                  @SGaist , this is difficult, but I will try:

                                  This is the XML:

                                  	<groupbox id="enSEX" eol="true" align="left" layout="vertical" title="Radios"
                                  			  dbfield="vcSex" properties="QGroupBox { border-radius: 8px; background-color:#ff0000; }">
                                  		<radiobutton id="rdoM" text="Male" default="true" position="0,0"/>
                                  		<radiobutton id="rdoF" text="Female" position="1,0"/>
                                  	</groupbox>
                                  

                                  I have quite a lot of code that parses all the various node types, here is the bit for the above:

                                              if ( pobjParent != nullptr ) {
                                      //Does the parent have a layout?
                                                  pobjLayout = pobjParent->pobjGetLayout();
                                                  if ( pobjLayout != nullptr ) {
                                      //Yes, clear the pobjParWidget widget
                                                      pobjParWidget = nullptr;
                                                  }
                                              }
                                              pobjWidget = qobject_cast<QWidget*>(clsCNT::pCreate(this,
                                                                                                  mstrName,
                                                                                                  strCSS,
                                                                                                  slstProperties,
                                                                                                  pobjParWidget));
                                  

                                  The above code calls pCreate which based on the content of mstrName creates the widget:

                                  QWidget* clsCNT::pCreate(clsXMLnode* pobjNode, const QString& crstrType
                                                          ,QString& rstrCSS, QStringList& rslstProperties
                                                          ,QWidget* pobjParent) {
                                      if ( clsCNT::blnValidType(crstrType) == true ) {
                                          if ( crstrType.compare(clsCNT::mscszButton) == 0 ) {
                                              return new clsQtPushBtn(pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszCheckbox) == 0 ) {
                                              return new clsQtCheckBox(pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszCombo) == 0 ) {
                                              return new clsQtComboBox(pobjNode, &rstrCSS, &rslstProperties, pobjParent);
                                          } else if( crstrType.compare(clsCNT::mscszFrame) == 0 ) {
                                              return new clsQtFrame(pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszGroupBox) == 0 ) {
                                              return new clsQtGroupBox(pobjNode, &rstrCSS, &rslstProperties, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszLabel) == 0 ) {
                                              return new clsQtLabel(pobjNode, &rstrCSS, &rslstProperties, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszLineEdit) == 0 ) {
                                              return new clsQtLineEdit(pobjNode, &rstrCSS, &rslstProperties, pobjParent);            
                                          } else if ( crstrType.compare(clsCNT::mscszListWidget) == 0 ) {
                                              return new clsQtListWidget(pobjNode, &rstrCSS, &rslstProperties, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszRadioButton) == 0 ) {
                                              return new clsQtRadioButton(pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszSliderVert) == 0 ) {
                                              return new clsQtSlider(Qt::Vertical, pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszSliderHorz) == 0 ) {
                                              return new clsQtSlider(Qt::Horizontal, pobjNode, &rstrCSS, pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszSpinBox) == 0 ) {
                                              return new QSpinBox(pobjParent);
                                          } else if ( crstrType.compare(clsCNT::mscszTextEdit) == 0 ) {
                                              return new clsQtTextEdit(pobjNode, &rstrCSS, &rslstProperties, pobjParent);
                                          }
                                      }
                                      return nullptr;
                                  }
                                  

                                  Give the widget a human readable name and add it to the layout if one exists:

                                          if ( pobjWidget != nullptr ) {
                                  //Set the object name
                                              QString strID(strGetAttribute(clsXMLnode::mscszAttrID)),
                                                      strName(mstrName);
                                              if ( strID.isEmpty() != true ) {
                                                  strName += ", id: " + strID;
                                              }
                                              pobjWidget->setObjectName(strName);
                                          }
                                          if ( pobjLayout != nullptr ) {
                                  //Any alignment?
                                              QString strAlign(pobjParent->strGetAttribute(clsXMLnode::mscszAttrAlignment));
                                              Qt::Alignment eCellAlign(Qt::AlignVCenter | Qt::AlignHCenter);
                                              if ( strAlign.compare(clsXMLnode::mscszGeomCenter) == 0 ) {
                                                  eCellAlign |= Qt::AlignHCenter;
                                              } else if ( strAlign.compare(clsXMLnode::mscszGeomLeft) == 0 ) {
                                                  eCellAlign |= Qt::AlignLeft;
                                              } else if ( strAlign.compare(clsXMLnode::mscszGeomRight) == 0 ) {
                                                  eCellAlign |= Qt::AlignRight;
                                              }
                                  //Add the widget to the layout
                                              QLayoutItem* pobjItem(new QWidgetItem(pobjWidget));
                                              pobjItem->setAlignment(eCellAlign);
                                              pobjLayout->addItem(pobjItem);
                                  //            pobjLayout->addWidget(pobjWidget);
                                          }
                                  

                                  Hopefully this is enough, however at this point I really don't think its something I've done wrong but a limitation or restriction in Qt, bearing in mind that my code does this all at run-time where as the UI create in Qt is parsed when the project is compiled.

                                  [Edit], I just decided to debug again and single stepped into addItem, interesting:

                                  void QBoxLayout::addItem(QLayoutItem *item)
                                  {
                                      Q_D(QBoxLayout);
                                      QBoxLayoutItem *it = new QBoxLayoutItem(item);
                                      d->list.append(it);
                                      invalidate();
                                  }
                                  

                                  When I get into addItem, on QBoxLayoutItem *it = new QBoxLayoutItem(item);:
                                  Screenshot 2021-10-20 at 20.25.33.png

                                  If I click on Step Over twice, the cursor is pointing to invalidate();, the watch window has changed to:
                                  Screenshot 2021-10-20 at 20.27.08.png

                                  Looks very wrong and after returning back to where the call took place there is no addition to the layout.

                                  addWidget calls addItem internally, so if its a bug in addItem this explains everything.

                                  [Edit 2] I have file a bug:
                                  https://bugreports.qt.io/browse/QTBUG-97662

                                  Kind Regards,
                                  Sy

                                  JKSH 1 Reply Last reply Reply Quote 0
                                  • JKSH
                                    JKSH Moderators @SPlatten last edited by JKSH

                                    @SPlatten said in Layout, widgets not inside ???:

                                    my code does this all at run-time where as the UI create in Qt is parsed when the project is compiled.

                                    While Qt Designer's .ui files are parsed at compile time, the parser generates C++ code that runs at runtime.

                                    You said that the WYSIWYG editor works. So, build a working GUI and then inspect the ui_*.h file that is generated in your build folder. Copy the code inside the setupUi() method.

                                    Qt Doc Search for browsers: forum.qt.io/topic/35616/web-browser-extension-for-improved-doc-searches

                                    SPlatten 2 Replies Last reply Reply Quote 4
                                    • SPlatten
                                      SPlatten @JKSH last edited by

                                      @JKSH , thank you, I will take a look.

                                      Kind Regards,
                                      Sy

                                      1 Reply Last reply Reply Quote 0
                                      • SPlatten
                                        SPlatten @JKSH last edited by SPlatten

                                        @JKSH , this is from the header file:

                                            QGroupBox *groupBox;
                                            QWidget *verticalLayoutWidget;
                                            QVBoxLayout *verticalLayout;
                                            QRadioButton *Radio1;
                                            QRadioButton *Radio2;
                                            QRadioButton *Radio3;
                                        

                                        Will take a good look, because the verticalLayoutWidget looks like its added by Qt Creator and is obviously required, and thats something thats missing in my code.

                                        Then there is this code:

                                                groupBox = new QGroupBox(centralWidget);
                                                groupBox->setObjectName(QString::fromUtf8("groupBox"));
                                                groupBox->setGeometry(QRect(120, 170, 301, 201));
                                                verticalLayoutWidget = new QWidget(groupBox);
                                                verticalLayoutWidget->setObjectName(QString::fromUtf8("verticalLayoutWidget"));
                                                verticalLayoutWidget->setGeometry(QRect(9, 29, 291, 161));
                                                verticalLayout = new QVBoxLayout(verticalLayoutWidget);
                                                verticalLayout->setSpacing(6);
                                                verticalLayout->setContentsMargins(11, 11, 11, 11);
                                                verticalLayout->setObjectName(QString::fromUtf8("verticalLayout"));
                                                verticalLayout->setContentsMargins(0, 0, 0, 0);
                                                Radio1 = new QRadioButton(verticalLayoutWidget);
                                                Radio1->setObjectName(QString::fromUtf8("Radio1"));
                                        
                                                verticalLayout->addWidget(Radio1);
                                        
                                                Radio2 = new QRadioButton(verticalLayoutWidget);
                                                Radio2->setObjectName(QString::fromUtf8("Radio2"));
                                        
                                                verticalLayout->addWidget(Radio2);
                                        
                                                Radio3 = new QRadioButton(verticalLayoutWidget);
                                                Radio3->setObjectName(QString::fromUtf8("Radio3"));
                                        
                                                verticalLayout->addWidget(Radio3);
                                        

                                        Which clearly creates a new widget and positions it between QGroupBox and QVBoxLayout.

                                        Kind Regards,
                                        Sy

                                        1 Reply Last reply Reply Quote 0
                                        • JoeCFD
                                          JoeCFD last edited by

                                          @SPlatten said in Layout, widgets not inside ???:

                                          Which clearly creates a new widget and positions it between QGroupBox and QVBoxLayout.

                                          What is for? The gaps between verticalLayoutWidget and groupBox can be handled with verticalLayout->setContentsMargins(11, 11, 11, 11); verticalLayoutWidget is not needed. Check the example in my link.

                                          SPlatten 1 Reply Last reply Reply Quote 0
                                          • SPlatten
                                            SPlatten @JoeCFD last edited by

                                            @JoeCFD , I think you misunderstand what I am saying. I used Qt Creator to build this example, I did the following:

                                            • Dragged Group Box from Containers palette and dropped onto centralWidget
                                            • Dragged Vertical Layout from Layouts palette and dropped onto Group Box
                                            • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
                                            • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
                                            • Dragged Radio Button from Buttons palette and dropped onto Vertical Layout
                                            • Edited text attribute for each control

                                            I did not do anything to add the mysterious verticalLayoutWidget which was added by Qt Creator. I am saying this works and its only because Qt Creator is doing something else instead of adding a QGroupBox, QVBoxLayout and child widgets.

                                            Kind Regards,
                                            Sy

                                            JoeCFD 1 Reply Last reply Reply Quote 0
                                            • First post
                                              Last post