Как реализовать вертикальные вкладки в QT?

Я пытаюсь реализовать вертикальные вкладки с горизонтальным текстом с помощью QT, но не могу найти аналогичную опцию в QTabWidget.

Кто-то в SO попросил нечто подобное здесь, однако , ответы содержат неработающие ссылки, и я сомневаюсь, что они представляют собой реальное решение.

Кто-нибудь смог это сделать?


person Bub Espinja    schedule 29.05.2018    source источник


Ответы (3)


Вы должны реализовать пользовательский метод QTabBar, перезаписывающий методы tabSizeHint() и paintEvent(), как показано ниже:

#include <QApplication>
#include <QStyleOptionTab>
#include <QStylePainter>
#include <QTabBar>
#include <QTabWidget>

class TabBar: public QTabBar{
public:
    QSize tabSizeHint(int index) const{
        QSize s = QTabBar::tabSizeHint(index);
        s.transpose();
        return s;
    }
protected:
    void paintEvent(QPaintEvent * /*event*/){
        QStylePainter painter(this);
        QStyleOptionTab opt;

        for(int i = 0;i < count();i++)
        {
            initStyleOption(&opt,i);
            painter.drawControl(QStyle::CE_TabBarTabShape, opt);
            painter.save();

            QSize s = opt.rect.size();
            s.transpose();
            QRect r(QPoint(), s);
            r.moveCenter(opt.rect.center());
            opt.rect = r;

            QPoint c = tabRect(i).center();
            painter.translate(c);
            painter.rotate(90);
            painter.translate(-c);
            painter.drawControl(QStyle::CE_TabBarTabLabel,opt);
            painter.restore();
        }
    }
};

class TabWidget : public QTabWidget
{
public:
    TabWidget(QWidget *parent=0):QTabWidget(parent){
        setTabBar(new TabBar);
        setTabPosition(QTabWidget::West);
    }
};

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    TabWidget w;
    w.addTab(new QWidget, "tab1");
    w.addTab(new QWidget, "tab2");
    w.addTab(new QWidget, "tab3");
    w.show();

    return a.exec();
}

введите описание изображения здесь

person eyllanesc    schedule 29.05.2018
comment
Привет! Я попробовал ваш код и получил: ibb.co/dVuuYd Затем я добавил this->setStyleSheet("QTabBar::tab {width: 115px; height: 30px; color: #000000; font-weight: bold; font-size: 10px;");, чтобы сделать его больше. Я получаю текст вертикальным: ibb.co/gJqMeJ Как вы устанавливаете размер вкладки, поэтому текст горизонтальный? Спасибо. - person Cobra91151; 22.07.2018
comment
@Cobra91151 используйте "QTabBar::tab {height: 115px; width: 30px; color: #000000; font-weight: bold; font-size: 10px;}" - person eyllanesc; 22.07.2018
comment
Да, я понял. Я изменил код this->setStyleSheet("QTabBar::tab {height: 115px; width: 30px; color: #000000; font-weight: bold; font-size: 10px;}");, но теперь получаю: ibb.co/nGZWwy - person Cobra91151; 22.07.2018
comment
Я исправил это, изменив: QTabBar::paintEvent(event); на QWidget::paintEvent(event); после кода метода paintEvent. Теперь все работает: ibb.co/j8rN3d Спасибо. - person Cobra91151; 22.07.2018
comment
Это был мой код, смешанный с вашим. Теперь проблема решена. Спасибо. - person Cobra91151; 22.07.2018
comment
Меня интересует последовательность перевода/поворота/перевода во время paintEvent. Я получаю вращение, но зачем двигаться и двигаться назад? - person anonymous; 24.07.2018
comment
@JonHarper, когда вы поворачиваете то, что вы делаете, относительно источника, по этой причине я перемещаюсь в центр прямоугольника, сломанной вещи, но затем мне приходится рисовать его из исходного источника, так как это ожидает функцию drawControl, я рекомендую что вы экспериментируете, удаляя эту линию. - person eyllanesc; 24.07.2018
comment
Оооо. Моя декартова математика нуждалась в обновлении. Спасибо. - person anonymous; 24.07.2018
comment
@JonHarper LOL, это немного унылая и скучная математика :) - person eyllanesc; 24.07.2018
comment
Привет! Это работает хорошо, но когда я применил эту таблицу стилей: "QTabBar::tab:selected {background-color: #FA9944; color: #000000; border-top: 1px solid #FA9944;} QTabBar::tab:hover {color: #000000; border-top: 1px solid #FA9944; background-color: #FFFFFF;}" Изображения: ibb.co/ezSb6K ibb.co/jstutz Он неправильно оформляет вкладки. Должен ли я повторно реализовать событие для прессы и событие наведения? Какие-либо предложения? Спасибо. - person Cobra91151; 05.08.2018
comment
@Cobra91151, вы могли бы разместить изображение лучшего качества, чтобы понять вас - person eyllanesc; 05.08.2018
comment
@ Cobra91151 Не могли бы вы сказать мне, в чем неудобство, с одними изображениями я не мог этого понять. - person eyllanesc; 05.08.2018
comment
Проблема в том, что он рисует только часть вкладки, а не всю вкладку, когда вкладка выбрана или наведена. - person Cobra91151; 05.08.2018
comment
Я создал изображение проблемы в формате gif: ddgobkiprc33d.cloudfront.net/ Есть идеи, как это исправить? Спасибо. - person Cobra91151; 05.08.2018
comment
Я исправил проблему. Я опубликую это в ответ. - person Cobra91151; 06.08.2018
comment
@ Cobra91151 Cobra91151 Единственное отличие, которое я вижу, это: QWidget::paintEvent(event);, С другой стороны, мой исходный код не вызывает у меня проблем, какой стиль вы используете? - person eyllanesc; 06.08.2018
comment
@eyllanesc На самом деле разница заключается в методе tabSizeHint, который устанавливает размер вкладок. Ваш код работает только для размера по умолчанию. - person Cobra91151; 06.08.2018
comment
@ Cobra91151 ммм, я ожидал более чистого решения, реализация его на C++ делает менее гибким для qss. - person eyllanesc; 06.08.2018
comment
@eyllanesc Кстати, ваше решение работает только для позиции табуляции QTabWidget::West. Если вы хотите QTabWidget::East положение табуляции, вы должны изменить painter.rotate(270); иначе текст будет перевернутым. В своем проекте я сохранил позицию и добавил дополнительную проверку в метод paintEvent. - person Cobra91151; 06.08.2018
comment
@ Cobra91151, Да, потому что этого требовал от меня ОП, и как вы видите, изменить это просто. :) - person eyllanesc; 06.08.2018

Итак, мое решение исправить эту проблему со стилем:

Код:

apptabbar.h

#ifndef APPTABBAR_H
#define APPTABBAR_H

#include <QTabBar>
#include <QStylePainter>
#include <QStyleOptionTab>

class AppTabBar : public QTabBar
{
public:
    AppTabBar();
    AppTabBar(int tabWidth, int tabHeight);
    AppTabBar(QSize tabSize);
    AppTabBar(QWidget *parent);
    AppTabBar(QWidget *parent, int tabWidth, int tabHeight);
    AppTabBar(QWidget *parent, QSize tabSize);
    QSize tabSizeHint(int index) const override;
    ~AppTabBar();

protected:
    void paintEvent(QPaintEvent *event) override;

private:
    int width, height;
};

#endif // APPTABBAR_H

apptabbar.cpp

#include "apptabbar.h"

AppTabBar::AppTabBar() : QTabBar(),
width(30),
height(115)
{

}

AppTabBar::AppTabBar(int tabWidth, int tabHeight) : QTabBar(),
width(tabWidth),
height(tabHeight)
{

}

AppTabBar::AppTabBar(QSize tabSize) : QTabBar(),
width(tabSize.width()),
height(tabSize.height())
{

}

AppTabBar::AppTabBar(QWidget *parent) : QTabBar(parent),
width(30),
height(115)
{

}

AppTabBar::AppTabBar(QWidget *parent, int tabWidth, int tabHeight) : QTabBar(parent),
width(tabWidth),
height(tabHeight)
{

}

AppTabBar::AppTabBar(QWidget *parent, QSize tabSize) : QTabBar(parent),
width(tabSize.width()),
height(tabSize.height())
{

}

AppTabBar::~AppTabBar()
{

}

QSize AppTabBar::tabSizeHint(int index) const
{
    QSize s = QTabBar::tabSizeHint(index);
    s.setWidth(width);
    s.setHeight(height);
    s.transpose();
    return s;
}

void AppTabBar::paintEvent(QPaintEvent *event)
{
    QStylePainter painter(this);
    QStyleOptionTab opt;

    for (int i = 0; i < this->count(); i++) {
        initStyleOption(&opt, i);
        painter.drawControl(QStyle::CE_TabBarTabShape, opt);
        painter.save();

        QSize s = opt.rect.size();
        s.transpose();
        QRect r(QPoint(), s);
        r.moveCenter(opt.rect.center());
        opt.rect = r;

        QPoint c = tabRect(i).center();
        painter.translate(c);
        painter.rotate(90);
        painter.translate(-c);
        painter.drawControl(QStyle::CE_TabBarTabLabel, opt);
        painter.restore();
    }

    QWidget::paintEvent(event);
}

apptabcontrol.h

#ifndef APPTABCONTROL_H
#define APPTABCONTROL_H

#include <QTabWidget>
#include <QTabBar>
#include <QPalette>
#include <QPainter>
#include <QDebug>
#include "apptabbar.h"

class AppTabControl : public QTabWidget
{
public:
    AppTabControl();
    AppTabControl(QWidget *parent);
    AppTabControl(QWidget *parent, int width, int height);
    AppTabControl(QWidget *parent, QSize size);
    void setTabControlSize(int width, int height);
    void setTabControlSize(QSize tabSize);
    ~AppTabControl();

private:
    int tabWidth, tabHeight;
    QSize tabSize;
};

#endif // APPTABCONTROL_H

apptabcontrol.cpp

#include "apptabcontrol.h"

AppTabControl::AppTabControl()
{

}

AppTabControl::AppTabControl(QWidget *parent) : QTabWidget(parent)
{
    this->setTabBar(new AppTabBar(parent, tabWidth, tabHeight));
    this->setTabPosition(QTabWidget::West);
}

AppTabControl::AppTabControl(QWidget *parent, int width, int height) : QTabWidget(parent),
tabWidth(width),
tabHeight(height)
{
    this->setTabBar(new AppTabBar(parent, tabWidth, tabHeight));
    this->setTabPosition(QTabWidget::West);
}

AppTabControl::AppTabControl(QWidget *parent, QSize size) : QTabWidget(parent),
tabSize(size)
{
    this->setTabBar(new AppTabBar(parent, tabSize));
    this->setTabPosition(QTabWidget::West);
}

void AppTabControl::setTabControlSize(int width, int height)
{
    tabWidth = width;
    tabHeight = height;
}

void AppTabControl::setTabControlSize(QSize size)
{
    tabSize = size;
}

AppTabControl::~AppTabControl()
{

}

А теперь заключительная часть:

Test::Test(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Test)
{
    AppTabControl *tabControl = new AppTabControl(this, 30, 115);
    this->setStyleSheet("QTabBar::tab {color: #000000; font-weight: bold; font-size: 10px; font-family: Gotham, Helvetica Neue, Helvetica, Arial, sans-serif;} "
   "QTabBar::tab:selected {background-color: #FA9944; color: #000000; border-top: 1px solid #FA9944;} "
   "QTabBar::tab:hover {color: #000000; border-top: 1px solid #FA9944; background-color: #FFFFFF;}");
    AppTabBar *tabBar1 = new AppTabBar(tabControl);
    AppTabBar *tabBar2 = new AppTabBar(tabControl);
    AppTabBar *tabBar3 = new AppTabBar(tabControl);
    tabControl->addTab(tabBar1, "Tab1");
    tabControl->addTab(tabBar2, "Tab2");
    tabControl->addTab(tabBar3, "Tab3")
}

Результат:

0_1533540265396_2018-08-06_102357.png

0_1533540726975_tabs_solution.gif

Теперь все работает хорошо.

person Cobra91151    schedule 06.08.2018

Решение для PyQt5

Для PyQt5 решение такое (переведенный ответ @ellyanesc с C++ на Python):

from __future__ import annotations
from typing import *
import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *

class VTabBar(QTabBar):
    def __init__(self, *args, **kwargs) -> None:
        super().__init__(*args, **kwargs)
        return

    def tabSizeHint(self, index:int) -> QSize:
        s = super().tabSizeHint(index)
        s.transpose()
        return s

    def paintEvent(self, event:QPaintEvent) -> None:
        painter:QStylePainter = QStylePainter(self)
        opt:QStyleOptionTab = QStyleOptionTab()
        for i in range(self.count()):
            self.initStyleOption(opt, i)
            painter.drawControl(QStyle.CE_TabBarTabShape, opt)
            painter.save()

            s:QSize = opt.rect.size()
            s.transpose()
            r:QRect = QRect(QPoint(), s)
            r.moveCenter(opt.rect.center())
            opt.rect = r

            c:QPoint = self.tabRect(i).center()
            painter.translate(c)
            painter.rotate(90)
            painter.translate(-c)
            painter.drawControl(QStyle.CE_TabBarTabLabel, opt)
            painter.restore()
        return

class VTabWidget(QTabWidget):
    def __init__(self, parent:QWidget=None) -> None:
        super().__init__(parent)
        self.setTabBar(VTabBar())
        self.setTabPosition(QTabWidget.West)
        return

if __name__ == '__main__':
    app = QApplication(sys.argv)
    QApplication.setStyle(QStyleFactory.create('Plastique'))
    w = VTabWidget()
    w.addTab(QWidget(), 'tab1')
    w.addTab(QWidget(), 'tab2')
    w.addTab(QWidget(), 'tab3')
    w.show()
    sys.exit(app.exec_())
person K.Mulier    schedule 03.06.2021