更改单个 QTabWidget 选项卡的颜色

Change color of single QTabWidget tab

我想改变单个标签的颜色,请看下面的截图。 FOO_SUP标签应该是红色的(现在只有按钮是红色的),其他的都不是。

对于文本颜色有 bar->setTabTextColor(index, QColor(Qt::red)),但不是整个选项卡。为选项卡小部件设置选项卡样式表会更改所有选项卡的颜色。

我在这里找到了一个可以更改选项卡颜色的样式表: 但不是针对单个选项卡,我还需要能够在运行时决定选项卡是否为红色。

澄清一下,下面的小部件应保持黑色,标签只有红色。

一种选择是实现您自己的标签栏(如 所述)。

无论如何,我发现使用代理样式更有用且更简洁,因为它允许您部分覆盖 绘画而无需对选项卡栏使用继承。它还允许您轻松地将新样式应用于现有控件。

可以是这样的:

class TabBackgroundProxyStyle : public QProxyStyle {
public:
  TabBackgroundProxyStyle(const QString& base_style_name, const QMap<QString, QBrush>& backgrounds)
    : QProxyStyle(base_style_name),
      m_backgrounds(backgrounds) {
  }

  void drawControl(ControlElement element, const QStyleOption* option, QPainter* painter, const QWidget* widget) const override {
    if (element == CE_TabBarTab) {
      if (auto tab = qstyleoption_cast<const QStyleOptionTab*>(option)) {
        if (m_backgrounds.contains(tab->text)) {
          QStyleOptionTab opt(*tab);
          opt.palette.setBrush(QPalette::Background, m_backgrounds[tab->text]);
          return QProxyStyle::drawControl(element, &opt, painter, widget);
        }
      }
    }
    QProxyStyle::drawControl(element, option, painter, widget);
  }

private:
  const QMap<QString, QBrush> m_backgrounds;
};

要使用它,只需使用适当的选项卡颜色映射创建样式(使用 C++11 的示例):

auto theTabWidget = new QTabWidget();
for (int ii = 0; ii < 10; ++ii) theTabWidget->addTab(new QWidget(), QString("Tab %1").arg(ii + 1));
const QMap<QString, QBrush> backgrounds = {
  {"Tab 2", QBrush(Qt::red)},
  {"Tab 3", QBrush("#c0b050")},
};
theTabWidget->tabBar()->setStyle(new TabBackgroundProxyStyle("", backgrounds));

如果您的用户界面允许选项卡的文本在运行时更改(例如,即时翻译,或者文本是文件名...),那么您必须相应地修改地图。

使用选项卡的标签进行索引是因为样式选项不存储有关选项卡的任何其他直接信息(甚至不包括关联的小部件,因为 QTabBar 只负责呈现选项卡,它不是容器)。

另一种选择是检查选项卡的矩形,对于只有几十个选项卡的选项卡栏来说耗时不多,如果你不想处理标签则更通用:

class TabBackgroundProxyStyle : public QProxyStyle {
public:
  TabBackgroundProxyStyle(const QString& base_style_name, const QMap<int, QBrush>& backgrounds)
    : QProxyStyle(base_style_name),
      m_backgrounds(backgrounds) {
  }

  void drawControl(ControlElement element, const QStyleOption* option, QPainter* painter, const QWidget* widget) const override {
    if (element == CE_TabBarTab) {
      if (auto tab = qstyleoption_cast<const QStyleOptionTab*>(option)) {
        auto tabBar = qobject_cast<const QTabBar*>(widget);
        for (auto index : m_backgrounds.keys()) {
          if (tab->rect == tabBar->tabRect(index)) {
            QStyleOptionTab opt(*tab);
            opt.palette.setBrush(QPalette::Background, m_backgrounds[index]);
            return QProxyStyle::drawControl(element, &opt, painter, widget);
          }
        }
      }
    }
    QProxyStyle::drawControl(element, option, painter, widget);
  }

private:
  const QMap<int, QBrush> m_backgrounds;
};

使用:

auto theTabWidget = new QTabWidget();
for (int ii = 0; ii < 10; ++ii) theTabWidget->addTab(new QWidget(), QString("Tab %1").arg(ii + 1));
const QMap<int, QBrush> backgrounds = {
  {1, QBrush(Qt::red)},
  {4, QBrush("#c0b050")},
};
theTabWidget->tabBar()->setStyle(new TabBackgroundProxyStyle("", backgrounds));

完整的源代码可以从https://github.com/cbuchart/Whosebug/tree/master/54070408-change-color-of-single-qtabwidget-tab

下载

重要:此解决方案的主要缺点是它不能与现有的选项卡样式表很好地混合:您必须 disable/comment QTabBar::tab 为了能够应用样式。