/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the Qt3Support module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qplatformdefs.h" #ifndef QT_NO_TITLEBAR #include "qapplication.h" #include "qcursor.h" #include "qdatetime.h" #include "qevent.h" #include "qimage.h" #include "qpainter.h" #include "qiodevice.h" #include "qpixmap.h" #include "qstyle.h" #include "qstyleoption.h" #include "qtimer.h" #include "qtooltip.h" #include "qdebug.h" #if defined(Q_WS_WIN) #include "qt_windows.h" #endif #include "private/qapplication_p.h" #include "private/q3titlebar_p.h" #include "private/qwidget_p.h" QT_BEGIN_NAMESPACE class Q3TitleBarPrivate : public QWidgetPrivate { Q_DECLARE_PUBLIC(Q3TitleBar) public: Q3TitleBarPrivate() : toolTip(0), act(0), window(0), movable(1), pressed(0), autoraise(0), inevent(0) { } Qt::WindowFlags flags; QStyle::SubControl buttonDown; QPoint moveOffset; QToolTip *toolTip; bool act :1; QWidget* window; bool movable :1; bool pressed :1; bool autoraise :1; bool inevent : 1; int titleBarState() const; QStyleOptionTitleBar getStyleOption() const; void readColors(); }; inline int Q3TitleBarPrivate::titleBarState() const { uint state = window ? window->windowState() : static_cast(Qt::WindowNoState); state |= uint(act ? QStyle::State_Active : QStyle::State_None); return (int)state; } QStyleOptionTitleBar Q3TitleBarPrivate::getStyleOption() const { Q_Q(const Q3TitleBar); QStyleOptionTitleBar opt; opt.init(q); opt.text = q->windowTitle(); //################ QIcon icon = q->windowIcon(); QSize s = icon.actualSize(QSize(64, 64)); opt.icon = icon.pixmap(s); opt.subControls = QStyle::SC_All; opt.activeSubControls = QStyle::SC_None; opt.titleBarState = titleBarState(); opt.titleBarFlags = flags; return opt; } Q3TitleBar::Q3TitleBar(QWidget *w, QWidget *parent, Qt::WindowFlags f) : QWidget(*new Q3TitleBarPrivate, parent, Qt::WStyle_Customize | Qt::WStyle_NoBorder) { Q_D(Q3TitleBar); if (f == 0 && w) f = w->windowFlags(); d->flags = f; d->window = w; d->buttonDown = QStyle::SC_None; d->act = 0; if (w) { if (w->minimumSize() == w->maximumSize()) d->flags &= ~Qt::WindowMaximizeButtonHint; setWindowTitle(w->windowTitle()); } d->readColors(); setSizePolicy(QSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed)); setMouseTracking(true); setAutoRaise(style()->styleHint(QStyle::SH_TitleBar_AutoRaise, 0, this)); } void Q3TitleBar::setFakeWindowFlags(Qt::WindowFlags f) { Q_D(Q3TitleBar); d->flags = f; } Qt::WindowFlags Q3TitleBar::fakeWindowFlags() const { Q_D(const Q3TitleBar); return d->flags; } Q3TitleBar::~Q3TitleBar() { } QStyleOptionTitleBar Q3TitleBar::getStyleOption() const { return d_func()->getStyleOption(); } #ifdef Q_WS_WIN static inline QRgb colorref2qrgb(COLORREF col) { return qRgb(GetRValue(col),GetGValue(col),GetBValue(col)); } #endif void Q3TitleBarPrivate::readColors() { Q_Q(Q3TitleBar); QPalette pal = q->palette(); bool colorsInitialized = false; #ifdef Q_WS_WIN // ask system properties on windows if (QApplication::desktopSettingsAware()) { pal.setColor(QPalette::Active, QPalette::Highlight, colorref2qrgb(GetSysColor(COLOR_ACTIVECAPTION))); pal.setColor(QPalette::Inactive, QPalette::Highlight, colorref2qrgb(GetSysColor(COLOR_INACTIVECAPTION))); pal.setColor(QPalette::Active, QPalette::HighlightedText, colorref2qrgb(GetSysColor(COLOR_CAPTIONTEXT))); pal.setColor(QPalette::Inactive, QPalette::HighlightedText, colorref2qrgb(GetSysColor(COLOR_INACTIVECAPTIONTEXT))); colorsInitialized = true; BOOL gradient = false; SystemParametersInfo(SPI_GETGRADIENTCAPTIONS, 0, &gradient, 0); if (gradient) { pal.setColor(QPalette::Active, QPalette::Base, colorref2qrgb(GetSysColor(COLOR_GRADIENTACTIVECAPTION))); pal.setColor(QPalette::Inactive, QPalette::Base, colorref2qrgb(GetSysColor(COLOR_GRADIENTINACTIVECAPTION))); } else { pal.setColor(QPalette::Active, QPalette::Base, pal.color(QPalette::Active, QPalette::Highlight)); pal.setColor(QPalette::Inactive, QPalette::Base, pal.color(QPalette::Inactive, QPalette::Highlight)); } } #endif // Q_WS_WIN if (!colorsInitialized) { pal.setColor(QPalette::Active, QPalette::Highlight, pal.color(QPalette::Active, QPalette::Highlight)); pal.setColor(QPalette::Active, QPalette::Base, pal.color(QPalette::Active, QPalette::Highlight)); pal.setColor(QPalette::Inactive, QPalette::Highlight, pal.color(QPalette::Inactive, QPalette::Dark)); pal.setColor(QPalette::Inactive, QPalette::Base, pal.color(QPalette::Inactive, QPalette::Dark)); pal.setColor(QPalette::Inactive, QPalette::HighlightedText, pal.color(QPalette::Inactive, QPalette::Window)); } q->setPalette(pal); q->setActive(act); } void Q3TitleBar::changeEvent(QEvent *ev) { if(ev->type() == QEvent::ModifiedChange) update(); QWidget::changeEvent(ev); } void Q3TitleBar::mousePressEvent(QMouseEvent *e) { Q_D(Q3TitleBar); if (!d->act) emit doActivate(); if (e->button() == Qt::LeftButton) { d->pressed = true; QStyleOptionTitleBar opt = d->getStyleOption(); QStyle::SubControl ctrl = style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, e->pos(), this); switch (ctrl) { case QStyle::SC_TitleBarSysMenu: if (d->flags & Qt::WindowSystemMenuHint) { d->buttonDown = QStyle::SC_None; static QTime *t = 0; static Q3TitleBar *tc = 0; if (!t) t = new QTime; if (tc != this || t->elapsed() > QApplication::doubleClickInterval()) { emit showOperationMenu(); t->start(); tc = this; } else { tc = 0; emit doClose(); return; } } break; case QStyle::SC_TitleBarShadeButton: case QStyle::SC_TitleBarUnshadeButton: if (d->flags & Qt::WindowShadeButtonHint) d->buttonDown = ctrl; break; case QStyle::SC_TitleBarNormalButton: if (d->flags & Qt::WindowMinMaxButtonsHint) d->buttonDown = ctrl; break; case QStyle::SC_TitleBarMinButton: if (d->flags & Qt::WindowMinimizeButtonHint) d->buttonDown = ctrl; break; case QStyle::SC_TitleBarMaxButton: if (d->flags & Qt::WindowMaximizeButtonHint) d->buttonDown = ctrl; break; case QStyle::SC_TitleBarCloseButton: if (d->flags & Qt::WindowSystemMenuHint) d->buttonDown = ctrl; break; case QStyle::SC_TitleBarLabel: d->buttonDown = ctrl; d->moveOffset = mapToParent(e->pos()); break; default: break; } repaint(); } else { d->pressed = false; } } void Q3TitleBar::contextMenuEvent(QContextMenuEvent *e) { Q_D(Q3TitleBar); QStyleOptionTitleBar opt = d->getStyleOption(); QStyle::SubControl ctrl = style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, e->pos(), this); if(ctrl == QStyle::SC_TitleBarLabel || ctrl == QStyle::SC_TitleBarSysMenu) { e->accept(); emit popupOperationMenu(e->globalPos()); } else { e->ignore(); } } void Q3TitleBar::mouseReleaseEvent(QMouseEvent *e) { Q_D(Q3TitleBar); if (e->button() == Qt::LeftButton && d->pressed) { e->accept(); QStyleOptionTitleBar opt = d->getStyleOption(); QStyle::SubControl ctrl = style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, e->pos(), this); d->pressed = false; if (ctrl == d->buttonDown) { d->buttonDown = QStyle::SC_None; repaint(); switch(ctrl) { case QStyle::SC_TitleBarShadeButton: case QStyle::SC_TitleBarUnshadeButton: if(d->flags & Qt::WindowShadeButtonHint) emit doShade(); break; case QStyle::SC_TitleBarNormalButton: if(d->flags & Qt::WindowMaximizeButtonHint) emit doNormal(); break; case QStyle::SC_TitleBarMinButton: if(d->flags & Qt::WindowMinimizeButtonHint) { if (d->window && d->window->isMinimized()) emit doNormal(); else emit doMinimize(); } break; case QStyle::SC_TitleBarMaxButton: if(d->flags & Qt::WindowMaximizeButtonHint) { if(d->window && d->window->isMaximized()) emit doNormal(); else emit doMaximize(); } break; case QStyle::SC_TitleBarCloseButton: if(d->flags & Qt::WindowSystemMenuHint) { d->buttonDown = QStyle::SC_None; repaint(); emit doClose(); return; } break; default: break; } } } else { e->ignore(); } } void Q3TitleBar::mouseMoveEvent(QMouseEvent *e) { Q_D(Q3TitleBar); e->accept(); switch (d->buttonDown) { case QStyle::SC_None: if(autoRaise()) repaint(); break; case QStyle::SC_TitleBarSysMenu: break; case QStyle::SC_TitleBarShadeButton: case QStyle::SC_TitleBarUnshadeButton: case QStyle::SC_TitleBarNormalButton: case QStyle::SC_TitleBarMinButton: case QStyle::SC_TitleBarMaxButton: case QStyle::SC_TitleBarCloseButton: { QStyle::SubControl last_ctrl = d->buttonDown; QStyleOptionTitleBar opt = d->getStyleOption(); d->buttonDown = style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, e->pos(), this); if (d->buttonDown != last_ctrl) d->buttonDown = QStyle::SC_None; repaint(); d->buttonDown = last_ctrl; } break; case QStyle::SC_TitleBarLabel: if (d->buttonDown == QStyle::SC_TitleBarLabel && d->movable && d->pressed) { if ((d->moveOffset - mapToParent(e->pos())).manhattanLength() >= 4) { QPoint p = mapFromGlobal(e->globalPos()); QWidget *parent = d->window ? d->window->parentWidget() : 0; if(parent && parent->inherits("Q3WorkspaceChild")) { QWidget *workspace = parent->parentWidget(); p = workspace->mapFromGlobal(e->globalPos()); if (!workspace->rect().contains(p)) { if (p.x() < 0) p.rx() = 0; if (p.y() < 0) p.ry() = 0; if (p.x() > workspace->width()) p.rx() = workspace->width(); if (p.y() > workspace->height()) p.ry() = workspace->height(); } } QPoint pp = p - d->moveOffset; if (!parentWidget()->isMaximized()) parentWidget()->move(pp); } } else { QStyle::SubControl last_ctrl = d->buttonDown; d->buttonDown = QStyle::SC_None; if(d->buttonDown != last_ctrl) repaint(); } break; default: break; } } void Q3TitleBar::resizeEvent(QResizeEvent *r) { QWidget::resizeEvent(r); cutText(); } bool Q3TitleBar::isTool() const { return (d_func()->flags & Qt::WindowType_Mask) == Qt::Tool; } void Q3TitleBar::paintEvent(QPaintEvent *) { Q_D(Q3TitleBar); QStyleOptionTitleBar opt = d->getStyleOption(); opt.subControls = QStyle::SC_TitleBarLabel; opt.activeSubControls = d->buttonDown; if (d->flags & Qt::WindowSystemMenuHint) { opt.subControls |= QStyle::SC_TitleBarSysMenu | QStyle::SC_TitleBarCloseButton; if (d->window && (d->flags & Qt::WindowShadeButtonHint)) { if (d->window->isMinimized()) opt.subControls |= QStyle::SC_TitleBarUnshadeButton; else opt.subControls |= QStyle::SC_TitleBarShadeButton; } if (d->window && (d->flags & Qt::WindowMinMaxButtonsHint)) { if(d->window && d->window->isMinimized()) opt.subControls |= QStyle::SC_TitleBarNormalButton; else opt.subControls |= QStyle::SC_TitleBarMinButton; } if (d->window && (d->flags & Qt::WindowMaximizeButtonHint) && !d->window->isMaximized()) opt.subControls |= QStyle::SC_TitleBarMaxButton; } QStyle::SubControl under_mouse = QStyle::SC_None; if (underMouse()) { under_mouse = style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, mapFromGlobal(QCursor::pos()), this); opt.activeSubControls |= under_mouse; if (d->pressed) opt.state |= QStyle::State_Sunken; else if(autoRaise()) opt.state |= QStyle::State_MouseOver; } opt.palette.setCurrentColorGroup(usesActiveColor() ? QPalette::Active : QPalette::Inactive); QPainter p(this); if (!windowTitle().isEmpty()) opt.titleBarFlags |= Qt::WindowTitleHint; style()->drawComplexControl(QStyle::CC_TitleBar, &opt, &p, this); } void Q3TitleBar::mouseDoubleClickEvent(QMouseEvent *e) { Q_D(Q3TitleBar); if (e->button() != Qt::LeftButton) { e->ignore(); return; } e->accept(); QStyleOptionTitleBar opt = d->getStyleOption(); switch (style()->hitTestComplexControl(QStyle::CC_TitleBar, &opt, e->pos(), this)) { case QStyle::SC_TitleBarLabel: emit doubleClicked(); break; case QStyle::SC_TitleBarSysMenu: if (d->flags & Qt::WStyle_SysMenu) emit doClose(); break; default: break; } } void Q3TitleBar::cutText() { Q_D(Q3TitleBar); QFontMetrics fm(font()); QStyleOptionTitleBar opt = d->getStyleOption(); int maxw = style()->subControlRect(QStyle::CC_TitleBar, &opt, QStyle::SC_TitleBarLabel, this).width(); if (!d->window) return; QString txt = d->window->windowTitle(); if (style()->styleHint(QStyle::SH_TitleBar_ModifyNotification, 0, this) && d->window && d->window->isWindowModified()) txt += QLatin1String(" *"); QString cuttext = txt; if (fm.width(txt + QLatin1Char('m')) > maxw) { int i = txt.length(); int dotlength = fm.width(QLatin1String("...")); while (i>0 && fm.width(txt.left(i)) + dotlength > maxw) i--; if(i != (int)txt.length()) cuttext = txt.left(i) + QLatin1String("..."); } setWindowTitle(cuttext); } void Q3TitleBar::leaveEvent(QEvent *) { if(autoRaise() && !d_func()->pressed) repaint(); } void Q3TitleBar::enterEvent(QEvent *) { if(autoRaise() && !d_func()->pressed) repaint(); QEvent e(QEvent::Leave); QApplication::sendEvent(parentWidget(), &e); } void Q3TitleBar::setActive(bool active) { Q_D(Q3TitleBar); if (d->act == active) return ; d->act = active; update(); } bool Q3TitleBar::isActive() const { return d_func()->act; } bool Q3TitleBar::usesActiveColor() const { return (isActive() && isActiveWindow()) || (!window() && QWidget::window()->isActiveWindow()); } QWidget *Q3TitleBar::window() const { return d_func()->window; } bool Q3TitleBar::event(QEvent *e) { Q_D(Q3TitleBar); if (d->inevent) return QWidget::event(e); d->inevent = true; if (e->type() == QEvent::ApplicationPaletteChange) { d->readColors(); return true; } else if (e->type() == QEvent::WindowActivate) { setActive(d->act); } else if (e->type() == QEvent::WindowDeactivate) { bool wasActive = d->act; setActive(false); d->act = wasActive; } else if (e->type() == QEvent::WindowIconChange) { update(); } else if (e->type() == QEvent::WindowTitleChange) { cutText(); update(); } d->inevent = false; return QWidget::event(e); } void Q3TitleBar::setMovable(bool b) { d_func()->movable = b; } bool Q3TitleBar::isMovable() const { return d_func()->movable; } void Q3TitleBar::setAutoRaise(bool b) { d_func()->autoraise = b; } bool Q3TitleBar::autoRaise() const { return d_func()->autoraise; } QSize Q3TitleBar::sizeHint() const { ensurePolished(); QStyleOptionTitleBar opt = d_func()->getStyleOption(); QRect menur = style()->subControlRect(QStyle::CC_TitleBar, &opt, QStyle::SC_TitleBarSysMenu, this); return QSize(menur.width(), style()->pixelMetric(QStyle::PM_TitleBarHeight, &opt, this)); } QT_END_NAMESPACE #endif //QT_NO_TITLEBAR