AC
Alexandru CodreanuApril 5, 2020, 9:04 a.m.

Qt/C++ особенности QProxyStyle

Доброго дня.
Столкнулся с необходимостью создать на форме TabWidget вкладки которых должны быть по вертикали,
но с заголовками по горизонтали.

Реализовал при помощи QProxyStyle
listclassifier.h

#ifndef LISTCLASSIFIER_H
#define LISTCLASSIFIER_H

#include <QDialog>
#include <QProxyStyle>
#include <QStyleOptionTab>

//===================================================
namespace Ui {
class ListClassifier;
}
//===================================================
class CustomTabProxyStyle : public QProxyStyle {
public:
  virtual QSize sizeFromContents(QStyle::ContentsType ctype, const QStyleOption *option, const QSize &contentsSize, const QWidget *widget) const override
  {
      QSize s = QProxyStyle::sizeFromContents(ctype, option, contentsSize, widget);
      if (ctype == QStyle::CT_TabBarTab){
          s.transpose();
      }
      return s;
  }
  virtual void drawControl(QStyle::ControlElement element, const QStyleOption *option, QPainter *painter, const QWidget *widget) const override
  {
      if (element == CE_TabBarTabLabel) {
          if (const QStyleOptionTab* tab = qstyleoption_cast<const QStyleOptionTab*>(option)) {
              QStyleOptionTab opt(*tab);
              opt.shape = QTabBar::RoundedNorth;
              QProxyStyle::drawControl(element, &opt, painter, widget);
              return;
          }
      }
      QProxyStyle::drawControl(element, option, painter, widget);
  }
};
//===================================================
class ListClassifier : public QDialog
{
    Q_OBJECT

public:
    explicit ListClassifier(QWidget *parent = nullptr);
    ~ListClassifier();

private:
    Ui::ListClassifier *ui;
};

#endif // LISTCLASSIFIER_H

listclassifier.cpp

#include "listclassifier.h"
#include "ui_listclassifier.h"

ListClassifier::ListClassifier(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::ListClassifier)
{
    ui->setupUi(this);

    QTabWidget* theTabWidget = new QTabWidget();
    theTabWidget->tabBar()->setStyle(new CustomTabProxyStyle());
    for (int i = 0; i < 5; ++i)
        theTabWidget->addTab(new QWidget(), QString("Tab %1").arg(i + 1));
    theTabWidget->setTabPosition(QTabWidget::West);

    this->layout()->addWidget(theTabWidget);
}

... но возникла проблема.
Суть проблемы заключается в том, что при загрузке приложения устанавливается стиль в main.cpp (код касающийся QTabBar)

a.setStyleSheet(QString(""
          "/* --------------------- QTabBar --------------------- */"
          "QTabBar::tab"
          "{"
          "background: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,stop: 0 #E1E1E1, stop: 0.4 #DDDDDD,stop: 0.5 #D8D8D8, stop: 1.0 #D3D3D3);"
          "border: 2px solid #C4C4C3;" //1px inset gray;
          "border-bottom-color: #C2C7CB;"
          "border-top-left-radius: 5px; "
          "border-top-right-radius: 5px; "
          "min-width: 10ex; "
          "padding-top: 1px;"
          "padding-right: 8px;"
          "padding-left: 8px;"
          "padding-bottom: 1px;"
          "}"
          ""
          "QTabBar::tab:selected, QTabBar::tab:hover"
          "{"
          "background: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,stop: 0 #fafafa, stop: 0.4 #f4f4f4,stop: 0.5 #e7e7e7, stop: 1.0 #fafafa);"
          "}"
          ""
          "QTabBar::tab:selected"
          "{"
          "border-color: #9B9B9B;"
          "border-bottom-color: #C2C7CB;"
          "}"
          ""
          "QTabBar::tab:!selected"
          "{"
          "margin-top: 2px;"
          "}"
          ""));

Заголовки вкладок устанавливаются по горизонтали только в том случае если код вышеуказанного стиля закомментирован .
Хотя в офф.документации описано:

... динамическое переопределение элементов QStyle (если правильно понял).

Подскажите как быть в данной ситуации ?

We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.
4

Добрый день.
Это всё работает, вот только если есть переопределения каких-то методов, то нет гарантии, что что-то не сломается.
Думаю, что нужно допиливать стилевоё оформление в прокси классе.

AC

Добрый день. Объясните пожалуйста ...

... допиливать стилевое оформление в прокси классе

... где именно копать ?
В каком виртуальном методе лучше допиливать (если можно направление или пример) ?

Разбираться с методом drawControl, понимаю, что вы его просто скопипастили с одного из сайтов, на котором расположен этот пример кода, он кочует с сайта на сайт уже давно. Но по сути вопрос в том, чтобы правильно разобраться со всеми деталями, которые есть в этом методе.

На QStyleOptionTab есть документация, в которой моджно посмотреть, какие именно стилевые параметры можно настроить и их нужно переписать в методе drawControl

AC

Да, Вы правы. Код был скопирован с сайта (уже не помню с какого), но решил пойти по пути более легком.
Пришлось переписать - кому интересно:
использовал stackedWidget для переключения между закладками в итоги получилось следующее ...

Comments

Only authorized users can post comments.
Please, Log in or Sign up
Timeweb

Let me recommend you the excellent hosting on which EVILEG is located.

For many years, Timeweb has been proving his stability.

For projects on Django I recommend VDS hosting

View Hosting
DK

C ++ - Test 004. Pointers, Arrays and Loops

  • Result:70points,
  • Rating points1
Ke

C++ - Тест 003. Условия и циклы

  • Result:71points,
  • Rating points1
MZ

C++ - Тест 003. Условия и циклы

  • Result:64points,
  • Rating points-1
Last comments
s

Qt - Selecting data from a QSqlQuery database in a QThread stream and creating a QAbstractTableModel model based on it

Кстати сегодня почему-то все заработало :)
s

Qt - Selecting data from a QSqlQuery database in a QThread stream and creating a QAbstractTableModel model based on it

Я наверное слишком туп, можете пример привести как Вы это делаете?

Qt - Selecting data from a QSqlQuery database in a QThread stream and creating a QAbstractTableModel model based on it

Соединение в потоке используется только в потоке. Т.е выбирает данные сохраняете в какой-нибудь контейнер и передаете его основному потоку.
s

Qt - Selecting data from a QSqlQuery database in a QThread stream and creating a QAbstractTableModel model based on it

Перенес в класс потока все функции для работы с БД, но все по старому когда закрываю поток основное соединение тоже закрывается
Now discuss on the forum

QTabWidget

Здравствуйте, возник вопрос можно ли в QTabWidget удалить 2 кладки одновременно, если одна из них выбрана пользователем(currentWidget()) а про вторую известно tabWidget->widget(i) и ее инд…
E

Qml, tableview

это я понимаю, проблема в том, что при изменении данных в модели, само содержание ячейки меняется, а вот метод для ее окраски в зависимости от содержания, не вызывается

Связь таблиц Qt SQL

добрый, у вас по сути два варианта: 1. делать каскадное удаление из БД по внешним ключам(но если архетиктура базы с ошибками то могут быть сбои) 2. делать контролируемо удаление из вто…
A.

Работа с WinAPI в QT(изменение title bar)

void MainWindow::mousePressEvent(QMouseEvent *event){ if(event->pos().y() <= 45 && event->pos().y() >= 16) mpos = event->pos(); else if(event->pos().y…

QSqlTableModel - Как добавить картинки в таблицу, чтобы они отражались в диалоговом окне, но не были частью модели

Ну тогда в этом столбце указывайте пути на несколько картинок
About
Services
© EVILEG 2015-2020
Recommend hosting TIMEWEB