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 can I change QTableView's color of the area that doesn't have cells on it?
Forum Updated to NodeBB v4.3 + New Features

How can I change QTableView's color of the area that doesn't have cells on it?

Scheduled Pinned Locked Moved General and Desktop
36 Posts 8 Posters 30.9k 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.
  • S Offline
    S Offline
    sigrid
    wrote on last edited by
    #10

    I can reproduce this problem here and it looks like a bug. When setting the background-color for the item in addition to the background -color and the alternate-background-color for the table, then the alternate-background-color will be ignored. I suggest you report this issue in "Jira":https://bugreports.qt.nokia.com//secure/Dashboard.jspa.

    1 Reply Last reply
    0
    • G Offline
      G Offline
      giesbert
      wrote on last edited by
      #11

      if you set the color for cells in general and also alternate background color, then the cell wins, that makes sense. and where else will you see the alternate background color?

      Nokia Certified Qt Specialist.
      Programming Is Like Sex: One mistake and you have to support it for the rest of your life. (Michael Sinz)

      1 Reply Last reply
      0
      • E Offline
        E Offline
        Edico
        wrote on last edited by
        #12

        Is there a way to have a color for the outside area of the cells, another for cells and other color for alternate background color?

        1 Reply Last reply
        0
        • G Offline
          G Offline
          giesbert
          wrote on last edited by
          #13

          you will not have alternate color if you set cells color via style sheet. alternate color is a cell color.

          Nokia Certified Qt Specialist.
          Programming Is Like Sex: One mistake and you have to support it for the rest of your life. (Michael Sinz)

          1 Reply Last reply
          0
          • E Offline
            E Offline
            Edico
            wrote on last edited by
            #14

            So, the only way is to subclass QSqlRelationalTableModel (the model I use to access data)? That looks like a big pain.

            1 Reply Last reply
            0
            • G Offline
              G Offline
              giesbert
              wrote on last edited by
              #15

              if you want alternate row colors and a different cell background then empty background? I think yes. Or use a proxy model instead of a subclass, that's much less work.

              Nokia Certified Qt Specialist.
              Programming Is Like Sex: One mistake and you have to support it for the rest of your life. (Michael Sinz)

              1 Reply Last reply
              0
              • E Offline
                E Offline
                Edico
                wrote on last edited by
                #16

                I can't figure how to use a proxy model for this job.

                1 Reply Last reply
                0
                • G Offline
                  G Offline
                  giesbert
                  wrote on last edited by
                  #17

                  if you use a proxy model, the proxy models data method is called for each data query instead of the methods of the original model. This then calls the original (source) model. So you can do everything you want there, like changing backroundColor via the Qt::BackgroundRole. IMHO ProxyModels are easier then derived SQL models ;-)

                  Nokia Certified Qt Specialist.
                  Programming Is Like Sex: One mistake and you have to support it for the rest of your life. (Michael Sinz)

                  1 Reply Last reply
                  0
                  • E Offline
                    E Offline
                    Edico
                    wrote on last edited by
                    #18

                    Gerolf, this mean that I must subclass QSortFilterProxyModel and reimplement data() method?

                    1 Reply Last reply
                    0
                    • O Offline
                      O Offline
                      octal
                      wrote on last edited by
                      #19

                      Yes, you can create your custom ProxyModel by sublcassing QSortFilterProxyModel and reimplementing the data() method. From there, you can do everything you want, like Gerolf said.

                      Finally, you just set the sourceModel of your ProxyModel, and it should work :)

                      1 Reply Last reply
                      0
                      • A Offline
                        A Offline
                        andre
                        wrote on last edited by
                        #20

                        Yes. If you were using 4.8, I would suggest you use -QIdentityModel- [[doc:QIdentityProxyModel]] as your base class, but if you are still on 4.7, use [[doc:QSortFilterProxyModel]] instead. You only need to reimplement the data() method. In that method, you only need to implement the case where the Qt::BackgroundRole is requested. Based on the row number (odd or even), you can return a different color. For all other cases, you get the parent QModelIndex by calling mapToSource() on the model index, and then return the result of calling the data() method on the source model with the same role and the model index you just mapped.

                        1 Reply Last reply
                        0
                        • O Offline
                          O Offline
                          octal
                          wrote on last edited by
                          #21

                          bq. For all other cases, you get the parent QModelIndex by calling mapToSource() on the model index, and then return the result of calling the data() method on the source model with the same role and the model index you just mapped.

                          Or just calling the base class method ? Like :

                          @
                          QVariant MyProxyModel::data(const QModelIndex &index, int role)
                          {
                          if (role == Qt::BackgroundRole)
                          {
                          return ...
                          }

                          return QSortFilterProxyModel::data(index, role);
                          

                          @

                          bq. If you were using 4.8, I would suggest you use QIdentityModel as your base class

                          I made a research and I guess you meant "QIdentityProxyModel":http://doc.trolltech.com/4.8-snapshot/qidentityproxymodel.html

                          Interesting :)

                          1 Reply Last reply
                          0
                          • A Offline
                            A Offline
                            andre
                            wrote on last edited by
                            #22

                            Yes, I guess you can also call the base class implementation. That does the same:

                            @
                            Q_D(const QSortFilterProxyModel);
                            QModelIndex source_index = mapToSource(index);
                            if (index.isValid() && !source_index.isValid())
                            return QVariant();
                            return d->model->data(source_index, role);
                            @

                            I guess it does not matter much.

                            1 Reply Last reply
                            0
                            • E Offline
                              E Offline
                              Edico
                              wrote on last edited by
                              #23

                              I've set the outside area of the cells like that:

                              @
                              QPalette p = view->palette();
                              QColor color = p.color(QPalette::Window);
                              p.setColor(QPalette::Base, color);
                              view->setPalette(p);
                              @

                              Then data() in MyProxyModel:

                              @
                              QVariant MyProxyModel::data(const QModelIndex &index, int role) const
                              {
                              if (!index.isValid())
                              return QVariant::Invalid;

                              int row = index.row();
                              
                              QWidget widget;  // I didn't found a wiser method to obtain the palette colors than creating a QWidget inside the data() method
                              QColor alternateColor1 = widget.palette().color(QPalette::AlternateBase);
                              QColor alternateColor2 = widget.palette().color(QPalette::Base);
                              
                              if (role == Qt::BackgroundRole) {
                                  QBrush rowBackground;
                                  if (row % 2 == 0)
                                      rowBackground = QBrush(alternateColor1);
                                  else if (row % 2 != 0)
                                      rowBackground = QBrush(alternateColor2);
                                  return rowBackground;
                              }
                              return QSortFilterProxyModel::data(index, role);
                              

                              }
                              @

                              It does what I wanted.
                              Thank you all for the help.

                              1 Reply Last reply
                              0
                              • A Offline
                                A Offline
                                andre
                                wrote on last edited by
                                #24

                                The data method is called very frequently. It is not a good idea to create a widget in this method. Instead, why not just create a pair of getters and setter methods for the colors on the proxy model. Set them once when you create the proxy, and use them in your data method. That will be much more performant.

                                1 Reply Last reply
                                0
                                • E Offline
                                  E Offline
                                  Edico
                                  wrote on last edited by
                                  #25

                                  I know it's a little bit outside the subject but, how is much better:
                                  to have setters like this:

                                  @
                                  void MyProxyModel::setAlternateColor1()
                                  {
                                  QWidget widget;
                                  alternateColor1 = widget.palette().color(QPalette::Base);
                                  }

                                  void MyProxyModel::setAlternateColor2()
                                  {
                                  QWidget widget;
                                  alternateColor2 = widget.palette().color(QPalette::AlternateBase);
                                  }
                                  @

                                  and data() calling the getters,
                                  or setting the colors in the constructor like this:

                                  @
                                  MyProxyModel::MyProxyModel()
                                  {
                                  QWidget widget;
                                  alternateColor1 = widget.palette().color(QPalette::Base);
                                  alternateColor2 = widget.palette().color(QPalette::AlternateBase);
                                  }
                                  @

                                  and data using directly QColor alternateColor1 and QColor alternateColor2?

                                  1 Reply Last reply
                                  0
                                  • O Offline
                                    O Offline
                                    octal
                                    wrote on last edited by
                                    #26

                                    I think you misunterstood what Andre meant.

                                    bq. Instead, why not just create a pair of getters and setter methods for the colors on the proxy model.

                                    A pair of getters/setters would be :

                                    @
                                    void MyProxyModel::setAlternateColor1(const QColor &color)
                                    {
                                    mAlternateColor1 = color;
                                    }

                                    QColor MyProxyModel::alternateColor1() const
                                    {
                                    return mAlternateColor1;
                                    }
                                    @

                                    1 Reply Last reply
                                    0
                                    • A Offline
                                      A Offline
                                      andre
                                      wrote on last edited by
                                      #27

                                      Octal:
                                      Indeed, that's what I meant, and also a getter and a setter for the primary color. You can set them at initialization of your proxy model, which is very likely to be in the constructor of a widget class. Hence, you have cheap access to a widget pointer and the palette there, and you only need that access once.

                                      1 Reply Last reply
                                      0
                                      • E Offline
                                        E Offline
                                        Edico
                                        wrote on last edited by
                                        #28

                                        I've created setters and getters. Now I use the proxy like that:

                                        @
                                        proxyModel = new MyProxyModel;
                                        QColor altColor1 = this->palette().color(QPalette::Base);
                                        QColor altColor2 = this->palette().color(QPalette::AlternateBase);
                                        proxyModel->setAlternateColor1(altColor1);
                                        proxyModel->setAlternateColor2(altColor2);
                                        proxyModel->setSourceModel(model);
                                        @

                                        proxyModel's data() method calls the getters.
                                        Is it better like that?

                                        Andre, what do you mean with primary color?

                                        1 Reply Last reply
                                        0
                                        • A Offline
                                          A Offline
                                          andre
                                          wrote on last edited by
                                          #29

                                          As long as you use the colors you set there in your data() method: much better :-)

                                          One small tip: you will want to give your MyProxyModel object a parent object. Seeing that this proxy is really doing presentation work, I suggest you use the view that it is for as the parent object. Also: a more descriptive name than "MyProxyModel" would probably work better in the long run.

                                          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