Qt中如果通过QStyle自定义能够跨平台的界面控件
我们经常会碰到需要定制界面控件的要求。如果只是在一个平台上,比如说你的控件只需要在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的代码如下:
- class ProxyStyle
- {
- public:
- ProxyStyle(const QString &baseStyle) { style = QStyleFactory::create(baseStyle); }
- int proxyStyleHint(QStyle::StyleHint hint, const QStyleOption *option = 0,
- const QWidget *widget = 0, QStyleHintReturn *returnData = 0) const
- {
- if (hint == QStyle::SH_UnderlineShortcut)
- return 1;
- return style->styleHint(hint, option, widget, returnData);
- }
- private:
- QStyle *style;
- };
然后这么我们真正自定义的Style这样来实现:
- #define ADDSTYLESUBCLASS(BaseStyleClass, BaseStyleName)
- class My##BaseStyleClass : public BaseStyleClass, public ProxyStyle
- {
- public:
- My##BaseStyleClass() : BaseStyleClass(), ProxyStyle(BaseStyleName) {}
- int styleHint(StyleHint hint, const QStyleOption *option = 0, const QWidget *widget = 0, QStyleHintReturn *returnData = 0) const
- {
- return proxyStyleHint(hint, option, widget, returnData);
- }
- };
这样你可以通过下面的宏来简单的实现对不同平台Sytle的定义,如果下次Windows 7的Style出现自了Qt中,你只需要加上一行代码便能解决所有的问题:
- ADDSTYLESUBCLASS(QCleanlooksStyle, "cleanlooks");
- ADDSTYLESUBCLASS(QPlastiqueStyle, "plastique");
- ADDSTYLESUBCLASS(QMotifStyle, "motif");
- ADDSTYLESUBCLASS(QWindowsStyle, "windows");
- ADDSTYLESUBCLASS(QCDEStyle, "cde");
方案2:“动态”继承
动态继承的方法很简单,相信你看到会眼前一亮。主要是利用模板的特性。可以这样来写你的新的Style:
- template<typename T>
- class MyQStyle : public T
- {
- public:
- MyQStyle() : T() {}
- ~MyQStyle() {}
- virtual void drawItemText(…) { … }
- virtual int pixelMetric(…) { … }
- …
- };
- QStyle* get_application_style(const QString& style_name)
- {
- if( style_name == “windows”){
- return new MyQStyle<QWindowsXPStyle>;
- } else if( style_name == “mac”){
- return new MyQStyle<QMacStyle>;
- } …
- cerr ;
- }
看到了吧,是不是很方便啊?动态继承没有使用proxy,因此减少了很多不必要的代码,但是同时动态继承也有其缺点:它必须通过模板部分特例话的方 法来处理不同的QWindowsXPStyle,QMacStyle中不同的函数名带来的编译问题。不过我还是很倾向于动态继承的方法。
http://www.cnblogs.com/SkylineSoft/articles/2046272.html
Qt中如果通过QStyle自定义能够跨平台的界面控件的更多相关文章
- 【C#】自定义容器控件,设置界面控件,支持设计器拖入控件
先上效果图: 1.先重写设置界面的控件功能: public partial class SetterControl : UserControl { public SetterControl() { I ...
- MFC中 自定义类访问主对话框控件的方法
之前一直在找有木有好点的方法.现在终于被我找到,收藏之~~~~~~ 在使用mfc的时候经常遇到自定义类访问主对话框控件的问题,例如自定义类中的方法要输出一段字符串到主对话框的EDIT控件.控制对话框的 ...
- Android 自定义View之自绘控件
首先要提前声明一下,我对于自定义View的理解并不是很深,最近啃了几天guolin博主写的关于自定义View的博客,讲的非常棒,只不过涉及到源码和底层的一些东西,我自己就懵逼了,目前只是会了关于自定义 ...
- [Swift通天遁地]九、拔剑吧-(3)创建多种自定义Segment分段样式的控件
★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...
- asp.net mvc中使用jquery H5省市县三级地区选择控件
地区选择是项目开发中常用的操作,本文讲的控件是在手机端使用的选择控件,不仅可以用于实现地区选择,只要是3个级别的选择都可以实现,比如专业选择.行业选择.职位选择等.效果如下图所示: 附:本实例asp. ...
- DELPHI中如何让FORM窗体透明,只显示控件?
DELPHI中如何让FORM窗体透明,只显示控件?分享到: 对我有用[0] 丢个板砖[0] 引用 | 举报 | 管理 回复次数:7largewanglargewanglargewang等级:Blank ...
- QT_5_ Qt中信号和槽 + 自定义信号和槽 + lambda 表达式
1.Qt中信号和槽 1.1 需求:点击按钮关闭窗口 1.2 利用connect进行链接 1.3 参数1 信号发送者(指针) 参数2 发送的信号(信号地址) 参数3 信号的接受者(指针) 参数4 处理槽 ...
- 如何获得 Qt窗口部件在主窗口中的位置--确定鼠标是否在某一控件上与在控件上的位置
用Qt Creator 设计程序时,最方便的就是ui设计器,可以很容易的得到想要的布局. 但是这样自动布局带来的后果是很难知道窗口中某一部件在主窗口中的相对位置. 在处理子窗口鼠标事件时变的很麻烦.主 ...
- Qt make clickable label 制作可点击的Label控件
Qt中的Label标签控件的作用绝不仅仅限于显示静态文本,其实它的功能很强大,由于其有setPixmap()成员函数,故而可以当显示图片窗口使用,而且还可以实时显示摄像头捕获的图片,然后它对鼠标事件的 ...
随机推荐
- 为何遍历Ldr会得到空项?
转自:http://www.0xaa55.com/thread-1385-1-1.html 之前做过ldr遍历的操作,发现第一项竟然是空,也就是大部分元素都是0,下面来揭示一下原理: 经过研究,其实L ...
- 《CSAPP》读书杂记 - Chapter 2. Representing and Manipulating Information
1. 一段查看地址内容的代码 代码: #include <stdio.h> typedef unsigned char *byte_pointer; void show_bytes(byt ...
- php curl_exec optimize
需求:前端过来一个请求,后台php要通过两次http请求,请求不同的地址得到资源后拼接返回给前端 请求A站: 请求B站: 同时请求A站和B站(php 串行 curl_exec ) 同时请求A站和B站( ...
- js获取当前url参数的两方式
方法一:正则分析法function getQueryString(name) { var reg = new RegExp("(^|&)" + name + &quo ...
- The method replace(int, Fragment, String) in the type FragmentTransaction is not applicable for the arguments (int, SettingFragment, String)
The method replace(int, Fragment, String) in the type FragmentTransaction is not applicable for the ...
- Linux网络管理——端口作用
1. 网络基础 .note-content {font-family: "Helvetica Neue",Arial,"Hiragino Sans GB",&q ...
- 向mysql添加新用户并分配权限
首先要声明一点,大部分情况下,修改MySQL是需要有mysql里的root权限的,所以一般用户无法更改密码,除非请求管理员. 方法一使用phpmyadmin,这是最简单的了,修改mysql库的user ...
- 解决“无法连接到Python代码运行助手。请检查本机的设置”问题
廖雪峰老师python课程里有个代码运行助手,可以让你在线输入Python代码,然后通过本机运行的一个Python脚本来执行代码,很方便的一个脚本工具,但是很多人用过之后出现了这样的提示:“无法连接到 ...
- ListView的简单使用和性能优化
起源:ListView是Android开发中使用最广泛的一种控件,它以垂直列表的形式显示所有列表项. 创建ListView有两种方式: ☆ 直接使用ListView进行创建. ☆让Activity继承 ...
- VHDL设计时参数定义的方法 例子
-- SPtb LIBRARY ieee; USE ieee.std_logic_1164.ALL; USE IEEE.STD_LOGIC_ARITH.ALL; USE IEEE.STD_LOGIC_ ...