简述

Qt中关于样式的使用很常见,为了降低耦合性(与逻辑代码分离),我们通常会定义一个QSS文件,然后编写各种部件(例如:QLable、QLineEdit、QPushButton)的样式,最后使用QApplication进行样式加载,这样,就可以让整个应用程序就共享同一个样式。

新建QSS文件

首先,新建一个后缀名为qss的文件,例如:style.qss,将其加入资源文件(qrc)中。

提示:也可以使用绝对路径或相对路径。

编写QSS

在style.qss文件中编写自己的样式代码,例如:

  1. QToolTip {
  2. border: 1px solid rgb(45, 45, 45);
  3. background: white;
  4. color: black;
  5. }

加载QSS

为了方便以后调用,可以写一个静态加载样式的函数:

  1. #include <QFile>
  2. #include <QApplication>
  3. class CommonHelper
  4. {
  5. public:
  6. static void setStyle(const QString &style) {
  7. QFile qss(style);
  8. qss.open(QFile::ReadOnly);
  9. qApp->setStyleSheet(qss.readAll());
  10. qss.close();
  11. }
  12. };

然后,在主函数里进行加载:

  1. int main(int argc, char *argv[])
  2. {
  3. QApplication a(argc, argv);
  4. // 加载QSS样式
  5. CommonHelper::setStyle("style.qss");
  6. MainWindow window;
  7. window.show();
  8. return a.exec();
  9. }

实现原理

很容易发现,原来qApp是QCoreApplication的一个单例,然后,将其转换为QApplication。

  1. #if defined(qApp)
  2. #undef qApp
  3. #endif
  4. #define qApp (static_cast<QApplication *>(QCoreApplication::instance()))

那么,QApplication调用setStyleSheet()以后为何所有的部件样式都改变了呢?

通过逐层分析,我们发现其主要是调用了setStyle():

  1. void QApplication::setStyle(QStyle *style)
  2. {
  3. if (!style || style == QApplicationPrivate::app_style)
  4. return;
  5. QWidgetList all = allWidgets();
  6. // clean up the old style
  7. if (QApplicationPrivate::app_style) {
  8. if (QApplicationPrivate::is_app_running && !QApplicationPrivate::is_app_closing) {
  9. for (QWidgetList::ConstIterator it = all.constBegin(), cend = all.constEnd(); it != cend; ++it) {
  10. QWidget *w = *it;
  11. if (!(w->windowType() == Qt::Desktop) && // except desktop
  12. w->testAttribute(Qt::WA_WState_Polished)) { // has been polished
  13. QApplicationPrivate::app_style->unpolish(w);
  14. }
  15. }
  16. }
  17. QApplicationPrivate::app_style->unpolish(qApp);
  18. }
  19. QStyle *old = QApplicationPrivate::app_style; // save
  20. QApplicationPrivate::overrides_native_style =
  21. nativeStyleClassName() == QByteArray(style->metaObject()->className());
  22. #ifndef QT_NO_STYLE_STYLESHEET
  23. if (!QApplicationPrivate::styleSheet.isEmpty() && !qobject_cast<QStyleSheetStyle *>(style)) {
  24. // we have a stylesheet already and a new style is being set
  25. QStyleSheetStyle *newProxy = new QStyleSheetStyle(style);
  26. style->setParent(newProxy);
  27. QApplicationPrivate::app_style = newProxy;
  28. } else
  29. #endif // QT_NO_STYLE_STYLESHEET
  30. QApplicationPrivate::app_style = style;
  31. QApplicationPrivate::app_style->setParent(qApp); // take ownership
  32. // take care of possible palette requirements of certain gui
  33. // styles. Do it before polishing the application since the style
  34. // might call QApplication::setPalette() itself
  35. if (QApplicationPrivate::set_pal) {
  36. QApplication::setPalette(*QApplicationPrivate::set_pal);
  37. } else if (QApplicationPrivate::sys_pal) {
  38. clearSystemPalette();
  39. initSystemPalette();
  40. QApplicationPrivate::initializeWidgetPaletteHash();
  41. QApplicationPrivate::initializeWidgetFontHash();
  42. QApplicationPrivate::setPalette_helper(*QApplicationPrivate::sys_pal, /*className=*/0, /*clearWidgetPaletteHash=*/false);
  43. } else if (!QApplicationPrivate::sys_pal) {
  44. // Initialize the sys_pal if it hasn't happened yet...
  45. QApplicationPrivate::setSystemPalette(QApplicationPrivate::app_style->standardPalette());
  46. }
  47. // initialize the application with the new style
  48. QApplicationPrivate::app_style->polish(qApp);
  49. // re-polish existing widgets if necessary
  50. if (QApplicationPrivate::is_app_running && !QApplicationPrivate::is_app_closing) {
  51. for (QWidgetList::ConstIterator it = all.constBegin(), cend = all.constEnd(); it != cend; ++it) {
  52. QWidget *w = *it;
  53. if (w->windowType() != Qt::Desktop && w->testAttribute(Qt::WA_WState_Polished)) {
  54. if (w->style() == QApplicationPrivate::app_style)
  55. QApplicationPrivate::app_style->polish(w); // repolish
  56. #ifndef QT_NO_STYLE_STYLESHEET
  57. else
  58. w->setStyleSheet(w->styleSheet()); // touch
  59. #endif
  60. }
  61. }
  62. for (QWidgetList::ConstIterator it = all.constBegin(), cend = all.constEnd(); it != cend; ++it) {
  63. QWidget *w = *it;
  64. if (w->windowType() != Qt::Desktop && !w->testAttribute(Qt::WA_SetStyle)) {
  65. QEvent e(QEvent::StyleChange);
  66. QApplication::sendEvent(w, &e);
  67. w->update();
  68. }
  69. }
  70. }
  71. #ifndef QT_NO_STYLE_STYLESHEET
  72. if (QStyleSheetStyle *oldProxy = qobject_cast<QStyleSheetStyle *>(old)) {
  73. oldProxy->deref();
  74. } else
  75. #endif
  76. if (old && old->parent() == qApp) {
  77. delete old;
  78. }
  79. if (QApplicationPrivate::focus_widget) {
  80. QFocusEvent in(QEvent::FocusIn, Qt::OtherFocusReason);
  81. QApplication::sendEvent(QApplicationPrivate::focus_widget->style(), &in);
  82. QApplicationPrivate::focus_widget->update();
  83. }
  84. }

主要分为4步:

  1. 清理旧样式 - unpolish()
  2. 初始化新样式 - polish()
  3. 加载新样式 - polish() + sendEvent()、update()
  4. 删除旧样式 - delete

通过调用QWidgetList all = allWidgets()获取了所有控件的集合,然后,利用迭代器QWidgetList::ConstIterator对每一个控件进行处理,最后,通过QApplication::sendEvent()来发送QEvent::StyleChange事件,达到全局样式更改。

更多参考

Qt之加载QSS文件的更多相关文章

  1. QT运行时加载UI文件

      写QT程序里运行时加载UI文件,代码如下: 点击(此处)折叠或打开 #include "keyboard.h" #include <QtUiTools> #incl ...

  2. QUiLoader 动态加载.ui文件

    动态加载UI文件是指,用 Qt Designer 通过拖拽的方式生产.ui 文件.不用 uic工具把.ui 文件变成等价的 c++代码,而是在程序运行过程中需要用到UI文件时,用 QUiLoader ...

  3. 详解QUiLoader 动态加载.ui文件

    http://blog.chinaunix.net/uid-13838881-id-3652523.html 1.适用情况: 动态加载UI文件是指,用 Qt Designer 通过拖拽的方式生产.ui ...

  4. webpack模块加载css文件及图片地址

    webpack支持css文件加载并打包,只需安装相应加载器并在配置文件中配置 . 加载的css文件内容会与该模块里的js内容混合封装,这样做的好处是一个js文件包含了所有的css与js内容,有效减少了 ...

  5. 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间

    [源码下载] 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间 作者:webabcd 介绍速战速决 之 PHP 动态地创 ...

  6. Java提高篇——JVM加载class文件的原理机制

    在面试java工程师的时候,这道题经常被问到,故需特别注意. 1.JVM 简介 JVM 是我们Javaer 的最基本功底了,刚开始学Java 的时候,一般都是从“Hello World ”开始的,然后 ...

  7. assets 加载资源文件

    引用:http://abc20899.iteye.com/blog/1096620 1.获取资源的输入流 资源文件 sample.txt 位于 $PROJECT_HOME/assets/ 目录下,可以 ...

  8. 如何在SCENEKIT使用SWIFT RUNTIME动态加载COLLADA文件

    问题:今天接到一个项目,负责弄需求的美眉跟我讲能不能做一个原型能够加载Collada文件,流程如下: 用户用app下载Collada 压缩包(如内购项目) 压缩包解压 展示Collada文件里的内容 ...

  9. 动态加载js文件

    由于最近在弄echarts,关于地图类的效果,但是全国地图整体的js文件太大了,加载很耗费资源,所以要根据不同省份加载不同地区的js地图, 于是就想的比较简单, var script = docume ...

随机推荐

  1. c++ stl string char* 向 string 转换的问题

    请看下面代码 string AddString(const string& a,const string & b) { return a + b; } int _tmain(int a ...

  2. DbUtils使用时抛出Cannot get a connection

    java.sql.SQLException: Cannot get a connection, pool error Timeout waiting for idle object Caused by ...

  3. Lucky String

    Lucky String -- 微软笔试 标签(空格分隔): 算法 A string s is LUCKY if and only if the number of different charact ...

  4. [POJ3667]Hotel(线段树,区间合并)

    题目链接:http://poj.org/problem?id=3667 题意:有一个hotel有n间房子,现在有2种操作: 1 a,check in,表示入住.需要a间连续的房子.返回尽量靠左的房间编 ...

  5. TYPES、DATA、TYPE、LIKE、CONSTANTS、STATICS、TABLES

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  6. 本地设置正常,放服务器上就报 System.Security系统找不到指定的文件解决方法

    在应用程序池设置中将“加载用户配置文件”(Load User Profile)设置为true,问题就解决.

  7. iOS - OC RunLoop 运行循环/消息循环

    1.RunLoop 1)运行循环: 运行循环在 iOS 开发中几乎不用,但是概念的理解却非常重要. 同一个方法中的代码一般都在同一个运行循环中执行,运行循环监听 UI 界面的修改事件,待本次运行循环结 ...

  8. [转载] tmux的使用tips

    原文: http://tangosource.com/blog/a-tmux-crash-course-tips-and-tweaks/

  9. mysql 倒引号

    1.在mysql中,保留字不能作为表名,字段名等用处,如下:mysql> alter table student add column desc varchar(16) after name;1 ...

  10. Android activity之间传值关键性代码

    从当前activity中获取et 表单中的值,并跳转到myactivity.java所绑定的xml布局文件上. private EditText et; Intent intent=new Inten ...