Qt中如何通过QStyle自定义能够跨平台的界面控件

转载 2012年03月23日 11:16:33

我们经常会碰到需要定制界面控件的要求。如果只是在一个平台上,比如说你的控件只需要在Windows上显示,那很好办,Hard code 你的look and feel就可以了。但是如果界面需要在不同平台上显示native的界面风格,比如在Windows上就是Windows的风格,Mac上就是Mac的风格,难道需要为每个平台写不同的代码吗?另外,如果是在Qt这个平台上你需要修改整体的界面风格,同时你的程序也需要跨平台,那你可能需要为每个平台派生该平台下的QStyle对象(Windows上有QWindowsXPStyle,QVistaStyle,Mac上有QMacStyle,等等)。同样修改之后也需要将修改应用到其他平台上。有更好的解决方案吗?

回答是肯定的。 首先你可以参照这篇有点老的文章“http://doc.trolltech.com/qq/qq09-q-and-a.html#style ”,使用Proxy Pattern来实现。创建一个Proxy,然后定义一个指向当前平台的style实例,这样就可以很容易的做一些定制。不过创建Proxy的过程可能有一些繁琐,后期也需要根据QStyle的更新进行维护。Proxy是一个在Qt中使用的非常广泛的pattern,如果你对proxy pattern不熟悉的话,可以去Wiki上看看简要的说明。当然这种方法也存在一些问题。其中最大的一个问题,当调用被代理style的函数时,比如 MyProxyStyle::DrawControl会去调用QWindowXPStyle::DrawControl,但是在DrawControl返回之前,后者永远只会调用QWindowXPStyle里面的函数,那我Proxy中customize的行为是得不到调用的。其实这个问题一直都是存在的,现在依然存在。那如何解决呢?

方案1:依然使用Proxy模式, 比如“这里 ”提供的方法。假设现在的需求是:为所有快捷方式显示下划线。方法如下:

首先创建一个PorxyStyle,不过不是QStyle的子类。是不是认为这样错了呢?怎么可能创建新的Style但是不从QStyle派生,那有多少工作需要做啊?不过请耐心往下看。ProxyStyle的代码如下:

  1. class ProxyStyle 
  2. public
  3.     ProxyStyle(const QString &baseStyle) { style = QStyleFactory::create(baseStyle); } 
  4.     int proxyStyleHint(QStyle::StyleHint hint, const QStyleOption *option = 0, 
  5.                              const QWidget *widget = 0, QStyleHintReturn *returnData = 0) const 
  6.     { 
  7.         if (hint == QStyle::SH_UnderlineShortcut) 
  8.             return 1; 
  9.         return style->styleHint(hint, option, widget, returnData); 
  10.     } 
  11. private
  12.     QStyle *style; 
  13. }; 

然后这么我们真正自定义的Style这样来实现:

  1. #define ADDSTYLESUBCLASS(BaseStyleClass, BaseStyleName)  
  2. class My##BaseStyleClass : public BaseStyleClass, public ProxyStyle  
  3. public
  4.     My##BaseStyleClass() : BaseStyleClass(), ProxyStyle(BaseStyleName) {} 
  5.     int styleHint(StyleHint hint, const QStyleOption *option = 0, const QWidget *widget = 0, QStyleHintReturn *returnData = 0) const 
  6.     { 
  7.         return proxyStyleHint(hint, option, widget, returnData); 
  8.     } 
  9. }; 

这样你可以通过下面的宏来简单的实现对不同平台Sytle的定义,如果下次Windows 7的Style出现自了Qt中,你只需要加上一行代码便能解决所有的问题:

  1. ADDSTYLESUBCLASS(QCleanlooksStyle, "cleanlooks"); 
  2. ADDSTYLESUBCLASS(QPlastiqueStyle, "plastique"); 
  3. ADDSTYLESUBCLASS(QMotifStyle, "motif"); 
  4. ADDSTYLESUBCLASS(QWindowsStyle, "windows"); 
  5. ADDSTYLESUBCLASS(QCDEStyle, "cde"); 

方案2:“动态”继承

动态继承的方法很简单,相信你看到会眼前一亮。主要是利用模板的特性。可以这样来写你的新的Style:

  1. template<typename T> 
  2. class MyQStyle : public
  3. public
  4.     MyQStyle() : T() {} 
  5.     ~MyQStyle() {} 
  6.     virtual void drawItemText(…) { … } 
  7.     virtual int pixelMetric(…) { … } 
  8.     … 
  9. }; 
  10. QStyle* get_application_style(const QString& style_name) 
  11.     if( style_name == “windows”){ 
  12.         return new MyQStyle<QWindowsXPStyle>; 
  13.     } else if( style_name == “mac”){ 
  14.         return new MyQStyle<QMacStyle>; 
  15.     } … 
  16.     cerr ; 

看到了吧,是不是很方便啊?动态继承没有使用proxy,因此减少了很多不必要的代码,但是同时动态继承也有其缺点:它必须通过模板部分特例话的方法来处理不同的QWindowsXPStyle,QMacStyle中不同的函数名带来的编译问题。不过我还是很倾向于动态继承的方法。

QT自定义控件外观(QStyle)

QStyle学习与使用 QStyle 简介 1、创建自定义的QSytyle 静态方式:一般情况下,我们通过子类化一个QStyle,实现它的虚函数来实现自定义行为。或者从头开始自己创建一个QS...

Qt之自定义界面(窗体缩放-跨平台终极版)

来源:Qt之自定义界面(窗体缩放-跨平台终极版)  简述 通过上一节内容,我们实现了窗体的缩放,功能很不错,但是很遗憾-不支持跨平台!如果对于多平台来说,这是一个硬伤,所以...

Qt之自定义界面(窗体缩放-跨平台终极版)

简述通过上一节内容,我们实现了窗体的缩放,功能很不错,但是很遗憾-不支持跨平台!如果对于多平台来说,这是一个硬伤,所以,我们急需要一个能够支持跨平台的实现方式。在网上看到过很多不同的实现方式,多多少少...

【Qt】Qt之自定义界面(窗体缩放-跨平台终极版)【转】

简述 通过上一节内容,我们实现了窗体的缩放,功能很不错,但是很遗憾-不支持跨平台!如果对于多平台来说,这是一个硬伤,所以,我们急需要一个能够支持跨平台的实现方案。 在网上看到过很多不同的实现方式,多多...

自定义QT窗口部件外观之QStyle

自定义QT窗口部件外观 重新定义Qt内置窗口部件的外观常用的方法有两种:一是通过子类化QStyle 类或者预定义的一个样式,例如QWindowStyle,来定制应用程序的观感;二是使用Qt样式表。 ...

Qt 界面使用自定义控件 "提升为"

来源:Qt 界面使用自定义控件 "提升为" 1.效果图 我做了一个非常简单的例子,一个可以显示颜色的QLabel,边上有个按钮,点击,跳出颜...

Qt 界面使用自定义控件 "提升为"

1.效果图我做了一个非常简单的例子,一个可以显示颜色的QLabel,边上有个按钮,点击,跳出颜色选取的Dialog,然后选择一个颜色,这个QLabel会变成什么颜色。2.ColorLabel 我们先把...

Qt下自定义控件的制作与使用

  • 2017年11月03日 13:44
  • 1.12MB
  • 下载

Qt 自定义 Calendar 日历控件

  • 2017年07月31日 16:10
  • 6KB
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Qt中如何通过QStyle自定义能够跨平台的界面控件
举报原因:
原因补充:

(最多只允许输入30个字)