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?
Forum Updated to NodeBB v4.3 + New Features

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.7k Views 1 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.
  • 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
                                        • VRoninV VRonin

                                          @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

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

                                          @VRonin: It is taken as reference to an unique_pointer. Now i have changed it as following:

                                          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);
                                          }
                                          

                                          Please have a look at the index implementation also:

                                          QModelIndex Model::index(int row, int column, const QModelIndex& parent /*= QModelIndex()*/) const {
                                              if (!hasIndex(row, column, parent)) {
                                                  return QModelIndex();
                                              }
                                          
                                              if (!parent.isValid()) {
                                                  return createIndex(row, column, categories_.at(row).get());
                                              }
                                          
                                              const auto currentCategory = static_cast<CategoryItem*>(parent.internalPointer());
                                              ScenarioPropertyItem* currentItem = currentCategory->entryAt(row);
                                          
                                              if (currentItem) {
                                                  return createIndex(row, column, currentItem);
                                              } else {
                                                  return QModelIndex();
                                              }
                                          }
                                          

                                          Tried the test again but its failing at 340 at the same place:
                                          0_1535718574820_qq.PNG

                                          I tried to comment in the debug output and i get the following:
                                          0_1535718651532_qq.PNG

                                          VRoninV 1 Reply Last reply
                                          0

                                          • Login

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