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. How to modify the data in the model and update the view when new data is received from external?
QtWS25 Last Chance

How to modify the data in the model and update the view when new data is received from external?

Scheduled Pinned Locked Moved Solved General and Desktop
76 Posts 4 Posters 50.1k Views
  • 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.
  • ? Offline
    ? Offline
    A Former User
    wrote on last edited by
    #18
    This post is deleted!
    1 Reply Last reply
    0
    • VRoninV Offline
      VRoninV Offline
      VRonin
      wrote on last edited by
      #19

      What I always suggest is that if you are not 100% confident in your model design (and, believe me, it's a hard) just use QAbstractItemModel* model = new QStandardItemModel(parent). Then use insertRows, insertColumns, and setData to setup your model

      "La mort n'est rien, mais vivre vaincu et sans gloire, c'est mourir tous les jours"
      ~Napoleon Bonaparte

      On a crusade to banish setIndexWidget() from the holy land of Qt

      1 Reply Last reply
      0
      • ? Offline
        ? Offline
        A Former User
        wrote on last edited by
        #20

        In my Model.cpp i did the following:

            emit beginResetModel();
            for (auto& category : categories_) {
                category->entryAt(0)->setValue(1);
                // for testing  i just set randomly my data value to 1
            }
            emit endResetModel();
        

        But after above changes the view is not displaying the new values.

        JonBJ 1 Reply Last reply
        0
        • ? A Former User

          In my Model.cpp i did the following:

              emit beginResetModel();
              for (auto& category : categories_) {
                  category->entryAt(0)->setValue(1);
                  // for testing  i just set randomly my data value to 1
              }
              emit endResetModel();
          

          But after above changes the view is not displaying the new values.

          JonBJ Offline
          JonBJ Offline
          JonB
          wrote on last edited by
          #21

          @VInay123
          I don't see any entryAt() function in Qt. So what type is category, what does category->entryAt(0)->setValue(1) do in the way of setting anything in the model??

          ? 1 Reply Last reply
          0
          • JonBJ JonB

            @VInay123
            I don't see any entryAt() function in Qt. So what type is category, what does category->entryAt(0)->setValue(1) do in the way of setting anything in the model??

            ? Offline
            ? Offline
            A Former User
            wrote on last edited by
            #22

            @JonB: If you see my code Category is a class which has entryAt() function which returns back the propertyitem. So here i am setting the model data i.e. my categories_ member variable in the model class.

            JonBJ 1 Reply Last reply
            0
            • ? A Former User

              @JonB: If you see my code Category is a class which has entryAt() function which returns back the propertyitem. So here i am setting the model data i.e. my categories_ member variable in the model class.

              JonBJ Offline
              JonBJ Offline
              JonB
              wrote on last edited by
              #23

              @VInay123
              OK for entryAt(). But where in your code does entryAt(0)->setValue(1) call QAbstractModel::setData() (http://doc.qt.io/qt-5/qabstractitemmodel.html#setData)? Where is the http://doc.qt.io/qt-5/qabstractitemmodel.html#dataChanged signal being emitted?

              ? 1 Reply Last reply
              0
              • JonBJ JonB

                @VInay123
                OK for entryAt(). But where in your code does entryAt(0)->setValue(1) call QAbstractModel::setData() (http://doc.qt.io/qt-5/qabstractitemmodel.html#setData)? Where is the http://doc.qt.io/qt-5/qabstractitemmodel.html#dataChanged signal being emitted?

                ? Offline
                ? Offline
                A Former User
                wrote on last edited by A Former User
                #24

                @JonB: That is the problem i am facing. I do not know how to call setData which in turn emits the signal dataChanged. I do not know how to get the indexes. Here the data is only coming from the "Info_" variable. If the it was via delegates or adapters i could easily get the indexes and call setData. But in my case i get in a structure and go through each item and update its value. Could you please show me how can i achieve this in my code i.e. change the value via setData?

                JonBJ 1 Reply Last reply
                0
                • ? A Former User

                  @JonB: That is the problem i am facing. I do not know how to call setData which in turn emits the signal dataChanged. I do not know how to get the indexes. Here the data is only coming from the "Info_" variable. If the it was via delegates or adapters i could easily get the indexes and call setData. But in my case i get in a structure and go through each item and update its value. Could you please show me how can i achieve this in my code i.e. change the value via setData?

                  JonBJ Offline
                  JonBJ Offline
                  JonB
                  wrote on last edited by
                  #25

                  @VInay123
                  Your code is way too big for me to wade through.

                  • If you are using QAbstractModel, you are using a model which holds data in rows & columns, right?

                  • When you put some data (received from JSON parsing or whatever), you have to decide where in the model's row/column you want to put this data, right?

                  • Given that row & column, you can construct a QModelIndex() (http://doc.qt.io/qt-5/qabstractitemmodel.html#createIndex) for the data.

                  • Then you can pass that to your Model::setData() to set the value, and emit dataChanged(index, index) to notify your view to update.

                  ? 1 Reply Last reply
                  1
                  • JonBJ JonB

                    @VInay123
                    Your code is way too big for me to wade through.

                    • If you are using QAbstractModel, you are using a model which holds data in rows & columns, right?

                    • When you put some data (received from JSON parsing or whatever), you have to decide where in the model's row/column you want to put this data, right?

                    • Given that row & column, you can construct a QModelIndex() (http://doc.qt.io/qt-5/qabstractitemmodel.html#createIndex) for the data.

                    • Then you can pass that to your Model::setData() to set the value, and emit dataChanged(index, index) to notify your view to update.

                    ? Offline
                    ? Offline
                    A Former User
                    wrote on last edited by
                    #26

                    @JonB : Thank you. I implemented as following in my model:

                        QModelIndex index = createIndex(1, 1, categories_[0]->entryAt(0));
                        this->setData(index, 11, Qt::EditRole);
                    

                    It is changing the value clearly. But in the view its not being displayed.

                    JonBJ 1 Reply Last reply
                    0
                    • ? A Former User

                      @JonB : Thank you. I implemented as following in my model:

                          QModelIndex index = createIndex(1, 1, categories_[0]->entryAt(0));
                          this->setData(index, 11, Qt::EditRole);
                      

                      It is changing the value clearly. But in the view its not being displayed.

                      JonBJ Offline
                      JonBJ Offline
                      JonB
                      wrote on last edited by
                      #27

                      @VInay123
                      Well at least we're getting somewhere! Check the return result of your setData(), maybe it's failing? Does (row, column) of (1, 1) exist in your model (else you need to insertRows() etc.)? Is that setData() call indeed going into your Model::setData() and doing the emit dataChanged() there (use a debugger or debug/print statements)? There's no point using both beginResetModel() & dataChanged(), and certainly not dataChanged() while still inside the beginResetModel() before the nedRestModel(). Change your code accordingly.

                      ? 1 Reply Last reply
                      1
                      • VRoninV Offline
                        VRoninV Offline
                        VRonin
                        wrote on last edited by
                        #28

                        @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                        CategoryItem& parentItem = currentItem->parentItem();
                        return createIndex(currentItem->row(), 0, &parentItem);

                        This is using the address of a temp item, it will not work.

                        QString data = file.readAll();

                        does not handle win/lunix line endings and encodings, use QString data = QTextStream(&file).readAll();


                        Use the model test to make sure your subclass works correctly

                        "La mort n'est rien, mais vivre vaincu et sans gloire, c'est mourir tous les jours"
                        ~Napoleon Bonaparte

                        On a crusade to banish setIndexWidget() from the holy land of Qt

                        ? 1 Reply Last reply
                        2
                        • JonBJ JonB

                          @VInay123
                          Well at least we're getting somewhere! Check the return result of your setData(), maybe it's failing? Does (row, column) of (1, 1) exist in your model (else you need to insertRows() etc.)? Is that setData() call indeed going into your Model::setData() and doing the emit dataChanged() there (use a debugger or debug/print statements)? There's no point using both beginResetModel() & dataChanged(), and certainly not dataChanged() while still inside the beginResetModel() before the nedRestModel(). Change your code accordingly.

                          ? Offline
                          ? Offline
                          A Former User
                          wrote on last edited by A Former User
                          #29

                          @JonB: Yes, I just call the setData ane here i have printed the changed:

                              if (role == Qt::EditRole && index.column() == 1 && index.isValid()) {
                                  auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                                  if (currentItem) {
                                      currentItem->setValue(value);
                                      qDebug() << currentItem->data(index.column());
                                      emit dataChanged(index, index);
                                      return true;
                                  }
                              }
                          

                          The value is being changed here. The output looks like

                          QVariant(int, 11)
                          

                          I have included a debug output in Data() but this is printing the default values that is empty("").

                          QVariantModel::data(const QModelIndex& index, int role /*= Qt::DisplayRole*/) const {
                              if (!index.isValid()) {
                                  return QVariant();
                              }
                          
                              if (role == Qt::DisplayRole && isCategory(index)) {
                                  const auto currentCategory = static_cast<CategoryItem*>(index.internalPointer());
                                  if (index.column() == 0) {
                                      return currentCategory->displayName();
                                  } else {
                                      return QVariant();
                                  }
                              }
                          
                              const auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                              if (role == Qt::DisplayRole && !isCategory(index)) {
                                  qDebug() << currentItem->data(index.column()); // Here its empty
                                  return currentItem->data(index.column());
                              }
                              return QVariant();
                          }
                          
                          
                          JonBJ 1 Reply Last reply
                          0
                          • ? A Former User

                            @JonB: Yes, I just call the setData ane here i have printed the changed:

                                if (role == Qt::EditRole && index.column() == 1 && index.isValid()) {
                                    auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                                    if (currentItem) {
                                        currentItem->setValue(value);
                                        qDebug() << currentItem->data(index.column());
                                        emit dataChanged(index, index);
                                        return true;
                                    }
                                }
                            

                            The value is being changed here. The output looks like

                            QVariant(int, 11)
                            

                            I have included a debug output in Data() but this is printing the default values that is empty("").

                            QVariantModel::data(const QModelIndex& index, int role /*= Qt::DisplayRole*/) const {
                                if (!index.isValid()) {
                                    return QVariant();
                                }
                            
                                if (role == Qt::DisplayRole && isCategory(index)) {
                                    const auto currentCategory = static_cast<CategoryItem*>(index.internalPointer());
                                    if (index.column() == 0) {
                                        return currentCategory->displayName();
                                    } else {
                                        return QVariant();
                                    }
                                }
                            
                                const auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                                if (role == Qt::DisplayRole && !isCategory(index)) {
                                    qDebug() << currentItem->data(index.column()); // Here its empty
                                    return currentItem->data(index.column());
                                }
                                return QVariant();
                            }
                            
                            
                            JonBJ Offline
                            JonBJ Offline
                            JonB
                            wrote on last edited by
                            #30

                            @VInay123
                            At this point I don't know any more. Maybe someone else does. We have described how things work. You might like to set up a tiny much simplified example to test how it all works, or look on the web for other examples and start from there.

                            1 Reply Last reply
                            0
                            • VRoninV VRonin

                              @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                              CategoryItem& parentItem = currentItem->parentItem();
                              return createIndex(currentItem->row(), 0, &parentItem);

                              This is using the address of a temp item, it will not work.

                              QString data = file.readAll();

                              does not handle win/lunix line endings and encodings, use QString data = QTextStream(&file).readAll();


                              Use the model test to make sure your subclass works correctly

                              ? Offline
                              ? Offline
                              A Former User
                              wrote on last edited by
                              #31

                              @VRonin: With the test i got an error:

                                // Common error test #2, make sure that a second level index has a parent
                                  // that is the first level index.
                              

                              What does above mean and how does it fit in my code?

                              VRoninV 1 Reply Last reply
                              0
                              • ? A Former User

                                @VRonin: With the test i got an error:

                                  // Common error test #2, make sure that a second level index has a parent
                                    // that is the first level index.
                                

                                What does above mean and how does it fit in my code?

                                VRoninV Offline
                                VRoninV Offline
                                VRonin
                                wrote on last edited by VRonin
                                #32

                                @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                                What does above mean and how does it fit in my code?

                                It means that if model->hasChildren(parent) is true then index(0,0,parent).isValid() must be true and index(0,0,parent).parent()==parent.
                                One of these 2 conditions is violated in your model.

                                Again I strongly suggest you use QStandardItemModel rather than wasting time reimplementing your own

                                "La mort n'est rien, mais vivre vaincu et sans gloire, c'est mourir tous les jours"
                                ~Napoleon Bonaparte

                                On a crusade to banish setIndexWidget() from the holy land of Qt

                                ? 1 Reply Last reply
                                5
                                • VRoninV VRonin

                                  @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                                  What does above mean and how does it fit in my code?

                                  It means that if model->hasChildren(parent) is true then index(0,0,parent).isValid() must be true and index(0,0,parent).parent()==parent.
                                  One of these 2 conditions is violated in your model.

                                  Again I strongly suggest you use QStandardItemModel rather than wasting time reimplementing your own

                                  ? Offline
                                  ? Offline
                                  A Former User
                                  wrote on last edited by A Former User
                                  #33

                                  @VRonin: My class looks good:

                                  QModelIndex Model::parent(const QModelIndex& index) const {
                                      if (!index.isValid()) {
                                          return QModelIndex();
                                      }
                                  
                                      if (isCategory(index)) {
                                          return QModelIndex();
                                      }
                                  
                                      const auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                                      CategoryItem& parentItem = currentItem->parentItem();
                                  
                                      return createIndex(currentItem->row(), 0, &parentItem); // Here the row is always 1.. is it right??
                                  }
                                  
                                  int Model::rowCount(const QModelIndex& parent /*= QModelIndex()*/) const {
                                      if (parent.column() > 0) {
                                          return 0;
                                      }
                                  
                                      if (!parent.isValid()) {
                                          return static_cast<int>(categories_.size());
                                      }
                                  
                                      if (isCategory(parent)) {
                                          const auto currentCategory = static_cast<CategoryItem*>(parent.internalPointer());
                                          return currentCategory->entryCount();
                                      }
                                  
                                      return 0;
                                  }
                                  
                                  int Model::columnCount(const QModelIndex& parent /*= QModelIndex()*/) const {
                                      return 2;
                                  }
                                  

                                  0_1535713717270_Capture.PNG

                                  Coordinates are my categories and S, T are scenariopropertiesitem. Items has category as parents. Categories dont have parents.

                                  VRoninV 1 Reply Last reply
                                  0
                                  • ? A Former User

                                    @VRonin: My class looks good:

                                    QModelIndex Model::parent(const QModelIndex& index) const {
                                        if (!index.isValid()) {
                                            return QModelIndex();
                                        }
                                    
                                        if (isCategory(index)) {
                                            return QModelIndex();
                                        }
                                    
                                        const auto currentItem = static_cast<ScenarioPropertyItem*>(index.internalPointer());
                                        CategoryItem& parentItem = currentItem->parentItem();
                                    
                                        return createIndex(currentItem->row(), 0, &parentItem); // Here the row is always 1.. is it right??
                                    }
                                    
                                    int Model::rowCount(const QModelIndex& parent /*= QModelIndex()*/) const {
                                        if (parent.column() > 0) {
                                            return 0;
                                        }
                                    
                                        if (!parent.isValid()) {
                                            return static_cast<int>(categories_.size());
                                        }
                                    
                                        if (isCategory(parent)) {
                                            const auto currentCategory = static_cast<CategoryItem*>(parent.internalPointer());
                                            return currentCategory->entryCount();
                                        }
                                    
                                        return 0;
                                    }
                                    
                                    int Model::columnCount(const QModelIndex& parent /*= QModelIndex()*/) const {
                                        return 2;
                                    }
                                    

                                    0_1535713717270_Capture.PNG

                                    Coordinates are my categories and S, T are scenariopropertiesitem. Items has category as parents. Categories dont have parents.

                                    VRoninV Offline
                                    VRoninV Offline
                                    VRonin
                                    wrote on last edited by
                                    #34

                                    @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                                    My class looks good:

                                    It doesn't. as I mentioned above

                                    @VRonin said in How to modify the data in the model and update the view when new data is received from external?:

                                    CategoryItem& parentItem = currentItem->parentItem();
                                    return createIndex(currentItem->row(), 0, &parentItem);

                                    This is using the address of a temp item, it will not work.

                                    "La mort n'est rien, mais vivre vaincu et sans gloire, c'est mourir tous les jours"
                                    ~Napoleon Bonaparte

                                    On a crusade to banish setIndexWidget() from the holy land of Qt

                                    ? 1 Reply Last reply
                                    1
                                    • VRoninV VRonin

                                      @VInay123 said in How to modify the data in the model and update the view when new data is received from external?:

                                      My class looks good:

                                      It doesn't. as I mentioned above

                                      @VRonin said in How to modify the data in the model and update the view when new data is received from external?:

                                      CategoryItem& parentItem = currentItem->parentItem();
                                      return createIndex(currentItem->row(), 0, &parentItem);

                                      This is using the address of a temp item, it will not work.

                                      ? Offline
                                      ? Offline
                                      A Former User
                                      wrote on last edited by A Former User
                                      #35

                                      @VRonin: Oh.. How could this be fixed? I have to stick to QAbstractItemModel.
                                      This is from Qt:
                                      0_1535716942348_qq.PNG

                                      1 Reply Last reply
                                      0
                                      • ? Offline
                                        ? Offline
                                        A Former User
                                        wrote on last edited by
                                        #36

                                        My row method was wrong and after updating i have a new failure in the test:

                                                    // Check that we can get back our real parent.
                                                    //qDebug() << "TTT 1: " << model->parent(index);
                                                    //qDebug() << "TTT 2: " << parent;
                                                    //qDebug() << "TTT 3: " << index;
                                                    tmp = model->parent(index);
                                                    Q_ASSERT(tmp == parent);
                                        

                                        At the above point i.e. line 375 its failing.
                                        My old row function in item used to always give 1 which was wrong. Now i made it to look at the parent and find the position.

                                        int ScenarioPropertyItem::row() const {
                                            int count = parentItem_.entryCount();
                                            for (int i = 0; i < count; i++) {
                                                if (this->displayName_ == parentItem_.entryAt(i)->displayName_) {
                                                    return i;
                                                }
                                            }
                                        }
                                        
                                        
                                        VRoninV 1 Reply Last reply
                                        0
                                        • ? A Former User

                                          My row method was wrong and after updating i have a new failure in the test:

                                                      // Check that we can get back our real parent.
                                                      //qDebug() << "TTT 1: " << model->parent(index);
                                                      //qDebug() << "TTT 2: " << parent;
                                                      //qDebug() << "TTT 3: " << index;
                                                      tmp = model->parent(index);
                                                      Q_ASSERT(tmp == parent);
                                          

                                          At the above point i.e. line 375 its failing.
                                          My old row function in item used to always give 1 which was wrong. Now i made it to look at the parent and find the position.

                                          int ScenarioPropertyItem::row() const {
                                              int count = parentItem_.entryCount();
                                              for (int i = 0; i < count; i++) {
                                                  if (this->displayName_ == parentItem_.entryAt(i)->displayName_) {
                                                      return i;
                                                  }
                                              }
                                          }
                                          
                                          
                                          VRoninV Offline
                                          VRoninV Offline
                                          VRonin
                                          wrote on last edited by
                                          #37

                                          @VInay123 How can parentItem_ not be a pointer? if you are taking a copy of the parent and store it in the child, you'll never go back to the old parent

                                          "La mort n'est rien, mais vivre vaincu et sans gloire, c'est mourir tous les jours"
                                          ~Napoleon Bonaparte

                                          On a crusade to banish setIndexWidget() from the holy land of Qt

                                          ? 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