Qt新手踩坑之旅
-
QtCreator启用了ClangCodeModel插件,会对代码进行分析,然后给出警告。这是个好功能,可以发现一些潜在的问题,但是也有代价。像我这边的一个旧项目,日志模块是单个的头文件,所以在大量的地方被使用,而因为这个日志模块使用大量的下划线命名,vc内置扩展函数等等,导致这个模块被给出大量警告,又由于它被大量文件包含和使用,从而使得到处都是警告,而且严重影响了速度,经常卡半天。只能无奈在插件管理中先把这个插件禁用了。
-
QLabel显示文本的时候,默认会以把文本的宽度设为最小大小,导致缩小的时候,无法让宽度变小
-
contextMenuEvent之类的事件给的参数中,如果涉及到坐标,那么就是针对覆写这个虚函数的QWidget对象而言的,传递这个坐标给子对象的时候一定要注意。
-
QTableWidget的closePersistentEditor方法调用时,如果当前被编辑的item没有被选中,那么会在该方法中触发itemSelectionChanged信号(removeItem、removeRow这些方法如果删除的是当前选中的item或row,也会触发itemSelectionChanged信号)
-
继承QWidget实现的子类,如果要使用setStyleSheet对自己(QWidget)设置样式,那么类似下面的样子需要覆写paintEvent,不然样式对自己无效
void CustomWidget::paintEvent(QPaintEvent *)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}
-
disconnect()或disconnect(this)只是会断开对象的自身方法槽,对应lambda槽是无效的
-
分割线可以使用QFrame来实现,一般是三步:
- setFrameShape设置方向,QFrame::HLine是横向分割线,QFrame::VLine是纵向分割线;
- setFixedHeight/setFixedWidth设置高度或宽度;
- 通过QSS设置线的颜色,
横向分隔线:QFrame[frameShape="4"] { color: black; }
纵向分割线:QFrame[frameShape="5"] { color: black; }
-
QWidget的setContentMargins可以用来设置作图的区域,和用layout的setContentMargins并不重叠,相互独立;
一般使用QWidget的setContentMargins来避开QWidget自身的边框 -
QStackedLayout的setContentMargins无效(原因不明),可以用QStackedWidget取代
-
QWidget创建的时候如果不指定parent,那么就是一个窗口,否则就是一个子控件;
这里会出现这么一种情况:如果创建的时候不指定parent,即使紧接着对其或直接或间接再设置了parent,那么在显示的时候,也会出现闪现一个窗口的情况,所以QWidget如果作为子控件的话,那么最好在创建时指定好parent。
还有另外一种情况是,QWidget的派生类如QPushButton这些,如果创建的时候不指定parent,但是紧接着又进行了setVisible(true)的操作,那么也会出现闪现窗口的情况。此时,要么创建时指定parent,要么换过另外一个时机来进行setVisible(true)的操作。 -
QSS支持使用自定义的类名作为类型选择子的,使用的是metaObject()->className()返回的类名来匹配,但是这里要注意,如果类定义在名字空间里,className()返回的格式是<namespace>::<classname>,这种格式和子控件选择子冲突,所以针对定义在非匿名空间下的类,选择子要使用<namespace>--<classname>这样的形式,即把::替换为--
-
继承QObject的类,不管用不用信号槽,最好定义的时候都带上Q_OBJECT宏,因为这个宏定义了该类的元对象,可以获取到类的信息;如果不用这个宏,那么在代码用到metaObject的时候,拿到的会是父类的类信息,从而有可能发生混淆;另外还可能在一些不明显的地方引起疑惑,比如用该类名作为qss的选择子,会发现设置的qss无效,就是因为选择子匹配的时候会当前类的类名返回的是父类的类名,从而造成不匹配
-
如果只是对QWidget派生类的某个事件感兴趣,进行简单处理,那么可以用installEventFilter方法给对象加上事件监听,然后过滤出需要的事件进行处理即可,这样可以避免再继承该类定义一个新类。
-
一般的QWidget派生类的焦点策略都是Qt::NoFocus,所以如果希望实现点击空白区域进行切换焦点的话,就需要把对应的QWidget的焦点策略修改为Qt::ClickFocus
-
QLineEdit点击进入编辑态的过程,先处理FocusInEvent,然后处理MousePressEvent,并且MousePressEvent处理中会清除选中,然后把输入符放在适当位置。所以,如果我们想实现QLineEdit获得焦点后自动全选文本的效果,在FocusInEvent里就不能直接使用selectAll,而是要异步一下
QTimer::singleShot(0, obj, &QLineEdit::selectAll)
,才能避免被后续的MousePressEvent事件处理清除掉 -
QSS的文档中虽然说到QWidget::setFont和QWidget::setPalette会从父控件传递到子控件,而样式表的设置默认不会。
但是要注意的是,setFont和样式表有不一样的传递系统,一旦应用了样式表,那么font就不再会被传递了。所以想要设置一些默认字体或颜色的时候,还是统一使用样式表比较好。 -
如果QPushButton设置了菜单,那么点击按钮的时候,是先菜单出现,等菜单消失之后,才会发pressed和release信号,并且不会发clicked信号。
-
Qt库部分控件右键菜单是英文的解决方案:对于QLineEdit、QScrollBar 、QSpinBox 等控件需要加载qt_zh_CN.qm翻译文件;对于QTextEdit、QPlainTextEdit 、QTextBrowser 等控件需要加载widgets.qm翻译文件,在Qt的安装目录中找不到此文件,需要自己生成。