Skip to content
  • Categories
  • Recent
  • Tags
  • Popular
  • Users
  • Groups
  • Search
  • Get Qt Extensions
  • Unsolved
Collapse
Brand Logo
  1. Home
  2. Qt Development
  3. General and Desktop
  4. Qt 6: Move QList contents without copying?
Forum Updated to NodeBB v4.3 + New Features

Qt 6: Move QList contents without copying?

Scheduled Pinned Locked Moved Unsolved General and Desktop
21 Posts 6 Posters 3.5k Views 3 Watching
  • 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.
  • CJhaC Offline
    CJhaC Offline
    CJha
    wrote on last edited by
    #1

    Hi, This might sound like an easy question, but I am not sure I understand Implicit Sharing and std::move correctly. I would like to avoid copying the contents of a QList while moving it between classes. My current method is

    void addList(QList<T>& list) { fullList.append(std::move(list)); }
    

    I pass any QList (e.g. addList(myList) where myList is a QList<T>) to this function and it appends the passed QList to fullList which is also QList<T>. I have noticed that the list which is passed still contains the content after I have used std::move to move it to the fullList. Is the content of list now implicitly shared or just copied? Is there any way I can make sure that no copy is being made when I am passing the list to be appended to my fullList?

    Axel SpoerlA 1 Reply Last reply
    0
    • CJhaC CJha

      Hi, This might sound like an easy question, but I am not sure I understand Implicit Sharing and std::move correctly. I would like to avoid copying the contents of a QList while moving it between classes. My current method is

      void addList(QList<T>& list) { fullList.append(std::move(list)); }
      

      I pass any QList (e.g. addList(myList) where myList is a QList<T>) to this function and it appends the passed QList to fullList which is also QList<T>. I have noticed that the list which is passed still contains the content after I have used std::move to move it to the fullList. Is the content of list now implicitly shared or just copied? Is there any way I can make sure that no copy is being made when I am passing the list to be appended to my fullList?

      Axel SpoerlA Offline
      Axel SpoerlA Offline
      Axel Spoerl
      Moderators
      wrote on last edited by
      #2

      @CJha
      There is no general answer to this question, because the exact behavior of std::move()depends on the compiler. In general terms it just tells the compiler, that the buffer originally used by the moved object (the list) is no longer reserved. Allowing others to play in that garden, doesn't mean that anybody is actually interested or has the need to play there.
      The behavior of QListis documented here. In a nut shell: It works with shallow copies to the extent possible and sensible. No deep copies are created as long as elements are accessed read-only.

      Having said that, the probable answer is: What is still shown as an object, is ready to be re-used and plundered.

      If it's necessary to be completely sure that nothing's copied, a QList of pointers to heap-allocated objects is maybe an alternative. A qDeleteAll(myList) in a class destructor is a handy tool in that case.

      Software Engineer
      The Qt Company, Oslo

      JonBJ 1 Reply Last reply
      1
      • Axel SpoerlA Axel Spoerl

        @CJha
        There is no general answer to this question, because the exact behavior of std::move()depends on the compiler. In general terms it just tells the compiler, that the buffer originally used by the moved object (the list) is no longer reserved. Allowing others to play in that garden, doesn't mean that anybody is actually interested or has the need to play there.
        The behavior of QListis documented here. In a nut shell: It works with shallow copies to the extent possible and sensible. No deep copies are created as long as elements are accessed read-only.

        Having said that, the probable answer is: What is still shown as an object, is ready to be re-used and plundered.

        If it's necessary to be completely sure that nothing's copied, a QList of pointers to heap-allocated objects is maybe an alternative. A qDeleteAll(myList) in a class destructor is a handy tool in that case.

        JonBJ Online
        JonBJ Online
        JonB
        wrote on last edited by JonB
        #3

        @Axel-Spoerl
        Hi Axel :) An observation from me. Since Qt6 decided in its wisdom to make QLists "QList stores its items in an array of continuous memory." so that they are arrays/vectors, by definition one cannot avoid copying when appending one list to another, unlike with a traditional or linked list. (I don't think Qt's implicit sharing helps here, even if the new list is not written to?)

        I am talking about copying for the QList elements themselves. This may indeed be a shallow copy if for example the elements are pointers to objects.

        Although you say

        If it's necessary to be completely sure that nothing's copied, a QList of pointers to heap-allocated objects is maybe an alternative.

        the OP should be aware that this does not "copy *nothing"". So for example if you want to append a QList with a million items to one with a single item the million items are going to have to be copied, even if that's just a million pointers to copy.... So QList might not be the best choice for such a requirement. Am I right?

        Axel SpoerlA CJhaC 2 Replies Last reply
        3
        • JonBJ JonB

          @Axel-Spoerl
          Hi Axel :) An observation from me. Since Qt6 decided in its wisdom to make QLists "QList stores its items in an array of continuous memory." so that they are arrays/vectors, by definition one cannot avoid copying when appending one list to another, unlike with a traditional or linked list. (I don't think Qt's implicit sharing helps here, even if the new list is not written to?)

          I am talking about copying for the QList elements themselves. This may indeed be a shallow copy if for example the elements are pointers to objects.

          Although you say

          If it's necessary to be completely sure that nothing's copied, a QList of pointers to heap-allocated objects is maybe an alternative.

          the OP should be aware that this does not "copy *nothing"". So for example if you want to append a QList with a million items to one with a single item the million items are going to have to be copied, even if that's just a million pointers to copy.... So QList might not be the best choice for such a requirement. Am I right?

          Axel SpoerlA Offline
          Axel SpoerlA Offline
          Axel Spoerl
          Moderators
          wrote on last edited by
          #4

          @JonB Hi Jon :-) You are (as always) right! It really depends on the use case, but the one you described is certainly not, what the inventors of QList had in mind.

          Software Engineer
          The Qt Company, Oslo

          JonBJ 1 Reply Last reply
          1
          • Axel SpoerlA Axel Spoerl

            @JonB Hi Jon :-) You are (as always) right! It really depends on the use case, but the one you described is certainly not, what the inventors of QList had in mind.

            JonBJ Online
            JonBJ Online
            JonB
            wrote on last edited by
            #5

            @Axel-Spoerl
            I know, I just wish they left vectors as vectors and lists as lists! Doubtless they gain efficiency/space in many/most cases,. But they even removed QLinkedList and say go use std::list() for that instead!

            Axel SpoerlA 1 Reply Last reply
            1
            • JonBJ JonB

              @Axel-Spoerl
              I know, I just wish they left vectors as vectors and lists as lists! Doubtless they gain efficiency/space in many/most cases,. But they even removed QLinkedList and say go use std::list() for that instead!

              Axel SpoerlA Offline
              Axel SpoerlA Offline
              Axel Spoerl
              Moderators
              wrote on last edited by
              #6

              @JonB
              I hear you! It wasn't me, but I know who it was and who approved it!

              Software Engineer
              The Qt Company, Oslo

              1 Reply Last reply
              1
              • JonBJ JonB

                @Axel-Spoerl
                Hi Axel :) An observation from me. Since Qt6 decided in its wisdom to make QLists "QList stores its items in an array of continuous memory." so that they are arrays/vectors, by definition one cannot avoid copying when appending one list to another, unlike with a traditional or linked list. (I don't think Qt's implicit sharing helps here, even if the new list is not written to?)

                I am talking about copying for the QList elements themselves. This may indeed be a shallow copy if for example the elements are pointers to objects.

                Although you say

                If it's necessary to be completely sure that nothing's copied, a QList of pointers to heap-allocated objects is maybe an alternative.

                the OP should be aware that this does not "copy *nothing"". So for example if you want to append a QList with a million items to one with a single item the million items are going to have to be copied, even if that's just a million pointers to copy.... So QList might not be the best choice for such a requirement. Am I right?

                CJhaC Offline
                CJhaC Offline
                CJha
                wrote on last edited by
                #7

                @JonB QList does have an append overload for moving:

                void QList::append(QList<T> &&value)
                This is an overloaded function.
                Moves the items of the value list to the end of this list.
                This function was introduced in Qt 6.0.

                Technically, it should not copy but rather move and append the value from the passed list to fullList. Why is this not doing that?

                JonBJ 1 Reply Last reply
                0
                • CJhaC CJha

                  @JonB QList does have an append overload for moving:

                  void QList::append(QList<T> &&value)
                  This is an overloaded function.
                  Moves the items of the value list to the end of this list.
                  This function was introduced in Qt 6.0.

                  Technically, it should not copy but rather move and append the value from the passed list to fullList. Why is this not doing that?

                  JonBJ Online
                  JonBJ Online
                  JonB
                  wrote on last edited by
                  #8

                  @CJha
                  I don't know since I have never used Qt 6.

                  Depends what you/they mean by "move" versus "copy". I stick by my claim/understanding that since QList elements are now contiguous like an array/vector it must "copy" the elements to the destination list to make it so. The only difference is that it could invalidate the passed-in source list.

                  I do not know how QList<T> might work with std::move(). I do not know whether it might leave list in an "unspecified state". Did you try, say, fullList.append(list); just to seen what happens to list after this case?

                  This is the limit of my knowledge/guesswork! You probably need a QT + std expert to help you further :)

                  jsulmJ CJhaC 2 Replies Last reply
                  0
                  • JonBJ JonB

                    @CJha
                    I don't know since I have never used Qt 6.

                    Depends what you/they mean by "move" versus "copy". I stick by my claim/understanding that since QList elements are now contiguous like an array/vector it must "copy" the elements to the destination list to make it so. The only difference is that it could invalidate the passed-in source list.

                    I do not know how QList<T> might work with std::move(). I do not know whether it might leave list in an "unspecified state". Did you try, say, fullList.append(list); just to seen what happens to list after this case?

                    This is the limit of my knowledge/guesswork! You probably need a QT + std expert to help you further :)

                    jsulmJ Offline
                    jsulmJ Offline
                    jsulm
                    Lifetime Qt Champion
                    wrote on last edited by
                    #9

                    @JonB No it does not have to copy if using std::move. As far as I know Qt container use heap allocated memory with a pointer for the actual data. Then using std::move it is enough to assign the pointer to the data in the destination container and set it to nullptr in the source container.

                    https://forum.qt.io/topic/113070/qt-code-of-conduct

                    JonBJ 1 Reply Last reply
                    0
                    • JonBJ JonB

                      @CJha
                      I don't know since I have never used Qt 6.

                      Depends what you/they mean by "move" versus "copy". I stick by my claim/understanding that since QList elements are now contiguous like an array/vector it must "copy" the elements to the destination list to make it so. The only difference is that it could invalidate the passed-in source list.

                      I do not know how QList<T> might work with std::move(). I do not know whether it might leave list in an "unspecified state". Did you try, say, fullList.append(list); just to seen what happens to list after this case?

                      This is the limit of my knowledge/guesswork! You probably need a QT + std expert to help you further :)

                      CJhaC Offline
                      CJhaC Offline
                      CJha
                      wrote on last edited by
                      #10

                      @JonB Thanks. Yes, I did try multiple cases, here are few important ones

                      fullList.append(list); // copies and appends the data from list to fullList but list itself is not changed
                      
                      fullList.append(std::move(list)); // same as above
                      
                      fullList = std::move(list); // moves the data from list to fullList, the list is empty but in defined state after this move
                      

                      It seems that it is only moving the data from list to fullList when the data is being assigned using = std::move() in all other cases it leaves the data in list as it was.

                      1 Reply Last reply
                      0
                      • jsulmJ jsulm

                        @JonB No it does not have to copy if using std::move. As far as I know Qt container use heap allocated memory with a pointer for the actual data. Then using std::move it is enough to assign the pointer to the data in the destination container and set it to nullptr in the source container.

                        JonBJ Online
                        JonBJ Online
                        JonB
                        wrote on last edited by
                        #11

                        @jsulm said in Qt 6: Move QList contents without copying?:

                        @JonB No it does not have to copy if using std::move. As far as I know Qt container use heap allocated memory with a pointer for the actual data. Then using std::move it is enough to assign the pointer to the data in the destination container and set it to nullptr in the source container.

                        Have you read what I wrote above about Qt6 and contiguous element allocation? What you write would indeed be the case for QList::QList(const QList<T> &other) constructor. However the user is asking about QList::append(QList<T> &&value) method introduced at Qt 6.0, that's the whole point. Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                        jsulmJ Christian EhrlicherC 2 Replies Last reply
                        0
                        • JonBJ JonB

                          @jsulm said in Qt 6: Move QList contents without copying?:

                          @JonB No it does not have to copy if using std::move. As far as I know Qt container use heap allocated memory with a pointer for the actual data. Then using std::move it is enough to assign the pointer to the data in the destination container and set it to nullptr in the source container.

                          Have you read what I wrote above about Qt6 and contiguous element allocation? What you write would indeed be the case for QList::QList(const QList<T> &other) constructor. However the user is asking about QList::append(QList<T> &&value) method introduced at Qt 6.0, that's the whole point. Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                          jsulmJ Offline
                          jsulmJ Offline
                          jsulm
                          Lifetime Qt Champion
                          wrote on last edited by
                          #12

                          @JonB Sorry, you're of course right about append!

                          https://forum.qt.io/topic/113070/qt-code-of-conduct

                          1 Reply Last reply
                          0
                          • JonBJ JonB

                            @jsulm said in Qt 6: Move QList contents without copying?:

                            @JonB No it does not have to copy if using std::move. As far as I know Qt container use heap allocated memory with a pointer for the actual data. Then using std::move it is enough to assign the pointer to the data in the destination container and set it to nullptr in the source container.

                            Have you read what I wrote above about Qt6 and contiguous element allocation? What you write would indeed be the case for QList::QList(const QList<T> &other) constructor. However the user is asking about QList::append(QList<T> &&value) method introduced at Qt 6.0, that's the whole point. Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                            Christian EhrlicherC Offline
                            Christian EhrlicherC Offline
                            Christian Ehrlicher
                            Lifetime Qt Champion
                            wrote on last edited by Christian Ehrlicher
                            #13

                            @JonB said in Qt 6: Move QList contents without copying?:

                            Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                            It moves the data from memory a to memory b. It's just not a copy but a move operation. Nothing more.

                            void addList(QList<T>& list) { fullList.append(std::move(list)); }

                            How should this be moved at all when list is a const ref? You can move a const object...

                            Qt Online Installer direct download: https://download.qt.io/official_releases/online_installers/
                            Visit the Qt Academy at https://academy.qt.io/catalog

                            JonBJ 2 Replies Last reply
                            0
                            • Christian EhrlicherC Christian Ehrlicher

                              @JonB said in Qt 6: Move QList contents without copying?:

                              Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                              It moves the data from memory a to memory b. It's just not a copy but a move operation. Nothing more.

                              void addList(QList<T>& list) { fullList.append(std::move(list)); }

                              How should this be moved at all when list is a const ref? You can move a const object...

                              JonBJ Online
                              JonBJ Online
                              JonB
                              wrote on last edited by JonB
                              #14

                              @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                              It moves the data from memory a to memory b. It's just not a copy but a move operation. Nothing more.

                              For this purpose I am calling that "copy" because (does it not?) have to copy the elements (in the QList, not what they might point to) from memory area a to area b? Certainly if iterates that is O(n), I don't know whether you mean CPUs have an instruction to do that fast instead? Think about if it appends a list of 1,000,000 items to an existing list of one item. Moving and appending a list (like a linked list) could be made an O(1) operation, but not with the vector implementation of Qt6 QList, that is my point. I think of "copy" for O(n), "move" to me implies O(1) (hopefully). Is this not correct?

                              Christian EhrlicherC S 2 Replies Last reply
                              0
                              • JonBJ JonB

                                @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                It moves the data from memory a to memory b. It's just not a copy but a move operation. Nothing more.

                                For this purpose I am calling that "copy" because (does it not?) have to copy the elements (in the QList, not what they might point to) from memory area a to area b? Certainly if iterates that is O(n), I don't know whether you mean CPUs have an instruction to do that fast instead? Think about if it appends a list of 1,000,000 items to an existing list of one item. Moving and appending a list (like a linked list) could be made an O(1) operation, but not with the vector implementation of Qt6 QList, that is my point. I think of "copy" for O(n), "move" to me implies O(1) (hopefully). Is this not correct?

                                Christian EhrlicherC Offline
                                Christian EhrlicherC Offline
                                Christian Ehrlicher
                                Lifetime Qt Champion
                                wrote on last edited by
                                #15

                                @JonB Don't look on the QList but it's contents. The content is moved but not copied - so the move ctor is called and not the copy ctor.

                                Qt Online Installer direct download: https://download.qt.io/official_releases/online_installers/
                                Visit the Qt Academy at https://academy.qt.io/catalog

                                JonBJ 1 Reply Last reply
                                0
                                • Christian EhrlicherC Christian Ehrlicher

                                  @JonB Don't look on the QList but it's contents. The content is moved but not copied - so the move ctor is called and not the copy ctor.

                                  JonBJ Online
                                  JonBJ Online
                                  JonB
                                  wrote on last edited by JonB
                                  #16

                                  @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                  so the move ctor is called and not the copy ctor.

                                  I never suggested otherwise. My question/answer at least is based on my assumption that the implementation in move constructor will be O(n) and not O(1) at Qt6 QList which requires contiguous memory. Which is what I would care about (at least as part of the whole) if I were asking the OP's question.

                                  Christian EhrlicherC 1 Reply Last reply
                                  1
                                  • JonBJ JonB

                                    @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                    so the move ctor is called and not the copy ctor.

                                    I never suggested otherwise. My question/answer at least is based on my assumption that the implementation in move constructor will be O(n) and not O(1) at Qt6 QList which requires contiguous memory. Which is what I would care about (at least as part of the whole) if I were asking the OP's question.

                                    Christian EhrlicherC Offline
                                    Christian EhrlicherC Offline
                                    Christian Ehrlicher
                                    Lifetime Qt Champion
                                    wrote on last edited by Christian Ehrlicher
                                    #17

                                    @JonB said in Qt 6: Move QList contents without copying?:

                                    Which is what I would care about (at least as part of the whole) if I were asking the OP's question.

                                    But even then a move would be faster than (or equal if QList contains a pod) a copy. But the complexity stays, yes. But as I said - it can't work the way it is written.

                                    Qt Online Installer direct download: https://download.qt.io/official_releases/online_installers/
                                    Visit the Qt Academy at https://academy.qt.io/catalog

                                    1 Reply Last reply
                                    0
                                    • Christian EhrlicherC Christian Ehrlicher

                                      @JonB said in Qt 6: Move QList contents without copying?:

                                      Could you explain how it could append a list to an existing list without having to copy the elements onto the existing list while still maintaining contiguous memory like an array/vector? I welcome your input :)

                                      It moves the data from memory a to memory b. It's just not a copy but a move operation. Nothing more.

                                      void addList(QList<T>& list) { fullList.append(std::move(list)); }

                                      How should this be moved at all when list is a const ref? You can move a const object...

                                      JonBJ Online
                                      JonBJ Online
                                      JonB
                                      wrote on last edited by JonB
                                      #18

                                      @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                      But the complexity stays, yes.

                                      OK.

                                      @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                      How should this be moved at all when list is a const ref? You can move a const object...

                                      @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                      But as I said - it can't work the way it is written.

                                      Sorry, don't know about others but I'm not understanding where " list is a const ref" here, where is the const? Please be gentle in your answer ;-)
                                      [Also, did you mean "You can move a const object..." or did you mean "can't"?]

                                      Christian EhrlicherC 1 Reply Last reply
                                      0
                                      • JonBJ JonB

                                        @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                        But the complexity stays, yes.

                                        OK.

                                        @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                        How should this be moved at all when list is a const ref? You can move a const object...

                                        @Christian-Ehrlicher said in Qt 6: Move QList contents without copying?:

                                        But as I said - it can't work the way it is written.

                                        Sorry, don't know about others but I'm not understanding where " list is a const ref" here, where is the const? Please be gentle in your answer ;-)
                                        [Also, did you mean "You can move a const object..." or did you mean "can't"?]

                                        Christian EhrlicherC Offline
                                        Christian EhrlicherC Offline
                                        Christian Ehrlicher
                                        Lifetime Qt Champion
                                        wrote on last edited by Christian Ehrlicher
                                        #19

                                        @JonB said in Qt 6: Move QList contents without copying?:

                                        list is a const ref" here, where is the const? Please be gentle in your answer ;-)

                                        void addList(QList<T>& list) { fullList.append(std::move(list)); }

                                        list is const - so nothing to move here.

                                        Qt Online Installer direct download: https://download.qt.io/official_releases/online_installers/
                                        Visit the Qt Academy at https://academy.qt.io/catalog

                                        Christian EhrlicherC 1 Reply Last reply
                                        0
                                        • Christian EhrlicherC Christian Ehrlicher

                                          @JonB said in Qt 6: Move QList contents without copying?:

                                          list is a const ref" here, where is the const? Please be gentle in your answer ;-)

                                          void addList(QList<T>& list) { fullList.append(std::move(list)); }

                                          list is const - so nothing to move here.

                                          Christian EhrlicherC Offline
                                          Christian EhrlicherC Offline
                                          Christian Ehrlicher
                                          Lifetime Qt Champion
                                          wrote on last edited by
                                          #20

                                          Here I somehow was in the wrong path... sorry for the noise:

                                          class CopyMoveTest
                                          {
                                          public:
                                            CopyMoveTest() { std::cout << "CopyMoveTest: plain ctor\n"; }
                                            CopyMoveTest(const CopyMoveTest &) { std::cout << "CopyMoveTest: copy ctor\n"; }
                                            CopyMoveTest(CopyMoveTest &&) { std::cout << "CopyMoveTest: move ctor\n"; }
                                          
                                            CopyMoveTest operator=(const CopyMoveTest &) { std::cout << "CopyMoveTest: copy operator\n"; return *this; }
                                            CopyMoveTest operator=(const CopyMoveTest &&) { std::cout << "CopyMoveTest: move operator\n"; return *this; }
                                          };
                                          
                                          void refCopy(CopyMoveTest &in)
                                          {
                                            CopyMoveTest c(in);
                                          }
                                          
                                          void rValueRefCopy(CopyMoveTest &&in)
                                          {
                                            CopyMoveTest c(in);
                                          }
                                          
                                          void refMove(CopyMoveTest &in)
                                          {
                                            CopyMoveTest c(std::move(in));
                                          }
                                          
                                          void rValueRefMove(CopyMoveTest &&in)
                                          {
                                            CopyMoveTest c(std::move(in));
                                          }
                                          
                                          int main(int , char **)
                                          {
                                            CopyMoveTest a;
                                            std::cout << "\nref source copy\n";
                                            refCopy(a);
                                            std::cout << "\nref source move\n";
                                            refMove(a);
                                            std::cout << "\nrvalue ref source copy\n";
                                            rValueRefCopy(std::move(a));
                                            std::cout << "\nrvalue ref source move\n";
                                            rValueRefMove(std::move(a));
                                          }
                                          
                                          -->
                                          CopyMoveTest: plain ctor
                                          
                                          ref source copy
                                          CopyMoveTest: copy ctor
                                          
                                          ref source move
                                          CopyMoveTest: move ctor
                                          
                                          rvalue ref source copy
                                          CopyMoveTest: copy ctor
                                          
                                          rvalue ref source move
                                          CopyMoveTest: move ctor
                                          
                                          

                                          Qt Online Installer direct download: https://download.qt.io/official_releases/online_installers/
                                          Visit the Qt Academy at https://academy.qt.io/catalog

                                          1 Reply Last reply
                                          1

                                          • Login

                                          • Login or register to search.
                                          • First post
                                            Last post
                                          0
                                          • Categories
                                          • Recent
                                          • Tags
                                          • Popular
                                          • Users
                                          • Groups
                                          • Search
                                          • Get Qt Extensions
                                          • Unsolved