Qt UI 文件机制

使用 Qt 设计界面程序时,若界面是静态的,可以借助 Qt Designer 进行所见即所得的界面设计。设计好界面后,在界面类中对 ui 对象进行操作非常方便。

QtCreator 自动生成的界面类

构建运行一个有 Qt 界面文件的项目时,会在程序构建目录下面生成一些前缀为 ui_ 的文件,在界面类(以 window 为例)中引用的就是对应 ui_window 文件中的内容。window.h 和 window.cpp 中的内容示例:

// window.h
namespace Ui {
class Window;
} class Window : public QWidget {
Q_OBJECT
public:
explicit Window(QWidget *parent = 0); private:
Ui::Window *ui;
}
// window.cpp
#include "window.h"
#include "ui_window.cpp" Window::Window() : QWidget(parent), ui(new Ui::Window)
{ }

由 Qt 的 moc 系统自动生成的 ui_window 中的内容示例:

QT_BEGIN_NAMESPACE

class Ui_Window
{
public:
QAction *action;
... void setupUi(QWidget *Window)
{
... retranslateUi(Window);
QMetaObject::connectSlotsByName(Window);
} void retranslateUi(QWidget *Window)
{
....
action->setText(QApplication::translate("Window", "aciton", 0));
} } namespace Ui {
class Window: public Ui_Window {};
} // namespace Ui QT_END_NAMESPACE

ui_window 文件中包含有 Ui_Window 类,还有个 Ui 命名空间,命名空间中包含有一个和界面类同名的 Window 类,而这个类是 Ui_Window 的子类。

Ui::Window 类的方法有: setupUi 和 retranslateUi,从字面上看能够知道一个是用于界面初始化的,一个是用来做多语言功能的。

分离界面和业务逻辑

Qt Designer 使得界面程序的开发变得简单,在 Designer 中拖动组件就可以快速完成界面的开发。但有些界面程序需要在界面上右键显示弹出菜单,如果能够像 Qt Designer 所作的那样,把右键菜单和具体的业务逻辑分开,那么界面类实现起来感觉就会更加流畅。(似乎没法直接用 Qt Designer 制作一个 QMenu 的 ui 文件)

自己动手写界面代码,就按照 Qt Creator 中提供的模版进行编写就可以了。将要用到的界面全都放进 ui_curve.h 中,然后在 curve.cpp 中引用 Ui::Curve。

#ifndef UI_CURVE_H
#define UI_CURVE_H #include <QVariant>
#include <QApplication>
#include <QWidget>
#include <QGridLayout>
#include <QMenu>
#include <QAction>
// 界面类 class Ui_Curve{
public:
QGridLayout *mainLayout;
QMenu *mainMenu;
QMenu *subCurveMenu;
QAction *autoXAxisRange;
QAction *autoYAxisRange;
QAction *saveDataAction;
QAction *xCurveAction;
QAction *yCurveAction;
QAction *zCurveAction;
QAction *subPlotAction;
QAction *resetAction;
QAction *rescaleAction;
QAction *setXRangeAction;
QAction *setYRangeAction;
QAction *unionSubYAxis;
QAction *wheelOnYOnly; void setupUi(QWidget *curve) {
if( curve->objectName().isEmpty() )
curve->setObjectName("Curve"); mainLayout = new QGridLayout();
curve->setLayout( mainLayout ); mainMenu = new QMenu( curve ); // setup Sub Curve on One plot => menu
subCurveMenu = new QMenu( curve );
xCurveAction = new QAction( curve );
xCurveAction->setData( "X" ); // used for identify
xCurveAction->setCheckable( true );
xCurveAction->setChecked( true );
subCurveMenu->addAction( xCurveAction ); yCurveAction = new QAction( curve );
yCurveAction->setData( "Y" );
yCurveAction->setCheckable( true );
yCurveAction->setChecked( true );
subCurveMenu->addAction( yCurveAction ); zCurveAction = new QAction( curve );
zCurveAction->setData( "Z" );
zCurveAction->setCheckable( true );
zCurveAction->setChecked( true );
subCurveMenu->addAction( zCurveAction ); mainMenu->addMenu( subCurveMenu );
mainMenu->setProperty( "SubCurveCount", 3 ); // display sub curve in independent widgets => action
subPlotAction = new QAction( curve );
subPlotAction->setCheckable( true );
subPlotAction->setChecked(true);
mainMenu->addAction( subPlotAction );
mainMenu->addSeparator(); resetAction = new QAction( curve );
mainMenu->addAction( resetAction );
mainMenu->addSeparator(); // rescale Y AXis => action
rescaleAction = new QAction( curve );
mainMenu->addAction( rescaleAction ); //
autoXAxisRange = new QAction( curve );
autoXAxisRange->setCheckable( true );
autoXAxisRange->setChecked( true );
mainMenu->addAction( autoXAxisRange ); autoYAxisRange = new QAction( curve );
autoYAxisRange->setCheckable( true );
autoYAxisRange->setChecked( true );
mainMenu->addAction( autoYAxisRange );
mainMenu->addSeparator(); setXRangeAction = new QAction( curve );
mainMenu->addAction( setXRangeAction );
setYRangeAction = new QAction( curve );
mainMenu->addAction( setYRangeAction ); unionSubYAxis = new QAction( curve );
unionSubYAxis->setCheckable( true );
unionSubYAxis->setChecked( true ); mainMenu->addAction( unionSubYAxis ); wheelOnYOnly = new QAction( curve );
wheelOnYOnly->setCheckable( true );
wheelOnYOnly->setChecked( true );
mainMenu->addAction( wheelOnYOnly ); mainMenu->addSeparator(); saveDataAction = new QAction( curve );
saveDataAction->setCheckable( true );
saveDataAction->setChecked( false );
mainMenu->addAction( saveDataAction ); retranslateUi( curve );
QMetaObject::connectSlotsByName(curve);
} void retranslateUi(QWidget *curve) {
subCurveMenu->setTitle(QApplication::translate("Curve", "Show Sub Curve On Main Widget", 0));
xCurveAction->setText(QApplication::translate("Curve", "Display X (red) Line", 0));
yCurveAction->setText(QApplication::translate("Curve", "Display Y (green) Line", 0));
zCurveAction->setText(QApplication::translate("Curve", "Display Z (blue) Line", 0));
subPlotAction->setText(QApplication::translate("Curve", "Display Sub Curves In Other Widgets", 0));
resetAction->setText(QApplication::translate("Curve", "Reset Timer and clear ALL points", 0));
rescaleAction->setText(QApplication::translate("Curve", "Rescale Y Axis", 0));
autoXAxisRange->setText(QApplication::translate("Curve", "Auto Set X Axis Range", 0));
autoYAxisRange->setText(QApplication::translate("Curve", "Auto Set Y Axis Range", 0));
setXRangeAction->setText(QApplication::translate("Curve", "Set X Axis Range", 0));
setYRangeAction->setText(QApplication::translate("Curve", "Set Y Axis Range", 0));
unionSubYAxis->setText(QApplication::translate("Curve", "union All three sub y axis (x-y-z)", 0));
wheelOnYOnly->setText(QApplication::translate("Curve", "Wheel Grag or Zomm On Y axis only", 0));
saveDataAction->setText(QApplication::translate("Curve", "Save All Data On Close", 0));
}
}; namespace Ui {
class Curve : public Ui_Curve {};
} #endif // UI_CURVE_H

然后在界面类中重写 mousePressEvent 事件,检测到鼠标右击后调用 ui->mainMenu->popup( event->globalPos() ); 弹出菜单。

上述文件中有两个方法值得注意:

  • QMetaObject::connectSlotsByName 这是用来自动连接信号和槽(槽的名称模式为:void on_<object name>_<signal name>(<signal parameters>);)的,但是 Ui_Curve 中我没有给各个 QAction 对象写 objectName,所以这里对 connectSlotsByName 的调用没有效果。
  • QApplication::translate 是提供翻译功能的,和 Qt 中常用的 QObject::tr 功能相同,但显然 QObject::tr 用起来更方便。

虽然编写界面代码没什么技术含量,但是更重要的事情是理解 Qt 的 UI 模型。这个模型能够将界面代码从逻辑代码中分离出来,分离后的 UI 和业务逻辑使得代码的可维护性提高,整体代码的可读性和可维护性都提高了很多。

Qt 学习(4)的更多相关文章

  1. QT学习第1天

    QT学习第一天  坚持住!! 一 Qt概述 1.Qt发展历史 (1)1991年诞生(Haavard Nord/Eirik Chambe-Eng), (2)1994年创立Troll Tech(奇趣科技) ...

  2. 【转载】分享一些Qt学习资源,欢迎下载

    资源来源:http://bbs.csdn.net/topics/390358737 经过我一翻整理,把一些我收集到的Qt学习资源分享给大家,主要适合新手,老鸟可以直接忽略我.要说明一下,很多资源都是在 ...

  3. QT学习之路--创建一个对话框

    Q_OBJECT:这是一个宏,凡是定义信号槽的类都必须声明这个宏. 函数tr()全名是QObject::tr(),被他处理过的字符串可以使用工具提取出来翻译成其他语言,也就是做国际化使用. 对于QT学 ...

  4. 转载: Qt 学习之路 2归档

    Qt 学习之路 2归档 http://www.devbean.net/2012/08/qt-study-road-2-catelog/

  5. Qt 学习资料

    Qter开源社区http://www.qter.org/ [Qt教程], 作者yafeilinux [视频] QT学习之路:从入门到精通 <C++ Qt 编程视频教程>

  6. qt学习笔记(五) QGraphicsPixmapItem与QGraphicsScene的编程实例 图标拖动渐变效果

    应大家的要求,还是把完整的project文件贴出来,大家省点事:http://www.kuaipan.cn/file/id_48923272389086450.htm 先看看执行效果,我用的群创7寸屏 ...

  7. Qt学习博客推荐

    附录C Qt资源 C.1 Qt 官方资源 全 球各大公司以及独立开发人员每天都在加入 Qt 的开发社区.他们已经认识到了Qt 的架构本身便可加快应用程序开发进度.这些开发人员,无论是想开发单平台软件. ...

  8. qt学习教程1.qt开发环境搭建

    qt学习教程1.qt开发环境搭建 首先下载qt 下载地址:http://download.qt.io/archive/qt/ 此教程使用的版本为5.1.1 下载好后,打开安装包,然后点下一步 选择一个 ...

  9. QT学习教程

    原地址:http://www.devbean.NET/2012/08/qt-study-road-2-catelog/ 网上看到的不错的教程 本教程以qt5为主,部分地方会涉及qt4.据说非常适合qt ...

  10. Qt学习之路

      Qt学习之路_14(简易音乐播放器)   Qt学习之路_13(简易俄罗斯方块)   Qt学习之路_12(简易数据管理系统)   Qt学习之路_11(简易多文档编辑器)   Qt学习之路_10(Qt ...

随机推荐

  1. layui下select下拉框不显示或没有效果

    Layui会对select.checkbox.radio等原始元素隐藏,从而进行美化修饰处理.但这需要依赖于form组件,所以你必须加载 form,并且执行一个实例.值得注意的是:导航的Hover效果 ...

  2. android studio 程序员有福了—从layout自动生成viewholder类

    狂点这里下载 超级牛逼的插件啊,比那些使用SparseArray的强太多了! 在android studio 1.0上测试,没有问题. 不说了直接说功能 Android Toolbox Plugin ...

  3. PS2018学习笔记(03-18节)

    3-认识主界面 # 主界面包括: 菜单栏.选项栏.工具栏.面板.图像编辑窗口(中间)和状态栏(底部): # 界面设置: 方法1:Ctrl+k:打开界面设置; 方法2:编辑-首选项-界面 # shift ...

  4. 自定义等高 Cell

    1.介绍 1.1 代码自定义 cell(使用 frame) 创建一个继承自 UITableViewCell 的子类,比如 BookCell1. 在 initWithStyle:reuseIdentif ...

  5. How to publish a pointcloud of ros msgs in a topic from a pcd file?

    How to publish a pointcloud of ros msgs in a topic from a pcd file? Two methods 1. modified source 2 ...

  6. SP705 SUBST1 - New Distinct Substrings

    \(\color{#0066ff}{ 题目描述 }\) 给定一个字符串,求该字符串含有的本质不同的子串数量. \(\color{#0066ff}{输入格式}\) T- number of test c ...

  7. P2253 好一个一中腰鼓!

    题意:给你一个序列,初始是0,每次一个操作,把一个数^=1 每次求出最长01串的长度 正解:线段树(虽然暴力能过) 对于每个区间,记录三个值 lmax,以l为首的01串长度 rmax,以r为尾的01串 ...

  8. Django 想要单独执行文件

    Django  单独文件执行  文件  我是在项目根目录文件创建的. 配置文件: #!/usr/bin/env python import os import sys if __name__ == & ...

  9. CF628D Magic Numbers (数据大+数位dp)求[a,b]中,偶数位的数字都是d,其余为数字都不是d,且能被m整除的数的个数

    题意:求[a,b]中,偶数位的数字都是d,其余为数字都不是d,且能被m整除的数的个数(这里的偶数位是的是从高位往低位数的偶数位).a,b<10^2000,m≤2000,0≤d≤9 a,b< ...

  10. vue.js学习笔记(二)——vue-router详解

    vue-router详解 原文链接:www.jianshu.com 一.前言 要学习vue-router就要先知道这里的路由是什么?为什么我们不能像原来一样直接用<a></a> ...