QtCreator啟用了ClangCodeModel插件,會對代碼進(jìn)行分析,然后給出警告。這是個(gè)好功能,可以發(fā)現(xiàn)一些潛在的問題,但是也有代價(jià)。像我這邊的一個(gè)舊項(xiàng)目,日志模塊是單個(gè)的頭文件,所以在大量的地方被使用,而因?yàn)檫@個(gè)日志模塊使用大量的下劃線命名,vc內(nèi)置擴(kuò)展函數(shù)等等,導(dǎo)致這個(gè)模塊被給出大量警告,又由于它被大量文件包含和使用,從而使得到處都是警告,而且嚴(yán)重影響了速度,經(jīng)??ò胩?。只能無奈在插件管理中先把這個(gè)插件禁用了。
QLabel顯示文本的時(shí)候,默認(rèn)會以把文本的寬度設(shè)為最小大小,導(dǎo)致縮小的時(shí)候,無法讓寬度變小
contextMenuEvent之類的事件給的參數(shù)中,如果涉及到坐標(biāo),那么就是針對覆寫這個(gè)虛函數(shù)的QWidget對象而言的,傳遞這個(gè)坐標(biāo)給子對象的時(shí)候一定要注意。
QTableWidget的closePersistentEditor方法調(diào)用時(shí),如果當(dāng)前被編輯的item沒有被選中,那么會在該方法中觸發(fā)itemSelectionChanged信號(removeItem、removeRow這些方法如果刪除的是當(dāng)前選中的item或row,也會觸發(fā)itemSelectionChanged信號)
繼承QWidget實(shí)現(xiàn)的子類,如果要使用setStyleSheet對自己(QWidget)設(shè)置樣式,那么類似下面的樣子需要覆寫paintEvent,不然樣式對自己無效
void CustomWidget::paintEvent(QPaintEvent *)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}
disconnect()或disconnect(this)只是會斷開對象的自身方法槽,對應(yīng)lambda槽是無效的
-
分割線可以使用QFrame來實(shí)現(xiàn),一般是三步:
- setFrameShape設(shè)置方向,QFrame::HLine是橫向分割線,QFrame::VLine是縱向分割線;
- setFixedHeight/setFixedWidth設(shè)置高度或?qū)挾龋?/li>
- 通過QSS設(shè)置線的顏色,
橫向分隔線:QFrame[frameShape="4"] { color: black; }
縱向分割線:QFrame[frameShape="5"] { color: black; }
QWidget的setContentMargins可以用來設(shè)置作圖的區(qū)域,和用layout的setContentMargins并不重疊,相互獨(dú)立;
一般使用QWidget的setContentMargins來避開QWidget自身的邊框QStackedLayout的setContentMargins無效(原因不明),可以用QStackedWidget取代
QWidget創(chuàng)建的時(shí)候如果不指定parent,那么就是一個(gè)窗口,否則就是一個(gè)子控件;
這里會出現(xiàn)這么一種情況:如果創(chuàng)建的時(shí)候不指定parent,即使緊接著對其或直接或間接再設(shè)置了parent,那么在顯示的時(shí)候,也會出現(xiàn)閃現(xiàn)一個(gè)窗口的情況,所以QWidget如果作為子控件的話,那么最好在創(chuàng)建時(shí)指定好parent。
還有另外一種情況是,QWidget的派生類如QPushButton這些,如果創(chuàng)建的時(shí)候不指定parent,但是緊接著又進(jìn)行了setVisible(true)的操作,那么也會出現(xiàn)閃現(xiàn)窗口的情況。此時(shí),要么創(chuàng)建時(shí)指定parent,要么換過另外一個(gè)時(shí)機(jī)來進(jìn)行setVisible(true)的操作。QSS支持使用自定義的類名作為類型選擇子的,使用的是metaObject()->className()返回的類名來匹配,但是這里要注意,如果類定義在名字空間里,className()返回的格式是<namespace>::<classname>,這種格式和子控件選擇子沖突,所以針對定義在非匿名空間下的類,選擇子要使用<namespace>--<classname>這樣的形式,即把::替換為--
繼承QObject的類,不管用不用信號槽,最好定義的時(shí)候都帶上Q_OBJECT宏,因?yàn)檫@個(gè)宏定義了該類的元對象,可以獲取到類的信息;如果不用這個(gè)宏,那么在代碼用到metaObject的時(shí)候,拿到的會是父類的類信息,從而有可能發(fā)生混淆;另外還可能在一些不明顯的地方引起疑惑,比如用該類名作為qss的選擇子,會發(fā)現(xiàn)設(shè)置的qss無效,就是因?yàn)檫x擇子匹配的時(shí)候會當(dāng)前類的類名返回的是父類的類名,從而造成不匹配
如果只是對QWidget派生類的某個(gè)事件感興趣,進(jìn)行簡單處理,那么可以用installEventFilter方法給對象加上事件監(jiān)聽,然后過濾出需要的事件進(jìn)行處理即可,這樣可以避免再繼承該類定義一個(gè)新類。
一般的QWidget派生類的焦點(diǎn)策略都是Qt::NoFocus,所以如果希望實(shí)現(xiàn)點(diǎn)擊空白區(qū)域進(jìn)行切換焦點(diǎn)的話,就需要把對應(yīng)的QWidget的焦點(diǎn)策略修改為Qt::ClickFocus
QLineEdit點(diǎn)擊進(jìn)入編輯態(tài)的過程,先處理FocusInEvent,然后處理MousePressEvent,并且MousePressEvent處理中會清除選中,然后把輸入符放在適當(dāng)位置。所以,如果我們想實(shí)現(xiàn)QLineEdit獲得焦點(diǎn)后自動全選文本的效果,在FocusInEvent里就不能直接使用selectAll,而是要異步一下
QTimer::singleShot(0, obj, &QLineEdit::selectAll),才能避免被后續(xù)的MousePressEvent事件處理清除掉QSS的文檔中雖然說到QWidget::setFont和QWidget::setPalette會從父控件傳遞到子控件,而樣式表的設(shè)置默認(rèn)不會。
但是要注意的是,setFont和樣式表有不一樣的傳遞系統(tǒng),一旦應(yīng)用了樣式表,那么font就不再會被傳遞了。所以想要設(shè)置一些默認(rèn)字體或顏色的時(shí)候,還是統(tǒng)一使用樣式表比較好。如果QPushButton設(shè)置了菜單,那么點(diǎn)擊按鈕的時(shí)候,是先菜單出現(xiàn),等菜單消失之后,才會發(fā)pressed和release信號,并且不會發(fā)clicked信號。
Qt庫部分控件右鍵菜單是英文的解決方案:對于QLineEdit、QScrollBar 、QSpinBox 等控件需要加載qt_zh_CN.qm翻譯文件;對于QTextEdit、QPlainTextEdit 、QTextBrowser 等控件需要加載widgets.qm翻譯文件,在Qt的安裝目錄中找不到此文件,需要自己生成。