/**************************************************************************** ** ** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). ** All rights reserved. ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** This file is part of the QtGui module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** No Commercial Usage ** This file contains pre-release code and may not be distributed. ** You may use this file in accordance with the terms and conditions ** contained in the Technology Preview License Agreement accompanying ** this package. ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** ** ** ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ /* Note: The qdoc comments for QMacStyle are contained in .../doc/src/qstyles.qdoc. */ #include "qmacstyle_mac.h" #if defined(Q_WS_MAC) && !defined(QT_NO_STYLE_MAC) #define QMAC_QAQUASTYLE_SIZE_CONSTRAIN //#define DEBUG_SIZE_CONSTRAINT #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE extern QRegion qt_mac_convert_mac_region(RgnHandle); //qregion_mac.cpp // The following constants are used for adjusting the size // of push buttons so that they are drawn inside their bounds. static const int PushButtonLeftOffset = 6; static const int PushButtonTopOffset = 4; static const int PushButtonRightOffset = 12; static const int PushButtonBottomOffset = 12; static const int MiniButtonH = 26; static const int SmallButtonH = 30; static const int BevelButtonW = 50; static const int BevelButtonH = 22; static const int PushButtonContentPadding = 6; // These colors specify the titlebar gradient colors on // Leopard. Ideally we should get them from the system. static const QColor titlebarGradientActiveBegin(220, 220, 220); static const QColor titlebarGradientActiveEnd(151, 151, 151); static const QColor titlebarSeparatorLineActive(111, 111, 111); static const QColor titlebarGradientInactiveBegin(241, 241, 241); static const QColor titlebarGradientInactiveEnd(207, 207, 207); static const QColor titlebarSeparatorLineInactive(131, 131, 131); // Gradient colors used for the dock widget title bar and // non-unifed tool bar bacground. static const QColor mainWindowGradientBegin(240, 240, 240); static const QColor mainWindowGradientEnd(200, 200, 200); #if (MAC_OS_X_VERSION_MAX_ALLOWED <= MAC_OS_X_VERSION_10_5) enum { kThemePushButtonTextured = 31, kThemePushButtonTexturedSmall = 32, kThemePushButtonTexturedMini = 33 }; /* Search fields */ enum { kHIThemeFrameTextFieldRound = 1000, kHIThemeFrameTextFieldRoundSmall = 1001, kHIThemeFrameTextFieldRoundMini = 1002 }; #endif // Resolve these at run-time, since the functions was moved in Leopard. typedef HIRect * (*PtrHIShapeGetBounds)(HIShapeRef, HIRect *); static PtrHIShapeGetBounds ptrHIShapeGetBounds = 0; static bool isVerticalTabs(const QTabBar::Shape shape) { return (shape == QTabBar::RoundedEast || shape == QTabBar::TriangularEast || shape == QTabBar::RoundedWest || shape == QTabBar::TriangularWest); } static int closeButtonSize = 12; void drawTabCloseButton(QPainter *p, bool hover, bool active, bool selected) { // draw background circle p->setRenderHints(QPainter::Antialiasing); QRect rect(0, 0, closeButtonSize, closeButtonSize); QColor background; if (hover) { background = QColor(124, 124, 124); } else { if (active) { if (selected) background = QColor(104, 104, 104); else background = QColor(83, 83, 83); } else { if (selected) background = QColor(144, 144, 144); else background = QColor(114, 114, 114); } } p->setPen(Qt::transparent); p->setBrush(background); p->drawEllipse(rect); // draw cross int min = 3; int max = 9; QPen crossPen; crossPen.setColor(QColor(194, 194, 194)); crossPen.setWidthF(1.3); crossPen.setCapStyle(Qt::FlatCap); p->setPen(crossPen); p->drawLine(min, min, max, max); p->drawLine(min, max, max, min); } QRect rotateTabPainter(QPainter *p, QTabBar::Shape shape, QRect tabRect) { if (isVerticalTabs(shape)) { int newX, newY, newRot; if (shape == QTabBar::RoundedEast || shape == QTabBar::TriangularEast) { newX = tabRect.width(); newY = tabRect.y(); newRot = 90; } else { newX = 0; newY = tabRect.y() + tabRect.height(); newRot = -90; } tabRect.setRect(0, 0, tabRect.height(), tabRect.width()); QMatrix m; m.translate(newX, newY); m.rotate(newRot); p->setMatrix(m, true); } return tabRect; } void drawTabShape(QPainter *p, const QStyleOptionTabV3 *tabOpt) { QRect r = tabOpt->rect; p->translate(tabOpt->rect.x(), tabOpt->rect.y()); r.moveLeft(0); r.moveTop(0); QRect tabRect = rotateTabPainter(p, tabOpt->shape, r); int width = tabRect.width(); int height = 20; bool active = (tabOpt->state & QStyle::State_Active); bool selected = (tabOpt->state & QStyle::State_Selected); if (selected) { QRect rect(1, 0, width - 2, height); // fill body if (active) { int d = (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_6) ? 16 : 0; p->fillRect(rect, QColor(151 + d, 151 + d, 151 + d)); } else { int d = (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_6) ? 9 : 0; QLinearGradient gradient(rect.topLeft(), rect.bottomLeft()); gradient.setColorAt(0, QColor(207 + d, 207 + d, 207 + d)); gradient.setColorAt(0.5, QColor(206 + d, 206 + d, 206 + d)); gradient.setColorAt(1, QColor(201 + d, 201 + d, 201 + d)); p->fillRect(rect, gradient); } // draw border QColor borderSides; QColor borderBottom; if (active) { borderSides = QColor(88, 88, 88); borderBottom = QColor(88, 88, 88); } else { borderSides = QColor(121, 121, 121); borderBottom = QColor(116, 116, 116); } p->setPen(borderSides); int bottom = height; // left line p->drawLine(0, 1, 0, bottom-2); // right line p->drawLine(width-1, 1, width-1, bottom-2); // bottom line if (active) { p->setPen(QColor(168, 168, 168)); p->drawLine(3, bottom-1, width-3, bottom-1); } p->setPen(borderBottom); p->drawLine(2, bottom, width-2, bottom); int w = 3; QRectF rectangleLeft(1, height - w, w, w); QRectF rectangleRight(width - 2, height - 1, w, w); int startAngle = 180 * 16; int spanAngle = 90 * 16; p->setRenderHint(QPainter::Antialiasing); p->drawArc(rectangleLeft, startAngle, spanAngle); p->drawArc(rectangleRight, startAngle, -spanAngle); } else { // when the mouse is over non selected tabs they get a new color bool hover = (tabOpt->state & QStyle::State_MouseOver); if (hover) { QRect rect(1, 2, width - 1, height - 1); p->fillRect(rect, QColor(110, 110, 110)); } // seperator lines between tabs bool west = (tabOpt->shape == QTabBar::RoundedWest || tabOpt->shape == QTabBar::TriangularWest); bool drawOnRight = !west; if ((!drawOnRight && tabOpt->selectedPosition != QStyleOptionTab::NextIsSelected) || (drawOnRight && tabOpt->selectedPosition != QStyleOptionTab::NextIsSelected)) { QColor borderColor; QColor borderHighlightColor; if (active) { borderColor = QColor(64, 64, 64); borderHighlightColor = QColor(140, 140, 140); } else { borderColor = QColor(135, 135, 135); borderHighlightColor = QColor(178, 178, 178); } int x = drawOnRight ? width : 0; // tab seperator line p->setPen(borderColor); p->drawLine(x, 2, x, height + 1); // tab seperator highlight p->setPen(borderHighlightColor); p->drawLine(x-1, 2, x-1, height + 1); p->drawLine(x+1, 2, x+1, height + 1); } } } void drawTabBase(QPainter *p, const QStyleOptionTabBarBaseV2 *tbb, const QWidget *w) { QRect r = tbb->rect; if (isVerticalTabs(tbb->shape)) { r.setWidth(w->width()); } else { r.setHeight(w->height()); } QRect tabRect = rotateTabPainter(p, tbb->shape, r); int width = tabRect.width(); int height = tabRect.height(); bool active = (tbb->state & QStyle::State_Active); // top border lines QColor borderHighlightTop; QColor borderTop; if (active) { borderTop = QColor(64, 64, 64); borderHighlightTop = QColor(174, 174, 174); } else { borderTop = QColor(135, 135, 135); borderHighlightTop = QColor(207, 207, 207); } p->setPen(borderHighlightTop); p->drawLine(tabRect.x(), 0, width, 0); p->setPen(borderTop); p->drawLine(tabRect.x(), 1, width, 1); // center block QRect centralRect(tabRect.x(), 2, width, height - 2); if (active) { QColor mainColor = QColor(120, 120, 120); p->fillRect(centralRect, mainColor); } else { QLinearGradient gradient(centralRect.topLeft(), centralRect.bottomLeft()); gradient.setColorAt(0, QColor(165, 165, 165)); gradient.setColorAt(0.5, QColor(164, 164, 164)); gradient.setColorAt(1, QColor(158, 158, 158)); p->fillRect(centralRect, gradient); } // bottom border lines QColor borderHighlightBottom; QColor borderBottom; if (active) { borderHighlightBottom = QColor(153, 153, 153); borderBottom = QColor(64, 64, 64); } else { borderHighlightBottom = QColor(177, 177, 177); borderBottom = QColor(127, 127, 127); } p->setPen(borderHighlightBottom); p->drawLine(tabRect.x(), height - 2, width, height - 2); p->setPen(borderBottom); p->drawLine(tabRect.x(), height - 1, width, height - 1); } /* AHIG: Apple Human Interface Guidelines http://developer.apple.com/documentation/UserExperience/Conceptual/OSXHIGuidelines/ Builder: Apple Interface Builder v. 3.1.1 */ // this works as long as we have at most 16 different control types #define CT1(c) CT2(c, c) #define CT2(c1, c2) ((uint(c1) << 16) | uint(c2)) enum QAquaWidgetSize { QAquaSizeLarge = 0, QAquaSizeSmall = 1, QAquaSizeMini = 2, QAquaSizeUnknown = -1 }; #define SIZE(large, small, mini) \ (controlSize == QAquaSizeLarge ? (large) : controlSize == QAquaSizeSmall ? (small) : (mini)) // same as return SIZE(...) but optimized #define return_SIZE(large, small, mini) \ do { \ static const int sizes[] = { (large), (small), (mini) }; \ return sizes[controlSize]; \ } while (0) static int getControlSize(const QStyleOption *option, const QWidget *widget) { if (option) { if (option->state & (QStyle::State_Small | QStyle::State_Mini)) return (option->state & QStyle::State_Mini) ? QAquaSizeMini : QAquaSizeSmall; } else if (widget) { switch (QMacStyle::widgetSizePolicy(widget)) { case QMacStyle::SizeSmall: return QAquaSizeSmall; case QMacStyle::SizeMini: return QAquaSizeMini; default: break; } } return QAquaSizeLarge; } static inline bool isTreeView(const QWidget *widget) { return (widget && widget->parentWidget() && (qobject_cast(widget->parentWidget()) #ifdef QT3_SUPPORT || widget->parentWidget()->inherits("Q3ListView") #endif )); } QString qt_mac_removeMnemonics(const QString &original) { QString returnText(original.size(), 0); int finalDest = 0; int currPos = 0; int l = original.length(); while (l) { if (original.at(currPos) == QLatin1Char('&') && (l == 1 || original.at(currPos + 1) != QLatin1Char('&'))) { ++currPos; --l; if (l == 0) break; } returnText[finalDest] = original.at(currPos); ++currPos; ++finalDest; --l; } returnText.truncate(finalDest); return returnText; } static inline ThemeTabDirection getTabDirection(QTabBar::Shape shape) { ThemeTabDirection ttd; switch (shape) { case QTabBar::RoundedSouth: case QTabBar::TriangularSouth: ttd = kThemeTabSouth; break; default: // Added to remove the warning, since all values are taken care of, really! case QTabBar::RoundedNorth: case QTabBar::TriangularNorth: ttd = kThemeTabNorth; break; case QTabBar::RoundedWest: case QTabBar::TriangularWest: ttd = kThemeTabWest; break; case QTabBar::RoundedEast: case QTabBar::TriangularEast: ttd = kThemeTabEast; break; } return ttd; } class QMacStylePrivate : public QObject { Q_OBJECT public: QMacStylePrivate(QMacStyle *style); // Stuff from QAquaAnimate: bool addWidget(QWidget *); void removeWidget(QWidget *); enum Animates { AquaPushButton, AquaProgressBar, AquaListViewItemOpen }; bool animatable(Animates, const QWidget *) const; void stopAnimate(Animates, QWidget *); void startAnimate(Animates, QWidget *); static ThemeDrawState getDrawState(QStyle::State flags); QAquaWidgetSize aquaSizeConstrain(const QStyleOption *option, const QWidget *widg, QStyle::ContentsType ct = QStyle::CT_CustomBase, QSize szHint=QSize(-1, -1), QSize *insz = 0) const; void getSliderInfo(QStyle::ComplexControl cc, const QStyleOptionSlider *slider, HIThemeTrackDrawInfo *tdi, const QWidget *needToRemoveMe); bool doAnimate(Animates); inline int animateSpeed(Animates) const { return 33; } // Utility functions void drawColorlessButton(const HIRect &macRect, HIThemeButtonDrawInfo *bdi, QPainter *p, const QStyleOption *opt) const; QSize pushButtonSizeFromContents(const QStyleOptionButton *btn) const; HIRect pushButtonContentBounds(const QStyleOptionButton *btn, const HIThemeButtonDrawInfo *bdi) const; void initComboboxBdi(const QStyleOptionComboBox *combo, HIThemeButtonDrawInfo *bdi, const QWidget *widget, const ThemeDrawState &tds); static HIRect comboboxInnerBounds(const HIRect &outerBounds, int buttonKind); static QRect comboboxEditBounds(const QRect &outerBounds, const HIThemeButtonDrawInfo &bdi); static void drawCombobox(const HIRect &outerBounds, const HIThemeButtonDrawInfo &bdi, QPainter *p); static void drawTableHeader(const HIRect &outerBounds, bool drawTopBorder, bool drawLeftBorder, const HIThemeButtonDrawInfo &bdi, QPainter *p); bool contentFitsInPushButton(const QStyleOptionButton *btn, HIThemeButtonDrawInfo *bdi, ThemeButtonKind buttonKindToCheck) const; void initHIThemePushButton(const QStyleOptionButton *btn, const QWidget *widget, const ThemeDrawState tds, HIThemeButtonDrawInfo *bdi) const; QPixmap generateBackgroundPattern() const; protected: bool eventFilter(QObject *, QEvent *); void timerEvent(QTimerEvent *); private slots: void startAnimationTimer(); public: QPointer defaultButton; //default push buttons int timerID; QList > progressBars; //existing progress bars that need animation struct ButtonState { int frame; enum { ButtonDark, ButtonLight } dir; } buttonState; UInt8 progressFrame; QPointer focusWidget; CFAbsoluteTime defaultButtonStart; QMacStyle *q; bool mouseDown; }; QT_BEGIN_INCLUDE_NAMESPACE #include "qmacstyle_mac.moc" QT_END_INCLUDE_NAMESPACE /***************************************************************************** External functions *****************************************************************************/ extern CGContextRef qt_mac_cg_context(const QPaintDevice *); //qpaintdevice_mac.cpp extern QRegion qt_mac_convert_mac_region(HIShapeRef); //qregion_mac.cpp void qt_mac_dispose_rgn(RgnHandle r); //qregion_mac.cpp extern QPaintDevice *qt_mac_safe_pdev; //qapplication_mac.cpp /***************************************************************************** QMacCGStyle globals *****************************************************************************/ const int qt_mac_hitheme_version = 0; //the HITheme version we speak const int macItemFrame = 2; // menu item frame width const int macItemHMargin = 3; // menu item hor text margin const int macItemVMargin = 2; // menu item ver text margin const int macRightBorder = 12; // right border on mac const ThemeWindowType QtWinType = kThemeDocumentWindow; // Window type we use for QTitleBar. QPixmap *qt_mac_backgroundPattern = 0; // stores the standard widget background. /***************************************************************************** QMacCGStyle utility functions *****************************************************************************/ static inline int qt_mac_hitheme_tab_version() { return 1; } static inline HIRect qt_hirectForQRect(const QRect &convertRect, const QRect &rect = QRect()) { return CGRectMake(convertRect.x() + rect.x(), convertRect.y() + rect.y(), convertRect.width() - rect.width(), convertRect.height() - rect.height()); } static inline const QRect qt_qrectForHIRect(const HIRect &hirect) { return QRect(QPoint(int(hirect.origin.x), int(hirect.origin.y)), QSize(int(hirect.size.width), int(hirect.size.height))); } inline bool qt_mac_is_metal(const QWidget *w) { for (; w; w = w->parentWidget()) { if (w->testAttribute(Qt::WA_MacBrushedMetal)) return true; if (w->isWindow() && w->testAttribute(Qt::WA_WState_Created)) { // If not created will fall through to the opaque check and be fine anyway. return macWindowIsTextured(qt_mac_window_for(w)); } if (w->d_func()->isOpaque) break; } return false; } static int qt_mac_aqua_get_metric(ThemeMetric met) { SInt32 ret; GetThemeMetric(met, &ret); return ret; } static QSize qt_aqua_get_known_size(QStyle::ContentsType ct, const QWidget *widg, QSize szHint, QAquaWidgetSize sz) { QSize ret(-1, -1); if (sz != QAquaSizeSmall && sz != QAquaSizeLarge && sz != QAquaSizeMini) { qDebug("Not sure how to return this..."); return ret; } if ((widg && widg->testAttribute(Qt::WA_SetFont)) || !QApplication::desktopSettingsAware()) { // If you're using a custom font and it's bigger than the default font, // then no constraints for you. If you are smaller, we can try to help you out QFont font = qt_app_fonts_hash()->value(widg->metaObject()->className(), QFont()); if (widg->font().pointSize() > font.pointSize()) return ret; } if (ct == QStyle::CT_CustomBase && widg) { if (qobject_cast(widg)) ct = QStyle::CT_PushButton; else if (qobject_cast(widg)) ct = QStyle::CT_RadioButton; else if (qobject_cast(widg)) ct = QStyle::CT_CheckBox; else if (qobject_cast(widg)) ct = QStyle::CT_ComboBox; else if (qobject_cast(widg)) ct = QStyle::CT_ToolButton; else if (qobject_cast(widg)) ct = QStyle::CT_Slider; else if (qobject_cast(widg)) ct = QStyle::CT_ProgressBar; else if (qobject_cast(widg)) ct = QStyle::CT_LineEdit; else if (qobject_cast(widg) #ifdef QT3_SUPPORT || widg->inherits("Q3Header") #endif ) ct = QStyle::CT_HeaderSection; else if (qobject_cast(widg) #ifdef QT3_SUPPORT || widg->inherits("Q3MenuBar") #endif ) ct = QStyle::CT_MenuBar; else if (qobject_cast(widg)) ct = QStyle::CT_SizeGrip; else return ret; } switch (ct) { case QStyle::CT_PushButton: { const QPushButton *psh = qobject_cast(widg); // If this comparison is false, then the widget was not a push button. // This is bad and there's very little we can do since we were requested to find a // sensible size for a widget that pretends to be a QPushButton but is not. if(psh) { QString buttonText = qt_mac_removeMnemonics(psh->text()); if (buttonText.contains(QLatin1Char('\n'))) ret = QSize(-1, -1); else if (sz == QAquaSizeLarge) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricPushButtonHeight)); else if (sz == QAquaSizeSmall) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricSmallPushButtonHeight)); else if (sz == QAquaSizeMini) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricMiniPushButtonHeight)); if (!psh->icon().isNull()){ // If the button got an icon, and the icon is larger than the // button, we can't decide on a default size ret.setWidth(-1); if (ret.height() < psh->iconSize().height()) ret.setHeight(-1); } else if (buttonText == QLatin1String("OK") || buttonText == QLatin1String("Cancel")){ // Aqua Style guidelines restrict the size of OK and Cancel buttons to 68 pixels. // However, this doesn't work for German, therefore only do it for English, // I suppose it would be better to do some sort of lookups for languages // that like to have really long words. ret.setWidth(77 - 8); } } else { // The only sensible thing to do is to return whatever the style suggests... if (sz == QAquaSizeLarge) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricPushButtonHeight)); else if (sz == QAquaSizeSmall) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricSmallPushButtonHeight)); else if (sz == QAquaSizeMini) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricMiniPushButtonHeight)); else // Since there's no default size we return the large size... ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricPushButtonHeight)); } #if 0 //Not sure we are applying the rules correctly for RadioButtons/CheckBoxes --Sam } else if (ct == QStyle::CT_RadioButton) { QRadioButton *rdo = static_cast(widg); // Exception for case where multiline radio button text requires no size constrainment if (rdo->text().find('\n') != -1) return ret; if (sz == QAquaSizeLarge) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricRadioButtonHeight)); else if (sz == QAquaSizeSmall) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricSmallRadioButtonHeight)); else if (sz == QAquaSizeMini) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricMiniRadioButtonHeight)); } else if (ct == QStyle::CT_CheckBox) { if (sz == QAquaSizeLarge) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricCheckBoxHeight)); else if (sz == QAquaSizeSmall) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricSmallCheckBoxHeight)); else if (sz == QAquaSizeMini) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricMiniCheckBoxHeight)); #endif break; } case QStyle::CT_SizeGrip: if (sz == QAquaSizeLarge || sz == QAquaSizeSmall) { HIRect r; HIPoint p = { 0, 0 }; HIThemeGrowBoxDrawInfo gbi; gbi.version = 0; gbi.state = kThemeStateActive; gbi.kind = kHIThemeGrowBoxKindNormal; gbi.direction = QApplication::isRightToLeft() ? kThemeGrowLeft | kThemeGrowDown : kThemeGrowRight | kThemeGrowDown; gbi.size = sz == QAquaSizeSmall ? kHIThemeGrowBoxSizeSmall : kHIThemeGrowBoxSizeNormal; if (HIThemeGetGrowBoxBounds(&p, &gbi, &r) == noErr) ret = QSize(r.size.width, r.size.height); } break; case QStyle::CT_ComboBox: switch (sz) { case QAquaSizeLarge: ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricPopupButtonHeight)); break; case QAquaSizeSmall: ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricSmallPopupButtonHeight)); break; case QAquaSizeMini: ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricMiniPopupButtonHeight)); break; default: break; } break; case QStyle::CT_ToolButton: if (sz == QAquaSizeSmall) { int width = 0, height = 0; if (szHint == QSize(-1, -1)) { //just 'guess'.. const QToolButton *bt = qobject_cast(widg); // If this conversion fails then the widget was not what it claimed to be. if(bt) { if (!bt->icon().isNull()) { QSize iconSize = bt->iconSize(); QSize pmSize = bt->icon().actualSize(QSize(32, 32), QIcon::Normal); width = qMax(width, qMax(iconSize.width(), pmSize.width())); height = qMax(height, qMax(iconSize.height(), pmSize.height())); } if (!bt->text().isNull() && bt->toolButtonStyle() != Qt::ToolButtonIconOnly) { int text_width = bt->fontMetrics().width(bt->text()), text_height = bt->fontMetrics().height(); if (bt->toolButtonStyle() == Qt::ToolButtonTextUnderIcon) { width = qMax(width, text_width); height += text_height; } else { width += text_width; width = qMax(height, text_height); } } } else { // Let's return the size hint... width = szHint.width(); height = szHint.height(); } } else { width = szHint.width(); height = szHint.height(); } width = qMax(20, width + 5); //border height = qMax(20, height + 5); //border ret = QSize(width, height); } break; case QStyle::CT_Slider: { int w = -1; const QSlider *sld = qobject_cast(widg); // If this conversion fails then the widget was not what it claimed to be. if(sld) { if (sz == QAquaSizeLarge) { if (sld->orientation() == Qt::Horizontal) { w = qt_mac_aqua_get_metric(kThemeMetricHSliderHeight); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricHSliderTickHeight); } else { w = qt_mac_aqua_get_metric(kThemeMetricVSliderWidth); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricVSliderTickWidth); } } else if (sz == QAquaSizeSmall) { if (sld->orientation() == Qt::Horizontal) { w = qt_mac_aqua_get_metric(kThemeMetricSmallHSliderHeight); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricSmallHSliderTickHeight); } else { w = qt_mac_aqua_get_metric(kThemeMetricSmallVSliderWidth); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricSmallVSliderTickWidth); } } else if (sz == QAquaSizeMini) { if (sld->orientation() == Qt::Horizontal) { w = qt_mac_aqua_get_metric(kThemeMetricMiniHSliderHeight); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricMiniHSliderTickHeight); } else { w = qt_mac_aqua_get_metric(kThemeMetricMiniVSliderWidth); if (sld->tickPosition() != QSlider::NoTicks) w += qt_mac_aqua_get_metric(kThemeMetricMiniVSliderTickWidth); } } } else { // This is tricky, we were requested to find a size for a slider which is not // a slider. We don't know if this is vertical or horizontal or if we need to // have tick marks or not. // For this case we will return an horizontal slider without tick marks. w = qt_mac_aqua_get_metric(kThemeMetricHSliderHeight); w += qt_mac_aqua_get_metric(kThemeMetricHSliderTickHeight); } if (sld->orientation() == Qt::Horizontal) ret.setHeight(w); else ret.setWidth(w); break; } case QStyle::CT_ProgressBar: { int finalValue = -1; Qt::Orientation orient = Qt::Horizontal; if (const QProgressBar *pb = qobject_cast(widg)) orient = pb->orientation(); if (sz == QAquaSizeLarge) finalValue = qt_mac_aqua_get_metric(kThemeMetricLargeProgressBarThickness) + qt_mac_aqua_get_metric(kThemeMetricProgressBarShadowOutset); else finalValue = qt_mac_aqua_get_metric(kThemeMetricNormalProgressBarThickness) + qt_mac_aqua_get_metric(kThemeMetricSmallProgressBarShadowOutset); if (orient == Qt::Horizontal) ret.setHeight(finalValue); else ret.setWidth(finalValue); break; } case QStyle::CT_LineEdit: if (!widg || !qobject_cast(widg->parentWidget())) { //should I take into account the font dimentions of the lineedit? -Sam if (sz == QAquaSizeLarge) ret = QSize(-1, 22); else ret = QSize(-1, 19); } break; case QStyle::CT_HeaderSection: if (isTreeView(widg)) ret = QSize(-1, qt_mac_aqua_get_metric(kThemeMetricListHeaderHeight)); break; case QStyle::CT_MenuBar: if (sz == QAquaSizeLarge) { #ifndef QT_MAC_USE_COCOA SInt16 size; if (!GetThemeMenuBarHeight(&size)) ret = QSize(-1, size); #else ret = QSize(-1, [[NSApp mainMenu] menuBarHeight]); // In the qt_mac_set_native_menubar(false) case, // we come it here with a zero-height main menu, // preventing the in-window menu from displaying. // Use 22 pixels for the height, by observation. if (ret.height() <= 0) ret.setHeight(22); #endif } break; default: break; } return ret; } #if defined(QMAC_QAQUASTYLE_SIZE_CONSTRAIN) || defined(DEBUG_SIZE_CONSTRAINT) static QAquaWidgetSize qt_aqua_guess_size(const QWidget *widg, QSize large, QSize small, QSize mini) { if (large == QSize(-1, -1)) { if (small != QSize(-1, -1)) return QAquaSizeSmall; if (mini != QSize(-1, -1)) return QAquaSizeMini; return QAquaSizeUnknown; } else if (small == QSize(-1, -1)) { if (mini != QSize(-1, -1)) return QAquaSizeMini; return QAquaSizeLarge; } else if (mini == QSize(-1, -1)) { return QAquaSizeLarge; } #ifndef QT_NO_MAINWINDOW if (qobject_cast(widg->window()) || !qgetenv("QWIDGET_ALL_SMALL").isNull()) { //if (small.width() != -1 || small.height() != -1) return QAquaSizeSmall; } else if (!qgetenv("QWIDGET_ALL_MINI").isNull()) { return QAquaSizeMini; } #endif #if 0 /* Figure out which size we're closer to, I just hacked this in, I haven't tested it as it would probably look pretty strange to have some widgets big and some widgets small in the same window?? -Sam */ int large_delta=0; if (large.width() != -1) { int delta = large.width() - widg->width(); large_delta += delta * delta; } if (large.height() != -1) { int delta = large.height() - widg->height(); large_delta += delta * delta; } int small_delta=0; if (small.width() != -1) { int delta = small.width() - widg->width(); small_delta += delta * delta; } if (small.height() != -1) { int delta = small.height() - widg->height(); small_delta += delta * delta; } int mini_delta=0; if (mini.width() != -1) { int delta = mini.width() - widg->width(); mini_delta += delta * delta; } if (mini.height() != -1) { int delta = mini.height() - widg->height(); mini_delta += delta * delta; } if (mini_delta < small_delta && mini_delta < large_delta) return QAquaSizeMini; else if (small_delta < large_delta) return QAquaSizeSmall; #endif return QAquaSizeLarge; } #endif QAquaWidgetSize QMacStylePrivate::aquaSizeConstrain(const QStyleOption *option, const QWidget *widg, QStyle::ContentsType ct, QSize szHint, QSize *insz) const { #if defined(QMAC_QAQUASTYLE_SIZE_CONSTRAIN) || defined(DEBUG_SIZE_CONSTRAINT) if (option) { if (option->state & QStyle::State_Small) return QAquaSizeSmall; if (option->state & QStyle::State_Mini) return QAquaSizeMini; } if (!widg) { if (insz) *insz = QSize(); if (!qgetenv("QWIDGET_ALL_SMALL").isNull()) return QAquaSizeSmall; if (!qgetenv("QWIDGET_ALL_MINI").isNull()) return QAquaSizeMini; return QAquaSizeUnknown; } QSize large = qt_aqua_get_known_size(ct, widg, szHint, QAquaSizeLarge), small = qt_aqua_get_known_size(ct, widg, szHint, QAquaSizeSmall), mini = qt_aqua_get_known_size(ct, widg, szHint, QAquaSizeMini); bool guess_size = false; QAquaWidgetSize ret = QAquaSizeUnknown; QMacStyle::WidgetSizePolicy wsp = q->widgetSizePolicy(widg); if (wsp == QMacStyle::SizeDefault) guess_size = true; else if (wsp == QMacStyle::SizeMini) ret = QAquaSizeMini; else if (wsp == QMacStyle::SizeSmall) ret = QAquaSizeSmall; else if (wsp == QMacStyle::SizeLarge) ret = QAquaSizeLarge; if (guess_size) ret = qt_aqua_guess_size(widg, large, small, mini); QSize *sz = 0; if (ret == QAquaSizeSmall) sz = &small; else if (ret == QAquaSizeLarge) sz = &large; else if (ret == QAquaSizeMini) sz = &mini; if (insz) *insz = sz ? *sz : QSize(-1, -1); #ifdef DEBUG_SIZE_CONSTRAINT if (sz) { const char *size_desc = "Unknown"; if (sz == &small) size_desc = "Small"; else if (sz == &large) size_desc = "Large"; else if (sz == &mini) size_desc = "Mini"; qDebug("%s - %s: %s taken (%d, %d) [%d, %d]", widg ? widg->objectName().toLatin1().constData() : "*Unknown*", widg ? widg->metaObject()->className() : "*Unknown*", size_desc, widg->width(), widg->height(), sz->width(), sz->height()); } #endif return ret; #else if (insz) *insz = QSize(); Q_UNUSED(widg); Q_UNUSED(ct); Q_UNUSED(szHint); return QAquaSizeUnknown; #endif } /** Returns the free space awailable for contents inside the button (and not the size of the contents itself) */ HIRect QMacStylePrivate::pushButtonContentBounds(const QStyleOptionButton *btn, const HIThemeButtonDrawInfo *bdi) const { HIRect outerBounds = qt_hirectForQRect(btn->rect); // Adjust the bounds to correct for // carbon not calculating the content bounds fully correct if (bdi->kind == kThemePushButton || bdi->kind == kThemePushButtonSmall){ outerBounds.origin.y += PushButtonTopOffset; outerBounds.size.height -= PushButtonBottomOffset; } else if (bdi->kind == kThemePushButtonMini) { outerBounds.origin.y += PushButtonTopOffset; } HIRect contentBounds; HIThemeGetButtonContentBounds(&outerBounds, bdi, &contentBounds); return contentBounds; } /** Calculates the size of the button contents. This includes both the text and the icon. */ QSize QMacStylePrivate::pushButtonSizeFromContents(const QStyleOptionButton *btn) const { QSize csz(0, 0); QSize iconSize = btn->icon.isNull() ? QSize(0, 0) : (btn->iconSize + QSize(PushButtonContentPadding, 0)); QRect textRect = btn->text.isEmpty() ? QRect(0, 0, 1, 1) : btn->fontMetrics.boundingRect(QRect(), Qt::AlignCenter, btn->text); csz.setWidth(iconSize.width() + textRect.width() + ((btn->features & QStyleOptionButton::HasMenu) ? q->proxy()->pixelMetric(QStyle::PM_MenuButtonIndicator, btn, 0) : 0)); csz.setHeight(qMax(iconSize.height(), textRect.height())); return csz; } /** Checks if the actual contents of btn fits inside the free content bounds of 'buttonKindToCheck'. Meant as a helper function for 'initHIThemePushButton' for determining which button kind to use for drawing. */ bool QMacStylePrivate::contentFitsInPushButton(const QStyleOptionButton *btn, HIThemeButtonDrawInfo *bdi, ThemeButtonKind buttonKindToCheck) const { ThemeButtonKind tmp = bdi->kind; bdi->kind = buttonKindToCheck; QSize contentSize = pushButtonSizeFromContents(btn); QRect freeContentRect = qt_qrectForHIRect(pushButtonContentBounds(btn, bdi)); bdi->kind = tmp; return freeContentRect.contains(QRect(freeContentRect.x(), freeContentRect.y(), contentSize.width(), contentSize.height())); } /** Creates a HIThemeButtonDrawInfo structure that specifies the correct button kind and other details to use for drawing the given push button. Which button kind depends on the size of the button, the size of the contents, explicit user style settings, etc. */ void QMacStylePrivate::initHIThemePushButton(const QStyleOptionButton *btn, const QWidget *widget, const ThemeDrawState tds, HIThemeButtonDrawInfo *bdi) const { bool drawColorless = btn->palette.currentColorGroup() == QPalette::Active; ThemeDrawState tdsModified = tds; if (btn->state & QStyle::State_On) tdsModified = kThemeStatePressed; bdi->version = qt_mac_hitheme_version; bdi->state = tdsModified; bdi->value = kThemeButtonOff; if (drawColorless && tdsModified == kThemeStateInactive) bdi->state = kThemeStateActive; if (btn->state & QStyle::State_HasFocus) bdi->adornment = kThemeAdornmentFocus; else bdi->adornment = kThemeAdornmentNone; if (btn->features & (QStyleOptionButton::Flat)) { bdi->kind = kThemeBevelButton; } else { switch (aquaSizeConstrain(btn, widget)) { case QAquaSizeSmall: bdi->kind = kThemePushButtonSmall; break; case QAquaSizeMini: bdi->kind = kThemePushButtonMini; break; case QAquaSizeLarge: // ... We should honor if the user is explicit about using the // large button. But right now Qt will specify the large button // as default rather than QAquaSizeUnknown. // So we treat it like QAquaSizeUnknown // to get the dynamic choosing of button kind. case QAquaSizeUnknown: // Choose the button kind that closest match the button rect, but at the // same time displays the button contents without clipping. bdi->kind = kThemeBevelButton; if (btn->rect.width() >= BevelButtonW && btn->rect.height() >= BevelButtonH){ if (widget && widget->testAttribute(Qt::WA_MacVariableSize)) { if (btn->rect.height() <= MiniButtonH){ if (contentFitsInPushButton(btn, bdi, kThemePushButtonMini)) bdi->kind = kThemePushButtonMini; } else if (btn->rect.height() <= SmallButtonH){ if (contentFitsInPushButton(btn, bdi, kThemePushButtonSmall)) bdi->kind = kThemePushButtonSmall; } else if (contentFitsInPushButton(btn, bdi, kThemePushButton)) { bdi->kind = kThemePushButton; } } else { bdi->kind = kThemePushButton; } } } } } /** Creates a HIThemeButtonDrawInfo structure that specifies the correct button kind and other details to use for drawing the given combobox. Which button kind depends on the size of the combo, wether or not it is editable, explicit user style settings, etc. */ void QMacStylePrivate::initComboboxBdi(const QStyleOptionComboBox *combo, HIThemeButtonDrawInfo *bdi, const QWidget *widget, const ThemeDrawState &tds) { bdi->version = qt_mac_hitheme_version; bdi->adornment = kThemeAdornmentArrowLeftArrow; bdi->value = kThemeButtonOff; if (combo->state & QStyle::State_HasFocus) bdi->adornment = kThemeAdornmentFocus; bool drawColorless = combo->palette.currentColorGroup() == QPalette::Active && tds == kThemeStateInactive; if (combo->activeSubControls & QStyle::SC_ComboBoxArrow) bdi->state = kThemeStatePressed; else if (drawColorless) bdi->state = kThemeStateActive; else bdi->state = tds; QAquaWidgetSize aSize = aquaSizeConstrain(combo, widget); switch (aSize) { case QAquaSizeMini: bdi->kind = combo->editable ? ThemeButtonKind(kThemeComboBoxMini) : ThemeButtonKind(kThemePopupButtonMini); break; case QAquaSizeSmall: bdi->kind = combo->editable ? ThemeButtonKind(kThemeComboBoxSmall) : ThemeButtonKind(kThemePopupButtonSmall); break; case QAquaSizeUnknown: case QAquaSizeLarge: // Unless the user explicitly specified large buttons, determine the // kind by looking at the combox size. // ... specifying small and mini-buttons it not a current feature of // Qt (e.g. QWidget::getAttribute(WA_ButtonSize)). But when it is, add // an extra check here before using the mini and small buttons. int h = combo->rect.size().height(); if (combo->editable){ if (h < 21) bdi->kind = kThemeComboBoxMini; else if (h < 26) bdi->kind = kThemeComboBoxSmall; else bdi->kind = kThemeComboBox; } else { // Even if we specify that we want the kThemePopupButton, Carbon // will use the kThemePopupButtonSmall if the size matches. So we // do the same size check explicit to have the size of the inner // text field be correct. Therefore, do this even if the user specifies // the use of LargeButtons explicit. if (h < 21) bdi->kind = kThemePopupButtonMini; else if (h < 26) bdi->kind = kThemePopupButtonSmall; else bdi->kind = kThemePopupButton; } break; } } /** Carbon draws comboboxes (and other views) outside the rect given as argument. Use this function to obtain the corresponding inner rect for drawing the same combobox so that it stays inside the given outerBounds. */ HIRect QMacStylePrivate::comboboxInnerBounds(const HIRect &outerBounds, int buttonKind) { HIRect innerBounds = outerBounds; // Carbon draw parts of the view outside the rect. // So make the rect a bit smaller to compensate // (I wish HIThemeGetButtonBackgroundBounds worked) switch (buttonKind){ case kThemePopupButton: innerBounds.origin.x += 2; innerBounds.origin.y += 3; innerBounds.size.width -= 5; innerBounds.size.height -= 6; break; case kThemePopupButtonSmall: innerBounds.origin.x += 3; innerBounds.origin.y += 3; innerBounds.size.width -= 6; innerBounds.size.height -= 7; break; case kThemePopupButtonMini: innerBounds.origin.x += 2; innerBounds.origin.y += 2; innerBounds.size.width -= 5; innerBounds.size.height -= 6; break; case kThemeComboBox: innerBounds.origin.x += 3; innerBounds.origin.y += 3; innerBounds.size.width -= 6; innerBounds.size.height -= 6; break; case kThemeComboBoxSmall: innerBounds.origin.x += 3; innerBounds.origin.y += 3; innerBounds.size.width -= 7; innerBounds.size.height -= 8; break; case kThemeComboBoxMini: innerBounds.origin.x += 3; innerBounds.origin.y += 3; innerBounds.size.width -= 4; innerBounds.size.height -= 8; break; default: break; } return innerBounds; } /** Inside a combobox Qt places a line edit widget. The size of this widget should depend on the kind of combobox we choose to draw. This function calculates and returns this size. */ QRect QMacStylePrivate::comboboxEditBounds(const QRect &outerBounds, const HIThemeButtonDrawInfo &bdi) { QRect ret = outerBounds; switch (bdi.kind){ case kThemeComboBox: ret.adjust(5, 8, -21, -4); break; case kThemeComboBoxSmall: ret.adjust(4, 5, -18, 0); ret.setHeight(16); break; case kThemeComboBoxMini: ret.adjust(4, 5, -16, 0); ret.setHeight(13); break; case kThemePopupButton: ret.adjust(10, 3, -23, -3); break; case kThemePopupButtonSmall: ret.adjust(9, 3, -20, -3); break; case kThemePopupButtonMini: ret.adjust(8, 3, -19, 0); ret.setHeight(13); break; } return ret; } /** Carbon comboboxes don't scale (sight). If the size of the combo suggest a scaled version, create it manually by drawing a small Carbon combo onto a pixmap (use pixmap cache), chop it up, and copy it back onto the widget. Othervise, draw the combobox supplied by Carbon directly. */ void QMacStylePrivate::drawCombobox(const HIRect &outerBounds, const HIThemeButtonDrawInfo &bdi, QPainter *p) { if (!(bdi.kind == kThemeComboBox && outerBounds.size.height > 28)){ // We have an unscaled combobox, or popup-button; use Carbon directly. HIRect innerBounds = QMacStylePrivate::comboboxInnerBounds(outerBounds, bdi.kind); HIThemeDrawButton(&innerBounds, &bdi, QMacCGContext(p), kHIThemeOrientationNormal, 0); } else { QPixmap buffer; QString key = QString(QLatin1String("$qt_cbox%1-%2")).arg(int(bdi.state)).arg(int(bdi.adornment)); if (!QPixmapCache::find(key, buffer)) { HIRect innerBoundsSmallCombo = {{3, 3}, {29, 25}}; buffer = QPixmap(35, 28); buffer.fill(Qt::transparent); QPainter buffPainter(&buffer); HIThemeDrawButton(&innerBoundsSmallCombo, &bdi, QMacCGContext(&buffPainter), kHIThemeOrientationNormal, 0); buffPainter.end(); QPixmapCache::insert(key, buffer); } const int bwidth = 20; const int fwidth = 10; const int fheight = 10; int w = qRound(outerBounds.size.width); int h = qRound(outerBounds.size.height); int bstart = w - bwidth; int blower = fheight + 1; int flower = h - fheight; int sheight = flower - fheight; int center = qRound(outerBounds.size.height + outerBounds.origin.y) / 2; // Draw upper and lower gap p->drawPixmap(fwidth, 0, bstart - fwidth, fheight, buffer, fwidth, 0, 1, fheight); p->drawPixmap(fwidth, flower, bstart - fwidth, fheight, buffer, fwidth, buffer.height() - fheight, 1, fheight); // Draw left and right gap. Right gap is drawn top and bottom separatly p->drawPixmap(0, fheight, fwidth, sheight, buffer, 0, fheight, fwidth, 1); p->drawPixmap(bstart, fheight, bwidth, center - fheight, buffer, buffer.width() - bwidth, fheight - 1, bwidth, 1); p->drawPixmap(bstart, center, bwidth, sheight / 2, buffer, buffer.width() - bwidth, fheight + 6, bwidth, 1); // Draw arrow p->drawPixmap(bstart, center - 4, bwidth - 3, 6, buffer, buffer.width() - bwidth, fheight, bwidth - 3, 6); // Draw corners p->drawPixmap(0, 0, fwidth, fheight, buffer, 0, 0, fwidth, fheight); p->drawPixmap(bstart, 0, bwidth, fheight, buffer, buffer.width() - bwidth, 0, bwidth, fheight); p->drawPixmap(0, flower, fwidth, fheight, buffer, 0, buffer.height() - fheight, fwidth, fheight); p->drawPixmap(bstart, h - blower, bwidth, blower, buffer, buffer.width() - bwidth, buffer.height() - blower, bwidth, blower); } } /** Carbon tableheaders don't scale (sight). So create it manually by drawing a small Carbon header onto a pixmap (use pixmap cache), chop it up, and copy it back onto the widget. */ void QMacStylePrivate::drawTableHeader(const HIRect &outerBounds, bool drawTopBorder, bool drawLeftBorder, const HIThemeButtonDrawInfo &bdi, QPainter *p) { static SInt32 headerHeight = 0; static OSStatus err = GetThemeMetric(kThemeMetricListHeaderHeight, &headerHeight); Q_UNUSED(err); QPixmap buffer; QString key = QString(QLatin1String("$qt_tableh%1-%2-%3")).arg(int(bdi.state)).arg(int(bdi.adornment)).arg(int(bdi.value)); if (!QPixmapCache::find(key, buffer)) { HIRect headerNormalRect = {{0., 0.}, {16., CGFloat(headerHeight)}}; buffer = QPixmap(headerNormalRect.size.width, headerNormalRect.size.height); buffer.fill(Qt::transparent); QPainter buffPainter(&buffer); HIThemeDrawButton(&headerNormalRect, &bdi, QMacCGContext(&buffPainter), kHIThemeOrientationNormal, 0); buffPainter.end(); QPixmapCache::insert(key, buffer); } const int buttonw = qRound(outerBounds.size.width); const int buttonh = qRound(outerBounds.size.height); const int framew = 1; const int frameh_n = 4; const int frameh_s = 3; const int transh = buffer.height() - frameh_n - frameh_s; int center = buttonh - frameh_s - int(transh / 2.0f) + 1; // Align bottom; int skipTopBorder = 0; if (!drawTopBorder) skipTopBorder = 1; p->translate(outerBounds.origin.x, outerBounds.origin.y); p->drawPixmap(QRect(QRect(0, -skipTopBorder, buttonw - framew , frameh_n)), buffer, QRect(framew, 0, 1, frameh_n)); p->drawPixmap(QRect(0, buttonh - frameh_s, buttonw - framew, frameh_s), buffer, QRect(framew, buffer.height() - frameh_s, 1, frameh_s)); // Draw upper and lower center blocks p->drawPixmap(QRect(0, frameh_n - skipTopBorder, buttonw - framew, center - frameh_n + skipTopBorder), buffer, QRect(framew, frameh_n, 1, 1)); p->drawPixmap(QRect(0, center, buttonw - framew, buttonh - center - frameh_s), buffer, QRect(framew, buffer.height() - frameh_s, 1, 1)); // Draw right center block borders p->drawPixmap(QRect(buttonw - framew, frameh_n - skipTopBorder, framew, center - frameh_n), buffer, QRect(buffer.width() - framew, frameh_n, framew, 1)); p->drawPixmap(QRect(buttonw - framew, center, framew, buttonh - center - 1), buffer, QRect(buffer.width() - framew, buffer.height() - frameh_s, framew, 1)); // Draw right corners p->drawPixmap(QRect(buttonw - framew, -skipTopBorder, framew, frameh_n), buffer, QRect(buffer.width() - framew, 0, framew, frameh_n)); p->drawPixmap(QRect(buttonw - framew, buttonh - frameh_s, framew, frameh_s), buffer, QRect(buffer.width() - framew, buffer.height() - frameh_s, framew, frameh_s)); // Draw center transition block p->drawPixmap(QRect(0, center - qRound(transh / 2.0f), buttonw - framew, buffer.height() - frameh_n - frameh_s), buffer, QRect(framew, frameh_n + 1, 1, transh)); // Draw right center transition block border p->drawPixmap(QRect(buttonw - framew, center - qRound(transh / 2.0f), framew, buffer.height() - frameh_n - frameh_s), buffer, QRect(buffer.width() - framew, frameh_n + 1, framew, transh)); if (drawLeftBorder){ // Draw left center block borders p->drawPixmap(QRect(0, frameh_n - skipTopBorder, framew, center - frameh_n + skipTopBorder), buffer, QRect(0, frameh_n, framew, 1)); p->drawPixmap(QRect(0, center, framew, buttonh - center - 1), buffer, QRect(0, buffer.height() - frameh_s, framew, 1)); // Draw left corners p->drawPixmap(QRect(0, -skipTopBorder, framew, frameh_n), buffer, QRect(0, 0, framew, frameh_n)); p->drawPixmap(QRect(0, buttonh - frameh_s, framew, frameh_s), buffer, QRect(0, buffer.height() - frameh_s, framew, frameh_s)); // Draw left center transition block border p->drawPixmap(QRect(0, center - qRound(transh / 2.0f), framew, buffer.height() - frameh_n - frameh_s), buffer, QRect(0, frameh_n + 1, framew, transh)); } p->translate(-outerBounds.origin.x, -outerBounds.origin.y); } /* Returns cutoff sizes for scroll bars. thumbIndicatorCutoff is the smallest size where the thumb indicator is drawn. scrollButtonsCutoff is the smallest size where the up/down buttons is drawn. */ enum ScrollBarCutoffType { thumbIndicatorCutoff = 0, scrollButtonsCutoff = 1 }; static int scrollButtonsCutoffSize(ScrollBarCutoffType cutoffType, QMacStyle::WidgetSizePolicy widgetSize) { // Mini scroll bars do not exist as of version 10.4. if (widgetSize == QMacStyle::SizeMini) return 0; const int sizeIndex = (widgetSize == QMacStyle::SizeSmall) ? 1 : 0; static const int sizeTable[2][2] = { { 61, 56 }, { 49, 44 } }; return sizeTable[sizeIndex][cutoffType]; } void QMacStylePrivate::getSliderInfo(QStyle::ComplexControl cc, const QStyleOptionSlider *slider, HIThemeTrackDrawInfo *tdi, const QWidget *needToRemoveMe) { memset(tdi, 0, sizeof(HIThemeTrackDrawInfo)); // We don't get it all for some reason or another... tdi->version = qt_mac_hitheme_version; tdi->reserved = 0; tdi->filler1 = 0; bool isScrollbar = (cc == QStyle::CC_ScrollBar); switch (aquaSizeConstrain(0, needToRemoveMe)) { case QAquaSizeUnknown: case QAquaSizeLarge: if (isScrollbar) tdi->kind = kThemeMediumScrollBar; else tdi->kind = kThemeMediumSlider; break; case QAquaSizeMini: if (isScrollbar) tdi->kind = kThemeSmallScrollBar; // should be kThemeMiniScrollBar, but not implemented else tdi->kind = kThemeMiniSlider; break; case QAquaSizeSmall: if (isScrollbar) tdi->kind = kThemeSmallScrollBar; else tdi->kind = kThemeSmallSlider; break; } tdi->bounds = qt_hirectForQRect(slider->rect); tdi->min = slider->minimum; tdi->max = slider->maximum; tdi->value = slider->sliderPosition; tdi->attributes = kThemeTrackShowThumb; if (slider->upsideDown) tdi->attributes |= kThemeTrackRightToLeft; if (slider->orientation == Qt::Horizontal) { tdi->attributes |= kThemeTrackHorizontal; if (isScrollbar && slider->direction == Qt::RightToLeft) { if (!slider->upsideDown) tdi->attributes |= kThemeTrackRightToLeft; else tdi->attributes &= ~kThemeTrackRightToLeft; } } // Tiger broke reverse scroll bars so put them back and "fake it" if (isScrollbar && (tdi->attributes & kThemeTrackRightToLeft)) { tdi->attributes &= ~kThemeTrackRightToLeft; tdi->value = tdi->max - slider->sliderPosition; } tdi->enableState = (slider->state & QStyle::State_Enabled) ? kThemeTrackActive : kThemeTrackDisabled; if (!(slider->state & QStyle::State_Active)) tdi->enableState = kThemeTrackInactive; if (!isScrollbar) { if (slider->state & QStyle::QStyle::State_HasFocus) tdi->attributes |= kThemeTrackHasFocus; if (slider->tickPosition == QSlider::NoTicks || slider->tickPosition == QSlider::TicksBothSides) tdi->trackInfo.slider.thumbDir = kThemeThumbPlain; else if (slider->tickPosition == QSlider::TicksAbove) tdi->trackInfo.slider.thumbDir = kThemeThumbUpward; else tdi->trackInfo.slider.thumbDir = kThemeThumbDownward; } else { tdi->trackInfo.scrollbar.viewsize = slider->pageStep; } } #endif QMacStylePrivate::QMacStylePrivate(QMacStyle *style) : timerID(-1), progressFrame(0), q(style), mouseDown(false) { defaultButtonStart = CFAbsoluteTimeGetCurrent(); memset(&buttonState, 0, sizeof(ButtonState)); if (ptrHIShapeGetBounds == 0) { QLibrary library(QLatin1String("/System/Library/Frameworks/Carbon.framework/Carbon")); library.setLoadHints(QLibrary::ExportExternalSymbolsHint); ptrHIShapeGetBounds = reinterpret_cast(library.resolve("HIShapeGetBounds")); } } bool QMacStylePrivate::animatable(QMacStylePrivate::Animates as, const QWidget *w) const { if (as == AquaPushButton) { QPushButton *pb = const_cast(static_cast(w)); if (w->window()->isActiveWindow() && pb && !mouseDown) { if (static_cast(w) != defaultButton) { // Changed on its own, update the value. const_cast(this)->stopAnimate(as, defaultButton); const_cast(this)->startAnimate(as, pb); } return true; } } else if (as == AquaProgressBar) { if (progressBars.contains((const_cast(w)))) return true; } return false; } void QMacStylePrivate::stopAnimate(QMacStylePrivate::Animates as, QWidget *w) { if (as == AquaPushButton && defaultButton) { QPushButton *tmp = defaultButton; defaultButton = 0; tmp->update(); } else if (as == AquaProgressBar) { progressBars.removeAll(w); } } void QMacStylePrivate::startAnimate(QMacStylePrivate::Animates as, QWidget *w) { if (as == AquaPushButton) defaultButton = static_cast(w); else if (as == AquaProgressBar) progressBars.append(w); startAnimationTimer(); } void QMacStylePrivate::startAnimationTimer() { if ((defaultButton || !progressBars.isEmpty()) && timerID <= -1) timerID = startTimer(animateSpeed(AquaListViewItemOpen)); } bool QMacStylePrivate::addWidget(QWidget *w) { //already knew of it if (static_cast(w) == defaultButton || progressBars.contains(static_cast(w))) return false; if (QPushButton *btn = qobject_cast(w)) { btn->installEventFilter(this); if (btn->isDefault() || (btn->autoDefault() && btn->hasFocus())) startAnimate(AquaPushButton, btn); return true; } else { bool isProgressBar = (qobject_cast(w) #ifdef QT3_SUPPORT || w->inherits("Q3ProgressBar") #endif ); if (isProgressBar) { w->installEventFilter(this); startAnimate(AquaProgressBar, w); return true; } } if (w->isWindow()) { w->installEventFilter(this); return true; } return false; } void QMacStylePrivate::removeWidget(QWidget *w) { QPushButton *btn = qobject_cast(w); if (btn && btn == defaultButton) { stopAnimate(AquaPushButton, btn); } else if (qobject_cast(w) #ifdef QT3_SUPPORT || w->inherits("Q3ProgressBar") #endif ) { stopAnimate(AquaProgressBar, w); } } ThemeDrawState QMacStylePrivate::getDrawState(QStyle::State flags) { ThemeDrawState tds = kThemeStateActive; if (flags & QStyle::State_Sunken) { tds = kThemeStatePressed; } else if (flags & QStyle::State_Active) { if (!(flags & QStyle::State_Enabled)) tds = kThemeStateUnavailable; } else { if (flags & QStyle::State_Enabled) tds = kThemeStateInactive; else tds = kThemeStateUnavailableInactive; } return tds; } void QMacStylePrivate::timerEvent(QTimerEvent *) { int animated = 0; if (defaultButton && defaultButton->isEnabled() && defaultButton->window()->isActiveWindow() && defaultButton->isVisibleTo(0) && (defaultButton->isDefault() || (defaultButton->autoDefault() && defaultButton->hasFocus())) && doAnimate(AquaPushButton)) { ++animated; defaultButton->update(); } if (!progressBars.isEmpty()) { int i = 0; while (i < progressBars.size()) { QWidget *maybeProgress = progressBars.at(i); if (!maybeProgress) { progressBars.removeAt(i); } else { if (QProgressBar *pb = qobject_cast(maybeProgress)) { if (pb->maximum() == 0 || pb->value() > 0 && pb->value() < pb->maximum()) { if (doAnimate(AquaProgressBar)) pb->update(); } } #ifdef QT3_SUPPORT else { // Watch me now... QVariant progress = maybeProgress->property("progress"); QVariant totalSteps = maybeProgress->property("totalSteps"); if (progress.isValid() && totalSteps.isValid()) { int intProgress = progress.toInt(); int intTotalSteps = totalSteps.toInt(); if (intTotalSteps == 0 || intProgress > 0 && intProgress < intTotalSteps) { if (doAnimate(AquaProgressBar)) maybeProgress->update(); } } } #endif ++i; } } if (i > 0) { ++progressFrame; animated += i; } } if (animated <= 0) { killTimer(timerID); timerID = -1; } } bool QMacStylePrivate::eventFilter(QObject *o, QEvent *e) { //animate if (QProgressBar *pb = qobject_cast(o)) { switch (e->type()) { default: break; case QEvent::Show: if (!progressBars.contains(pb)) startAnimate(AquaProgressBar, pb); break; case QEvent::Destroy: case QEvent::Hide: progressBars.removeAll(pb); } } else if (QPushButton *btn = qobject_cast(o)) { switch (e->type()) { default: break; case QEvent::FocusIn: if (btn->autoDefault()) startAnimate(AquaPushButton, btn); break; case QEvent::Destroy: case QEvent::Hide: if (btn == defaultButton) stopAnimate(AquaPushButton, btn); break; case QEvent::MouseButtonPress: // It is very confusing to keep the button pulsing, so just stop the animation. if (static_cast(e)->button() == Qt::LeftButton) mouseDown = true; stopAnimate(AquaPushButton, btn); break; case QEvent::MouseButtonRelease: if (static_cast(e)->button() == Qt::LeftButton) mouseDown = false; // fall through case QEvent::FocusOut: case QEvent::Show: case QEvent::WindowActivate: { QList list = qFindChildren(btn->window()); for (int i = 0; i < list.size(); ++i) { QPushButton *pBtn = list.at(i); if ((e->type() == QEvent::FocusOut && (pBtn->isDefault() || (pBtn->autoDefault() && pBtn->hasFocus())) && pBtn != btn) || ((e->type() == QEvent::Show || e->type() == QEvent::MouseButtonRelease || e->type() == QEvent::WindowActivate) && pBtn->isDefault())) { if (pBtn->window()->isActiveWindow()) { startAnimate(AquaPushButton, pBtn); } break; } } break; } } } return false; } bool QMacStylePrivate::doAnimate(QMacStylePrivate::Animates as) { if (as == AquaPushButton) { } else if (as == AquaProgressBar) { // something for later... } else if (as == AquaListViewItemOpen) { // To be revived later... } return true; } void QMacStylePrivate::drawColorlessButton(const HIRect &macRect, HIThemeButtonDrawInfo *bdi, QPainter *p, const QStyleOption *opt) const { int xoff = 0, yoff = 0, extraWidth = 0, extraHeight = 0, finalyoff = 0; const QStyleOptionComboBox *combo = qstyleoption_cast(opt); int width = int(macRect.size.width) + extraWidth; int height = int(macRect.size.height) + extraHeight; if (width <= 0 || height <= 0) return; // nothing to draw QString key = QLatin1String("$qt_mac_style_ctb_") + QString::number(bdi->kind) + QLatin1Char('_') + QString::number(bdi->value) + QLatin1Char('_') + QString::number(width) + QLatin1Char('_') + QString::number(height); QPixmap pm; if (!QPixmapCache::find(key, pm)) { QPixmap activePixmap(width, height); activePixmap.fill(Qt::transparent); { if (combo){ // Carbon combos don't scale. Therefore we draw it // ourselves, if a scaled version is needed. QPainter tmpPainter(&activePixmap); QMacStylePrivate::drawCombobox(macRect, *bdi, &tmpPainter); } else { QMacCGContext cg(&activePixmap); HIRect newRect = CGRectMake(xoff, yoff, macRect.size.width, macRect.size.height); HIThemeDrawButton(&newRect, bdi, cg, kHIThemeOrientationNormal, 0); } } if (!combo && bdi->value == kThemeButtonOff) { pm = activePixmap; } else if (combo) { QImage image = activePixmap.toImage(); for (int y = 0; y < height; ++y) { QRgb *scanLine = reinterpret_cast(image.scanLine(y)); for (int x = 0; x < width; ++x) { QRgb &pixel = scanLine[x]; int darkest = qRed(pixel); int mid = qGreen(pixel); int lightest = qBlue(pixel); if (darkest > mid) qSwap(darkest, mid); if (mid > lightest) qSwap(mid, lightest); if (darkest > mid) qSwap(darkest, mid); int gray = (mid + 2 * lightest) / 3; pixel = qRgba(gray, gray, gray, qAlpha(pixel)); } } pm = QPixmap::fromImage(image); } else { QImage activeImage = activePixmap.toImage(); QImage colorlessImage; { QPixmap colorlessPixmap(width, height); colorlessPixmap.fill(Qt::transparent); QMacCGContext cg(&colorlessPixmap); HIRect newRect = CGRectMake(xoff, yoff, macRect.size.width, macRect.size.height); int oldValue = bdi->value; bdi->value = kThemeButtonOff; HIThemeDrawButton(&newRect, bdi, cg, kHIThemeOrientationNormal, 0); bdi->value = oldValue; colorlessImage = colorlessPixmap.toImage(); } for (int y = 0; y < height; ++y) { QRgb *colorlessScanLine = reinterpret_cast(colorlessImage.scanLine(y)); const QRgb *activeScanLine = reinterpret_cast(activeImage.scanLine(y)); for (int x = 0; x < width; ++x) { QRgb &colorlessPixel = colorlessScanLine[x]; QRgb activePixel = activeScanLine[x]; if (activePixel != colorlessPixel) { int max = qMax(qMax(qRed(activePixel), qGreen(activePixel)), qBlue(activePixel)); QRgb newPixel = qRgba(max, max, max, qAlpha(activePixel)); if (qGray(newPixel) < qGray(colorlessPixel) || qAlpha(newPixel) > qAlpha(colorlessPixel)) colorlessPixel = newPixel; } } } pm = QPixmap::fromImage(colorlessImage); } QPixmapCache::insert(key, pm); } p->drawPixmap(int(macRect.origin.x), int(macRect.origin.y) + finalyoff, width, height, pm); } QMacStyle::QMacStyle() : QWindowsStyle() { d = new QMacStylePrivate(this); } QMacStyle::~QMacStyle() { delete qt_mac_backgroundPattern; qt_mac_backgroundPattern = 0; delete d; } /*! \internal Generates the standard widget background pattern. */ QPixmap QMacStylePrivate::generateBackgroundPattern() const { QPixmap px(4, 4); QMacCGContext cg(&px); HIThemeSetFill(kThemeBrushDialogBackgroundActive, 0, cg, kHIThemeOrientationNormal); const CGRect cgRect = CGRectMake(0, 0, px.width(), px.height()); CGContextFillRect(cg, cgRect); return px; } /*! \internal Fills the given \a rect with the pattern stored in \a brush. As an optimization, HIThemeSetFill us used directly if we are filling with the standard background. */ void qt_mac_fill_background(QPainter *painter, const QRegion &rgn, const QBrush &brush) { QPoint dummy; const QPaintDevice *target = painter->device(); const QPaintDevice *redirected = QPainter::redirected(target, &dummy); const bool usePainter = redirected && redirected != target; if (!usePainter && qt_mac_backgroundPattern && qt_mac_backgroundPattern->cacheKey() == brush.texture().cacheKey()) { painter->setClipRegion(rgn); CGContextRef cg = qt_mac_cg_context(target); CGContextSaveGState(cg); HIThemeSetFill(kThemeBrushDialogBackgroundActive, 0, cg, kHIThemeOrientationInverted); const QVector &rects = rgn.rects(); for (int i = 0; i < rects.size(); ++i) { const QRect rect(rects.at(i)); // Anchor the pattern to the top so it stays put when the window is resized. CGContextSetPatternPhase(cg, CGSizeMake(rect.width(), rect.height())); CGRect mac_rect = CGRectMake(rect.x(), rect.y(), rect.width(), rect.height()); CGContextFillRect(cg, mac_rect); } CGContextRestoreGState(cg); } else { const QRect rect(rgn.boundingRect()); painter->setClipRegion(rgn); painter->drawTiledPixmap(rect, brush.texture(), rect.topLeft()); } } void QMacStyle::polish(QPalette &pal) { if (!qt_mac_backgroundPattern) { if (!qApp) return; qt_mac_backgroundPattern = new QPixmap(d->generateBackgroundPattern()); } QColor pc(Qt::black); pc = qcolorForTheme(kThemeBrushDialogBackgroundActive); QBrush background(pc, *qt_mac_backgroundPattern); pal.setBrush(QPalette::All, QPalette::Window, background); pal.setBrush(QPalette::All, QPalette::Button, background); QCFString theme; const OSErr err = CopyThemeIdentifier(&theme); if (err == noErr && CFStringCompare(theme, kThemeAppearanceAquaGraphite, 0) == kCFCompareEqualTo) { pal.setBrush(QPalette::All, QPalette::AlternateBase, QColor(240, 240, 240)); } else { pal.setBrush(QPalette::All, QPalette::AlternateBase, QColor(237, 243, 254)); } } void QMacStyle::polish(QApplication *) { } void QMacStyle::unpolish(QApplication *) { } void QMacStyle::polish(QWidget* w) { d->addWidget(w); if (qt_mac_is_metal(w) && !w->testAttribute(Qt::WA_SetPalette)) { // Set a clear brush so that the metal shines through. QPalette pal = w->palette(); QBrush background(Qt::transparent); pal.setBrush(QPalette::All, QPalette::Window, background); pal.setBrush(QPalette::All, QPalette::Button, background); w->setPalette(pal); w->setAttribute(Qt::WA_SetPalette, false); } if (qobject_cast(w) || qobject_cast(w)) { w->setWindowOpacity(QSysInfo::MacintoshVersion >= QSysInfo::MV_10_5 ? 0.985 : 0.94); if (!w->testAttribute(Qt::WA_SetPalette)) { QPixmap px(64, 64); px.fill(Qt::white); HIThemeMenuDrawInfo mtinfo; mtinfo.version = qt_mac_hitheme_version; mtinfo.menuType = kThemeMenuTypePopUp; HIRect rect = CGRectMake(0, 0, px.width(), px.height()); HIThemeDrawMenuBackground(&rect, &mtinfo, QCFType(qt_mac_cg_context(&px)), kHIThemeOrientationNormal); QPalette pal = w->palette(); QBrush background(px); pal.setBrush(QPalette::All, QPalette::Window, background); pal.setBrush(QPalette::All, QPalette::Button, background); w->setPalette(pal); w->setAttribute(Qt::WA_SetPalette, false); } } if (QTabBar *tb = qobject_cast(w)) { if (tb->documentMode()) { w->setAttribute(Qt::WA_Hover); w->setFont(qt_app_fonts_hash()->value("QSmallFont", QFont())); QPalette p = w->palette(); p.setColor(QPalette::WindowText, QColor(17, 17, 17)); w->setPalette(p); } } QWindowsStyle::polish(w); if (QRubberBand *rubber = qobject_cast(w)) { rubber->setWindowOpacity(0.25); rubber->setAttribute(Qt::WA_PaintOnScreen, false); rubber->setAttribute(Qt::WA_NoSystemBackground, false); } } void QMacStyle::unpolish(QWidget* w) { d->removeWidget(w); if ((qobject_cast(w) || qt_mac_is_metal(w)) && !w->testAttribute(Qt::WA_SetPalette)) { QPalette pal = qApp->palette(w); w->setPalette(pal); w->setAttribute(Qt::WA_SetPalette, false); w->setWindowOpacity(1.0); } if (QComboBox *combo = qobject_cast(w)) { if (!combo->isEditable()) { if (QWidget *widget = combo->findChild()) widget->setWindowOpacity(1.0); } } if (QRubberBand *rubber = ::qobject_cast(w)) { rubber->setWindowOpacity(1.0); rubber->setAttribute(Qt::WA_PaintOnScreen, true); rubber->setAttribute(Qt::WA_NoSystemBackground, true); } if (QFocusFrame *frame = qobject_cast(w)) { frame->setAttribute(Qt::WA_NoSystemBackground, true); frame->setAutoFillBackground(true); } QWindowsStyle::unpolish(w); } int QMacStyle::pixelMetric(PixelMetric metric, const QStyleOption *opt, const QWidget *widget) const { int controlSize = getControlSize(opt, widget); SInt32 ret = 0; switch (metric) { case PM_TabCloseIndicatorWidth: case PM_TabCloseIndicatorHeight: ret = closeButtonSize; break; case PM_ToolBarIconSize: ret = proxy()->pixelMetric(PM_LargeIconSize); break; case PM_FocusFrameVMargin: case PM_FocusFrameHMargin: GetThemeMetric(kThemeMetricFocusRectOutset, &ret); break; case PM_DialogButtonsSeparator: ret = -5; break; case PM_DialogButtonsButtonHeight: { QSize sz; ret = d->aquaSizeConstrain(opt, 0, QStyle::CT_PushButton, QSize(-1, -1), &sz); if (sz == QSize(-1, -1)) ret = 32; else ret = sz.height(); break; } case PM_CheckListButtonSize: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricCheckBoxWidth, &ret); break; case QAquaSizeMini: GetThemeMetric(kThemeMetricMiniCheckBoxWidth, &ret); break; case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallCheckBoxWidth, &ret); break; } break; } case PM_DialogButtonsButtonWidth: { QSize sz; ret = d->aquaSizeConstrain(opt, 0, QStyle::CT_PushButton, QSize(-1, -1), &sz); if (sz == QSize(-1, -1)) ret = 70; else ret = sz.width(); break; } case PM_MenuBarHMargin: ret = 8; break; case PM_MenuBarVMargin: ret = 0; break; case QStyle::PM_MenuDesktopFrameWidth: ret = 5; break; case PM_CheckBoxLabelSpacing: case PM_RadioButtonLabelSpacing: ret = 2; break; case PM_MenuScrollerHeight: #if 0 SInt16 ash, asw; GetThemeMenuItemExtra(kThemeMenuItemScrollUpArrow, &ash, &asw); ret = ash; #else ret = 15; // I hate having magic numbers in here... #endif break; case PM_DefaultFrameWidth: #ifndef QT_NO_MAINWINDOW if (widget && (widget->isWindow() || !widget->parentWidget() || (qobject_cast(widget->parentWidget()) && static_cast(widget->parentWidget())->centralWidget() == widget)) && (qobject_cast(widget) #ifdef QT3_SUPPORT || widget->inherits("QScrollView") #endif || widget->inherits("QWorkspaceChild"))) ret = 0; else #endif // The combo box popup has no frame. if (qstyleoption_cast(opt) != 0) ret = 0; else ret = 1; break; case PM_MaximumDragDistance: ret = -1; break; case PM_ScrollBarSliderMin: ret = 24; break; case PM_SpinBoxFrameWidth: GetThemeMetric(kThemeMetricEditTextFrameOutset, &ret); switch (d->aquaSizeConstrain(opt, widget)) { default: ret += 2; break; case QAquaSizeMini: ret += 1; break; } break; case PM_ButtonShiftHorizontal: case PM_ButtonShiftVertical: ret = 0; break; case PM_SliderLength: ret = 17; break; case PM_ButtonDefaultIndicator: ret = 0; break; case PM_TitleBarHeight: if (const QStyleOptionTitleBar *tb = qstyleoption_cast(opt)) { HIThemeWindowDrawInfo wdi; wdi.version = qt_mac_hitheme_version; wdi.state = kThemeStateActive; wdi.windowType = QtWinType; if (tb->titleBarState) wdi.attributes = kThemeWindowHasFullZoom | kThemeWindowHasCloseBox | kThemeWindowHasCollapseBox; else if (tb->titleBarFlags & Qt::WindowSystemMenuHint) wdi.attributes = kThemeWindowHasCloseBox; else wdi.attributes = 0; wdi.titleHeight = tb->rect.height(); wdi.titleWidth = tb->rect.width(); QCFType region; HIRect hirect = qt_hirectForQRect(tb->rect); if (hirect.size.width <= 0) hirect.size.width = 100; if (hirect.size.height <= 0) hirect.size.height = 30; HIThemeGetWindowShape(&hirect, &wdi, kWindowTitleBarRgn, ®ion); HIRect rect; ptrHIShapeGetBounds(region, &rect); ret = int(rect.size.height); ret += 4; } break; case PM_TabBarTabVSpace: ret = 4; break; case PM_TabBarTabShiftHorizontal: case PM_TabBarTabShiftVertical: ret = 0; break; case PM_TabBarBaseHeight: ret = 0; break; case PM_TabBarTabOverlap: ret = 0; break; case PM_TabBarBaseOverlap: switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: ret = 11; break; case QAquaSizeSmall: ret = 8; break; case QAquaSizeMini: ret = 7; break; } break; case PM_ScrollBarExtent: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricScrollBarWidth, &ret); break; case QAquaSizeMini: case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallScrollBarWidth, &ret); break; } break; } case PM_IndicatorHeight: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricCheckBoxHeight, &ret); break; case QAquaSizeMini: GetThemeMetric(kThemeMetricMiniCheckBoxHeight, &ret); break; case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallCheckBoxHeight, &ret); break; } break; } case PM_IndicatorWidth: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricCheckBoxWidth, &ret); break; case QAquaSizeMini: GetThemeMetric(kThemeMetricMiniCheckBoxWidth, &ret); break; case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallCheckBoxWidth, &ret); break; } ++ret; break; } case PM_ExclusiveIndicatorHeight: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricRadioButtonHeight, &ret); break; case QAquaSizeMini: GetThemeMetric(kThemeMetricMiniRadioButtonHeight, &ret); break; case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallRadioButtonHeight, &ret); break; } break; } case PM_ExclusiveIndicatorWidth: { switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: GetThemeMetric(kThemeMetricRadioButtonWidth, &ret); break; case QAquaSizeMini: GetThemeMetric(kThemeMetricMiniRadioButtonWidth, &ret); break; case QAquaSizeSmall: GetThemeMetric(kThemeMetricSmallRadioButtonWidth, &ret); break; } ++ret; break; } case PM_MenuVMargin: ret = 4; break; case PM_MenuPanelWidth: ret = 0; break; case PM_ToolTipLabelFrameWidth: ret = 0; break; case PM_SizeGripSize: { QAquaWidgetSize aSize; if (widget && widget->window()->windowType() == Qt::Tool) aSize = QAquaSizeSmall; else aSize = QAquaSizeLarge; const QSize size = qt_aqua_get_known_size(CT_SizeGrip, widget, QSize(), aSize); ret = size.width(); break; } case PM_MdiSubWindowFrameWidth: ret = 1; break; case PM_DockWidgetFrameWidth: ret = 2; break; case PM_DockWidgetTitleMargin: ret = 0; break; case PM_DockWidgetSeparatorExtent: ret = 1; break; case PM_ToolBarHandleExtent: ret = 11; break; case PM_ToolBarItemMargin: ret = 0; break; case PM_ToolBarItemSpacing: ret = 4; break; case PM_SplitterWidth: ret = qMax(7, QApplication::globalStrut().width()); break; case PM_LayoutLeftMargin: case PM_LayoutTopMargin: case PM_LayoutRightMargin: case PM_LayoutBottomMargin: { bool isWindow = false; if (opt) { isWindow = (opt->state & State_Window); } else if (widget) { isWindow = widget->isWindow(); } if (isWindow) { bool isMetal = widget && widget->testAttribute(Qt::WA_MacBrushedMetal); if (isMetal) { if (metric == PM_LayoutTopMargin) { return_SIZE(9 /* AHIG */, 6 /* guess */, 6 /* guess */); } else if (metric == PM_LayoutBottomMargin) { return_SIZE(18 /* AHIG */, 15 /* guess */, 13 /* guess */); } else { return_SIZE(14 /* AHIG */, 11 /* guess */, 9 /* guess */); } } else { /* AHIG would have (20, 8, 10) here but that makes no sense. It would also have 14 for the top margin but this contradicts both Builder and most applications. */ return_SIZE(20, 10, 10); // AHIG } } else { // hack to detect QTabWidget if (widget && widget->parentWidget() && widget->parentWidget()->sizePolicy().controlType() == QSizePolicy::TabWidget) { if (metric == PM_LayoutTopMargin) { /* Builder would have 14 (= 20 - 6) instead of 12, but that makes the tab look disproportionate. */ return_SIZE(12, 6, 6); // guess } else { return_SIZE(20 /* Builder */, 8 /* guess */, 8 /* guess */); } } else { /* Child margins are highly inconsistent in AHIG and Builder. */ return_SIZE(12, 8, 6); // guess } } } case PM_LayoutHorizontalSpacing: case PM_LayoutVerticalSpacing: return -1; case QStyle::PM_TabBarTabHSpace: switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeLarge: case QAquaSizeUnknown: ret = QWindowsStyle::pixelMetric(metric, opt, widget); break; case QAquaSizeSmall: ret = 20; break; case QAquaSizeMini: ret = 16; break; } break; case PM_MenuHMargin: ret = 0; break; case PM_ToolBarFrameWidth: ret = 1; if (widget) { if (QMainWindow * mainWindow = qobject_cast(widget->parent())) if (mainWindow->unifiedTitleAndToolBarOnMac()) ret = 0; } break; default: ret = QWindowsStyle::pixelMetric(metric, opt, widget); break; } return ret; } QPalette QMacStyle::standardPalette() const { QPalette pal = QWindowsStyle::standardPalette(); pal.setColor(QPalette::Disabled, QPalette::Dark, QColor(191, 191, 191)); pal.setColor(QPalette::Active, QPalette::Dark, QColor(191, 191, 191)); pal.setColor(QPalette::Inactive, QPalette::Dark, QColor(191, 191, 191)); return pal; } int QMacStyle::styleHint(StyleHint sh, const QStyleOption *opt, const QWidget *w, QStyleHintReturn *hret) const { SInt32 ret = 0; switch (sh) { case SH_Menu_SelectionWrap: ret = false; break; case SH_Menu_KeyboardSearch: ret = true; break; case SH_Menu_SpaceActivatesItem: ret = true; break; case SH_Slider_AbsoluteSetButtons: ret = Qt::LeftButton|Qt::MidButton; break; case SH_Slider_PageSetButtons: ret = 0; break; case SH_ScrollBar_ContextMenu: ret = false; break; case SH_TitleBar_AutoRaise: ret = true; break; case SH_Menu_AllowActiveAndDisabled: ret = false; break; case SH_Menu_SubMenuPopupDelay: ret = 100; break; case SH_ScrollBar_LeftClickAbsolutePosition: { extern bool qt_scrollbar_jump_to_pos; //qapplication_mac.cpp if(QApplication::keyboardModifiers() & Qt::AltModifier) ret = !qt_scrollbar_jump_to_pos; else ret = qt_scrollbar_jump_to_pos; break; } case SH_TabBar_PreferNoArrows: ret = true; break; case SH_LineEdit_PasswordCharacter: ret = kBulletUnicode; break; /* case SH_DialogButtons_DefaultButton: ret = QDialogButtons::Reject; break; */ case SH_Menu_SloppySubMenus: ret = true; break; case SH_GroupBox_TextLabelVerticalAlignment: ret = Qt::AlignTop; break; case SH_ScrollView_FrameOnlyAroundContents: if (w && (w->isWindow() || !w->parentWidget() || w->parentWidget()->isWindow()) && (w->inherits("QWorkspaceChild") #ifdef QT3_SUPPORT || w->inherits("QScrollView") #endif )) ret = true; else ret = QWindowsStyle::styleHint(sh, opt, w, hret); break; case SH_Menu_FillScreenWithScroll: ret = false; break; case SH_Menu_Scrollable: ret = true; break; case SH_RichText_FullWidthSelection: ret = true; break; case SH_BlinkCursorWhenTextSelected: ret = false; break; case SH_ScrollBar_StopMouseOverSlider: ret = true; break; case SH_Q3ListViewExpand_SelectMouseType: ret = QEvent::MouseButtonRelease; break; case SH_TabBar_SelectMouseType: if (const QStyleOptionTabBarBaseV2 *opt2 = qstyleoption_cast(opt)) { ret = opt2->documentMode ? QEvent::MouseButtonPress : QEvent::MouseButtonRelease; } else { ret = QEvent::MouseButtonRelease; } break; case SH_ComboBox_Popup: if (const QStyleOptionComboBox *cmb = qstyleoption_cast(opt)) ret = !cmb->editable; else ret = 0; break; case SH_Workspace_FillSpaceOnMaximize: ret = true; break; case SH_Widget_ShareActivation: ret = true; break; case SH_Header_ArrowAlignment: ret = Qt::AlignRight; break; case SH_TabBar_Alignment: { if (const QTabWidget *tab = qobject_cast(w)) { if (tab->documentMode()) { ret = Qt::AlignLeft; break; } } if (const QTabBar *tab = qobject_cast(w)) { if (tab->documentMode()) { ret = Qt::AlignLeft; break; } } ret = Qt::AlignCenter; } break; case SH_UnderlineShortcut: ret = false; break; case SH_ToolTipLabel_Opacity: ret = 242; // About 95% break; case SH_Button_FocusPolicy: ret = Qt::TabFocus; break; case SH_EtchDisabledText: ret = false; break; case SH_FocusFrame_Mask: { ret = true; if(QStyleHintReturnMask *mask = qstyleoption_cast(hret)) { const uchar fillR = 192, fillG = 191, fillB = 190; QImage img; QSize pixmapSize = opt->rect.size(); if (pixmapSize.isValid()) { QPixmap pix(pixmapSize); pix.fill(QColor(fillR, fillG, fillB)); QPainter pix_paint(&pix); proxy()->drawControl(CE_FocusFrame, opt, &pix_paint, w); pix_paint.end(); img = pix.toImage(); } const QRgb *sptr = (QRgb*)img.bits(), *srow; const int sbpl = img.bytesPerLine(); const int w = sbpl/4, h = img.height(); QImage img_mask(img.width(), img.height(), QImage::Format_ARGB32); QRgb *dptr = (QRgb*)img_mask.bits(), *drow; const int dbpl = img_mask.bytesPerLine(); for (int y = 0; y < h; ++y) { srow = sptr+((y*sbpl)/4); drow = dptr+((y*dbpl)/4); for (int x = 0; x < w; ++x) { const int diff = (((qRed(*srow)-fillR)*(qRed(*srow)-fillR)) + ((qGreen(*srow)-fillG)*((qGreen(*srow)-fillG))) + ((qBlue(*srow)-fillB)*((qBlue(*srow)-fillB)))); (*drow++) = (diff < 100) ? 0xffffffff : 0xff000000; ++srow; } } QBitmap qmask = QBitmap::fromImage(img_mask); mask->region = QRegion(qmask); } break; } case SH_TitleBar_NoBorder: ret = 1; break; case SH_RubberBand_Mask: ret = 0; break; case SH_ComboBox_LayoutDirection: ret = Qt::LeftToRight; break; case SH_ItemView_EllipsisLocation: ret = Qt::AlignHCenter; break; case SH_ItemView_ShowDecorationSelected: ret = true; break; case SH_TitleBar_ModifyNotification: ret = false; break; case SH_ScrollBar_RollBetweenButtons: ret = true; break; case SH_WindowFrame_Mask: ret = 1; if (QStyleHintReturnMask *mask = qstyleoption_cast(hret)) { mask->region = opt->rect; mask->region -= QRect(opt->rect.left(), opt->rect.top(), 5, 1); mask->region -= QRect(opt->rect.left(), opt->rect.top() + 1, 3, 1); mask->region -= QRect(opt->rect.left(), opt->rect.top() + 2, 2, 1); mask->region -= QRect(opt->rect.left(), opt->rect.top() + 3, 1, 2); mask->region -= QRect(opt->rect.right() - 4, opt->rect.top(), 5, 1); mask->region -= QRect(opt->rect.right() - 2, opt->rect.top() + 1, 3, 1); mask->region -= QRect(opt->rect.right() - 1, opt->rect.top() + 2, 2, 1); mask->region -= QRect(opt->rect.right() , opt->rect.top() + 3, 1, 2); } break; case SH_TabBar_ElideMode: ret = Qt::ElideRight; break; case SH_DialogButtonLayout: ret = QDialogButtonBox::MacLayout; break; case SH_FormLayoutWrapPolicy: ret = QFormLayout::DontWrapRows; break; case SH_FormLayoutFieldGrowthPolicy: ret = QFormLayout::FieldsStayAtSizeHint; break; case SH_FormLayoutFormAlignment: ret = Qt::AlignHCenter | Qt::AlignTop; break; case SH_FormLayoutLabelAlignment: ret = Qt::AlignRight; break; case SH_ComboBox_PopupFrameStyle: ret = QFrame::NoFrame | QFrame::Plain; break; case SH_MessageBox_TextInteractionFlags: ret = Qt::TextSelectableByMouse | Qt::LinksAccessibleByMouse | Qt::TextSelectableByKeyboard; break; case SH_SpellCheckUnderlineStyle: ret = QTextCharFormat::DashUnderline; break; case SH_MessageBox_CenterButtons: ret = false; break; case SH_MenuBar_AltKeyNavigation: ret = false; break; case SH_ItemView_MovementWithoutUpdatingSelection: ret = false; break; case SH_FocusFrame_AboveWidget: ret = true; break; case SH_WizardStyle: ret = QWizard::MacStyle; break; case SH_ItemView_ArrowKeysNavigateIntoChildren: ret = false; break; case SH_Menu_FlashTriggeredItem: ret = true; break; case SH_Menu_FadeOutOnHide: ret = true; break; case SH_Menu_Mask: if (opt) { if (QStyleHintReturnMask *mask = qstyleoption_cast(hret)) { ret = true; HIRect menuRect = CGRectMake(opt->rect.x(), opt->rect.y() + 4, opt->rect.width(), opt->rect.height() - 8); HIThemeMenuDrawInfo mdi; mdi.version = 0; if (w && qobject_cast(w->parentWidget())) mdi.menuType = kThemeMenuTypeHierarchical; else mdi.menuType = kThemeMenuTypePopUp; QCFType shape; HIThemeGetMenuBackgroundShape(&menuRect, &mdi, &shape); mask->region = QRegion::fromHIShapeRef(shape); } } break; case SH_ItemView_PaintAlternatingRowColorsForEmptyArea: ret = true; break; case SH_TabBar_CloseButtonPosition: ret = QTabBar::LeftSide; break; case SH_DockWidget_ButtonsHaveFrame: ret = false; break; default: ret = QWindowsStyle::styleHint(sh, opt, w, hret); break; } return ret; } QPixmap QMacStyle::generatedIconPixmap(QIcon::Mode iconMode, const QPixmap &pixmap, const QStyleOption *opt) const { switch (iconMode) { case QIcon::Disabled: { QImage img = pixmap.toImage().convertToFormat(QImage::Format_ARGB32); int imgh = img.height(); int imgw = img.width(); QRgb pixel; for (int y = 0; y < imgh; ++y) { for (int x = 0; x < imgw; ++x) { pixel = img.pixel(x, y); img.setPixel(x, y, qRgba(qRed(pixel), qGreen(pixel), qBlue(pixel), qAlpha(pixel) / 2)); } } return QPixmap::fromImage(img); } default: ; } return QWindowsStyle::generatedIconPixmap(iconMode, pixmap, opt); } QPixmap QMacStyle::standardPixmap(StandardPixmap standardPixmap, const QStyleOption *opt, const QWidget *widget) const { // The default implementation of QStyle::standardIconImplementation() is to call standardPixmap() // I don't want infinite recursion so if we do get in that situation, just return the Window's // standard pixmap instead (since there is no mac-specific icon then). This should be fine until // someone changes how Windows standard // pixmap works. static bool recursionGuard = false; if (recursionGuard) return QWindowsStyle::standardPixmap(standardPixmap, opt, widget); recursionGuard = true; QIcon icon = standardIconImplementation(standardPixmap, opt, widget); recursionGuard = false; int size; switch (standardPixmap) { default: size = 32; break; case SP_MessageBoxCritical: case SP_MessageBoxQuestion: case SP_MessageBoxInformation: case SP_MessageBoxWarning: size = 64; break; } return icon.pixmap(size, size); } void QMacStyle::setFocusRectPolicy(QWidget *w, FocusRectPolicy policy) { switch (policy) { case FocusDefault: break; case FocusEnabled: case FocusDisabled: w->setAttribute(Qt::WA_MacShowFocusRect, policy == FocusEnabled); break; } } QMacStyle::FocusRectPolicy QMacStyle::focusRectPolicy(const QWidget *w) { return w->testAttribute(Qt::WA_MacShowFocusRect) ? FocusEnabled : FocusDisabled; } void QMacStyle::setWidgetSizePolicy(const QWidget *widget, WidgetSizePolicy policy) { QWidget *wadget = const_cast(widget); wadget->setAttribute(Qt::WA_MacNormalSize, policy == SizeLarge); wadget->setAttribute(Qt::WA_MacSmallSize, policy == SizeSmall); wadget->setAttribute(Qt::WA_MacMiniSize, policy == SizeMini); } QMacStyle::WidgetSizePolicy QMacStyle::widgetSizePolicy(const QWidget *widget) { while (widget) { if (widget->testAttribute(Qt::WA_MacMiniSize)) { return SizeMini; } else if (widget->testAttribute(Qt::WA_MacSmallSize)) { return SizeSmall; } else if (widget->testAttribute(Qt::WA_MacNormalSize)) { return SizeLarge; } widget = widget->parentWidget(); } return SizeDefault; } void QMacStyle::drawPrimitive(PrimitiveElement pe, const QStyleOption *opt, QPainter *p, const QWidget *w) const { ThemeDrawState tds = d->getDrawState(opt->state); QMacCGContext cg(p); switch (pe) { case PE_IndicatorArrowUp: case PE_IndicatorArrowDown: case PE_IndicatorArrowRight: case PE_IndicatorArrowLeft: { p->save(); p->setRenderHint(QPainter::Antialiasing); int xOffset = opt->direction == Qt::LeftToRight ? 2 : -1; QMatrix matrix; matrix.translate(opt->rect.center().x() + xOffset, opt->rect.center().y() + 2); QPainterPath path; switch(pe) { default: case PE_IndicatorArrowDown: break; case PE_IndicatorArrowUp: matrix.rotate(180); break; case PE_IndicatorArrowLeft: matrix.rotate(90); break; case PE_IndicatorArrowRight: matrix.rotate(-90); break; } path.moveTo(0, 5); path.lineTo(-4, -3); path.lineTo(4, -3); p->setMatrix(matrix); p->setPen(Qt::NoPen); p->setBrush(QColor(0, 0, 0, 135)); p->drawPath(path); p->restore(); break; } case PE_FrameTabBarBase: if (const QStyleOptionTabBarBaseV2 *tbb = qstyleoption_cast(opt)) { if (tbb->documentMode) { p->save(); drawTabBase(p, tbb, w); p->restore(); return; } QRegion region(tbb->rect); region -= tbb->tabBarRect; p->save(); p->setClipRegion(region); QStyleOptionTabWidgetFrame twf; twf.QStyleOption::operator=(*tbb); twf.shape = tbb->shape; switch (getTabDirection(twf.shape)) { case kThemeTabNorth: twf.rect = twf.rect.adjusted(0, 0, 0, 10); break; case kThemeTabSouth: twf.rect = twf.rect.adjusted(0, -10, 0, 0); break; case kThemeTabWest: twf.rect = twf.rect.adjusted(0, 0, 10, 0); break; case kThemeTabEast: twf.rect = twf.rect.adjusted(0, -10, 0, 0); break; } proxy()->drawPrimitive(PE_FrameTabWidget, &twf, p, w); p->restore(); } break; case PE_PanelTipLabel: p->fillRect(opt->rect, opt->palette.brush(QPalette::ToolTipBase)); break; case PE_FrameGroupBox: if (const QStyleOptionFrame *groupBox = qstyleoption_cast(opt)) { const QStyleOptionFrameV2 *frame2 = qstyleoption_cast(opt); if (frame2 && frame2->features & QStyleOptionFrameV2::Flat) { QWindowsStyle::drawPrimitive(pe, groupBox, p, w); } else { HIThemeGroupBoxDrawInfo gdi; gdi.version = qt_mac_hitheme_version; gdi.state = tds; if (w && qobject_cast(w->parentWidget())) gdi.kind = kHIThemeGroupBoxKindSecondary; else gdi.kind = kHIThemeGroupBoxKindPrimary; HIRect hirect = qt_hirectForQRect(opt->rect); HIThemeDrawGroupBox(&hirect, &gdi, cg, kHIThemeOrientationNormal); } } break; case PE_IndicatorToolBarSeparator: { QPainterPath path; if (opt->state & State_Horizontal) { int xpoint = opt->rect.center().x(); path.moveTo(xpoint + 0.5, opt->rect.top() + 1); path.lineTo(xpoint + 0.5, opt->rect.bottom()); } else { int ypoint = opt->rect.center().y(); path.moveTo(opt->rect.left() + 2 , ypoint + 0.5); path.lineTo(opt->rect.right() + 1, ypoint + 0.5); } QPainterPathStroker theStroker; theStroker.setCapStyle(Qt::FlatCap); theStroker.setDashPattern(QVector() << 1 << 2); path = theStroker.createStroke(path); p->fillPath(path, QColor(0, 0, 0, 119)); } break; case PE_FrameWindow: break; case PE_IndicatorDockWidgetResizeHandle: { // The docwidget resize handle is drawn as a one-pixel wide line. p->save(); if (opt->state & State_Horizontal) { p->setPen(QColor(160, 160, 160)); p->drawLine(opt->rect.topLeft(), opt->rect.topRight()); } else { p->setPen(QColor(145, 145, 145)); p->drawLine(opt->rect.topRight(), opt->rect.bottomRight()); } p->restore(); } break; case PE_IndicatorToolBarHandle: { p->save(); QPainterPath path; int x = opt->rect.x() + 6; int y = opt->rect.y() + 5; static const int RectHeight = 2; if (opt->state & State_Horizontal) { while (y < opt->rect.height() - RectHeight - 6) { path.moveTo(x, y); path.addRect(x, y, RectHeight, RectHeight); y += 6; } } else { while (x < opt->rect.width() - RectHeight - 6) { path.moveTo(x, y); path.addRect(x, y, RectHeight, RectHeight); x += 6; } } p->setPen(Qt::NoPen); QColor dark = opt->palette.dark().color(); dark.setAlphaF(0.75); QColor light = opt->palette.light().color(); light.setAlphaF(0.6); p->fillPath(path, light); p->save(); p->translate(1, 1); p->fillPath(path, dark); p->restore(); p->translate(3, 3); p->fillPath(path, light); p->translate(1, 1); p->fillPath(path, dark); p->restore(); break; } case PE_IndicatorHeaderArrow: if (const QStyleOptionHeader *header = qstyleoption_cast(opt)) { // In HITheme, up is down, down is up and hamburgers eat people. if (header->sortIndicator != QStyleOptionHeader::None) proxy()->drawPrimitive( (header->sortIndicator == QStyleOptionHeader::SortDown) ? PE_IndicatorArrowUp : PE_IndicatorArrowDown, header, p, w); } break; case PE_IndicatorMenuCheckMark: { const int checkw = 8; const int checkh = 8; const int xoff = qMax(0, (opt->rect.width() - checkw) / 2); const int yoff = qMax(0, (opt->rect.width() - checkh) / 2); const int x1 = xoff + opt->rect.x(); const int y1 = yoff + opt->rect.y() + checkw/2; const int x2 = xoff + opt->rect.x() + checkw/4; const int y2 = yoff + opt->rect.y() + checkh; const int x3 = xoff + opt->rect.x() + checkw; const int y3 = yoff + opt->rect.y(); QVector a(2); a << QLineF(x1, y1, x2, y2); a << QLineF(x2, y2, x3, y3); if (opt->palette.currentColorGroup() == QPalette::Active) p->setPen(QPen(Qt::white, 3)); else p->setPen(QPen(QColor(100, 100, 100), 3)); p->save(); p->setRenderHint(QPainter::Antialiasing); p->drawLines(a); p->restore(); break; } case PE_IndicatorViewItemCheck: case PE_Q3CheckListExclusiveIndicator: case PE_Q3CheckListIndicator: case PE_IndicatorRadioButton: case PE_IndicatorCheckBox: { bool drawColorless = (!(opt->state & State_Active)) && opt->palette.currentColorGroup() == QPalette::Active; HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = tds; if (drawColorless && tds == kThemeStateInactive) bdi.state = kThemeStateActive; bdi.adornment = kThemeDrawIndicatorOnly; if (opt->state & State_HasFocus) bdi.adornment |= kThemeAdornmentFocus; bool isRadioButton = (pe == PE_Q3CheckListExclusiveIndicator || pe == PE_IndicatorRadioButton); switch (d->aquaSizeConstrain(opt, w)) { case QAquaSizeUnknown: case QAquaSizeLarge: if (isRadioButton) bdi.kind = kThemeRadioButton; else bdi.kind = kThemeCheckBox; break; case QAquaSizeMini: if (isRadioButton) bdi.kind = kThemeMiniRadioButton; else bdi.kind = kThemeMiniCheckBox; break; case QAquaSizeSmall: if (isRadioButton) bdi.kind = kThemeSmallRadioButton; else bdi.kind = kThemeSmallCheckBox; break; } if (opt->state & State_NoChange) bdi.value = kThemeButtonMixed; else if (opt->state & State_On) bdi.value = kThemeButtonOn; else bdi.value = kThemeButtonOff; HIRect macRect; if (pe == PE_Q3CheckListExclusiveIndicator || pe == PE_Q3CheckListIndicator) macRect = qt_hirectForQRect(opt->rect); else macRect = qt_hirectForQRect(opt->rect); if (!drawColorless) HIThemeDrawButton(&macRect, &bdi, cg, kHIThemeOrientationNormal, 0); else d->drawColorlessButton(macRect, &bdi, p, opt); break; } case PE_FrameFocusRect: // Use the our own focus widget stuff. break; case PE_IndicatorBranch: { if (!(opt->state & State_Children)) break; HIThemeButtonDrawInfo bi; bi.version = qt_mac_hitheme_version; bi.state = tds; if (tds == kThemeStateInactive && opt->palette.currentColorGroup() == QPalette::Active) bi.state = kThemeStateActive; if (opt->state & State_Sunken) bi.state |= kThemeStatePressed; bi.kind = kThemeDisclosureButton; if (opt->state & State_Open) bi.value = kThemeDisclosureDown; else bi.value = opt->direction == Qt::LeftToRight ? kThemeDisclosureRight : kThemeDisclosureLeft; bi.adornment = kThemeAdornmentNone; HIRect hirect = qt_hirectForQRect(opt->rect); HIThemeDrawButton(&hirect, &bi, cg, kHIThemeOrientationNormal, 0); break; } case PE_Frame: { QPen oldPen = p->pen(); p->setPen(opt->palette.base().color().darker(140)); p->drawRect(opt->rect.adjusted(0, 0, -1, -1)); p->setPen(opt->palette.base().color().darker(180)); p->drawLine(opt->rect.topLeft(), opt->rect.topRight()); p->setPen(oldPen); break; } case PE_FrameLineEdit: if (const QStyleOptionFrame *frame = qstyleoption_cast(opt)) { if (frame->state & State_Sunken) { QColor baseColor(frame->palette.background().color()); HIThemeFrameDrawInfo fdi; fdi.version = qt_mac_hitheme_version; fdi.state = tds; SInt32 frame_size; if (pe == PE_FrameLineEdit) { fdi.kind = kHIThemeFrameTextFieldSquare; GetThemeMetric(kThemeMetricEditTextFrameOutset, &frame_size); if ((frame->state & State_ReadOnly) || !(frame->state & State_Enabled)) fdi.state = kThemeStateInactive; } else { baseColor = QColor(150, 150, 150); //hardcoded since no query function --Sam fdi.kind = kHIThemeFrameListBox; GetThemeMetric(kThemeMetricListBoxFrameOutset, &frame_size); } fdi.isFocused = (frame->state & State_HasFocus); int lw = frame->lineWidth; if (lw <= 0) lw = proxy()->pixelMetric(PM_DefaultFrameWidth, frame, w); { //clear to base color p->save(); p->setPen(QPen(baseColor, lw)); p->setBrush(Qt::NoBrush); p->drawRect(frame->rect); p->restore(); } HIRect hirect = qt_hirectForQRect(frame->rect, QRect(frame_size, frame_size, frame_size * 2, frame_size * 2)); HIThemeDrawFrame(&hirect, &fdi, cg, kHIThemeOrientationNormal); } else { QWindowsStyle::drawPrimitive(pe, opt, p, w); } } break; case PE_PanelLineEdit: QWindowsStyle::drawPrimitive(pe, opt, p, w); // Draw the focus frame for widgets other than QLineEdit (e.g. for line edits in Webkit). // Focus frame is drawn outside the rectangle passed in the option-rect. if (const QStyleOptionFrame *panel = qstyleoption_cast(opt)) { if ((opt->state & State_HasFocus) && !qobject_cast(w)) { int vmargin = pixelMetric(QStyle::PM_FocusFrameVMargin); int hmargin = pixelMetric(QStyle::PM_FocusFrameHMargin); QStyleOptionFrame focusFrame = *panel; focusFrame.rect = panel->rect.adjusted(-hmargin, -vmargin, hmargin, vmargin); drawControl(CE_FocusFrame, &focusFrame, p, w); } } break; case PE_FrameTabWidget: if (const QStyleOptionTabWidgetFrame *twf = qstyleoption_cast(opt)) { HIRect hirect = qt_hirectForQRect(twf->rect); HIThemeTabPaneDrawInfo tpdi; tpdi.version = qt_mac_hitheme_tab_version(); tpdi.state = tds; tpdi.direction = getTabDirection(twf->shape); tpdi.size = kHIThemeTabSizeNormal; tpdi.kind = kHIThemeTabKindNormal; tpdi.adornment = kHIThemeTabPaneAdornmentNormal; HIThemeDrawTabPane(&hirect, &tpdi, cg, kHIThemeOrientationNormal); } break; case PE_PanelScrollAreaCorner: { const QBrush brush(qApp->palette().brush(QPalette::Base)); p->fillRect(opt->rect, brush); p->setPen(QPen(QColor(217, 217, 217))); p->drawLine(opt->rect.topLeft(), opt->rect.topRight()); p->drawLine(opt->rect.topLeft(), opt->rect.bottomLeft()); } break; case PE_FrameStatusBarItem: break; case PE_IndicatorTabClose: { bool hover = (opt->state & State_MouseOver); bool selected = (opt->state & State_Selected); bool active = (opt->state & State_Active); drawTabCloseButton(p, hover, active, selected); } break; case PE_PanelStatusBar: { if (QSysInfo::MacintoshVersion <= QSysInfo::MV_10_4) { QWindowsStyle::drawPrimitive(pe, opt, p, w); break; } // Use the Leopard style only if the status bar is the status bar for a // QMainWindow with a unifed toolbar. if (w == 0 || w->parent() == 0 || qobject_cast(w->parent()) == 0 || qobject_cast(w->parent())->unifiedTitleAndToolBarOnMac() == false ) { QWindowsStyle::drawPrimitive(pe, opt, p, w); break; } // Fill the status bar with the titlebar gradient. QLinearGradient linearGrad(0, opt->rect.top(), 0, opt->rect.bottom()); if (opt->state & QStyle::State_Active) { linearGrad.setColorAt(0, titlebarGradientActiveBegin); linearGrad.setColorAt(1, titlebarGradientActiveEnd); } else { linearGrad.setColorAt(0, titlebarGradientInactiveBegin); linearGrad.setColorAt(1, titlebarGradientInactiveEnd); } p->fillRect(opt->rect, linearGrad); // Draw the black separator line at the top of the status bar. if (opt->state & QStyle::State_Active) p->setPen(titlebarSeparatorLineActive); else p->setPen(titlebarSeparatorLineInactive); p->drawLine(opt->rect.left(), opt->rect.top(), opt->rect.right(), opt->rect.top()); break; } default: QWindowsStyle::drawPrimitive(pe, opt, p, w); break; } } static inline QPixmap darkenPixmap(const QPixmap &pixmap) { QImage img = pixmap.toImage().convertToFormat(QImage::Format_ARGB32); int imgh = img.height(); int imgw = img.width(); int h, s, v, a; QRgb pixel; for (int y = 0; y < imgh; ++y) { for (int x = 0; x < imgw; ++x) { pixel = img.pixel(x, y); a = qAlpha(pixel); QColor hsvColor(pixel); hsvColor.getHsv(&h, &s, &v); s = qMin(100, s * 2); v = v / 2; hsvColor.setHsv(h, s, v); pixel = hsvColor.rgb(); img.setPixel(x, y, qRgba(qRed(pixel), qGreen(pixel), qBlue(pixel), a)); } } return QPixmap::fromImage(img); } void QMacStyle::drawControl(ControlElement ce, const QStyleOption *opt, QPainter *p, const QWidget *w) const { ThemeDrawState tds = d->getDrawState(opt->state); QMacCGContext cg(p); switch (ce) { case CE_HeaderSection: if (const QStyleOptionHeader *header = qstyleoption_cast(opt)) { HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; State flags = header->state; QRect ir = header->rect; bdi.kind = kThemeListHeaderButton; bdi.adornment = kThemeAdornmentNone; bdi.state = kThemeStateActive; if (flags & State_On) bdi.value = kThemeButtonOn; else bdi.value = kThemeButtonOff; if (header->orientation == Qt::Horizontal){ switch (header->position) { case QStyleOptionHeader::Beginning: ir.adjust(-1, -1, 0, 0); break; case QStyleOptionHeader::Middle: ir.adjust(-1, -1, 0, 0); break; case QStyleOptionHeader::OnlyOneSection: case QStyleOptionHeader::End: ir.adjust(-1, -1, 1, 0); break; default: break; } if (header->position != QStyleOptionHeader::Beginning && header->position != QStyleOptionHeader::OnlyOneSection) { bdi.adornment = header->direction == Qt::LeftToRight ? kThemeAdornmentHeaderButtonLeftNeighborSelected : kThemeAdornmentHeaderButtonRightNeighborSelected; } } if (flags & State_Active) { if (!(flags & State_Enabled)) bdi.state = kThemeStateUnavailable; else if (flags & State_Sunken) bdi.state = kThemeStatePressed; } else { if (flags & State_Enabled) bdi.state = kThemeStateInactive; else bdi.state = kThemeStateUnavailableInactive; } if (header->sortIndicator != QStyleOptionHeader::None) { bdi.value = kThemeButtonOn; if (header->sortIndicator == QStyleOptionHeader::SortDown) bdi.adornment = kThemeAdornmentHeaderButtonSortUp; } if (flags & State_HasFocus) bdi.adornment = kThemeAdornmentFocus; ir = visualRect(header->direction, header->rect, ir); HIRect bounds = qt_hirectForQRect(ir); bool noVerticalHeader = true; if (w) if (const QTableView *table = qobject_cast(w->parentWidget())) noVerticalHeader = !table->verticalHeader()->isVisible(); bool drawTopBorder = header->orientation == Qt::Horizontal; bool drawLeftBorder = header->orientation == Qt::Vertical || header->position == QStyleOptionHeader::OnlyOneSection || (header->position == QStyleOptionHeader::Beginning && noVerticalHeader); d->drawTableHeader(bounds, drawTopBorder, drawLeftBorder, bdi, p); } break; case CE_HeaderLabel: if (const QStyleOptionHeader *header = qstyleoption_cast(opt)) { QRect textr = header->rect; if (!header->icon.isNull()) { QIcon::Mode mode = QIcon::Disabled; if (opt->state & State_Enabled) mode = QIcon::Normal; QPixmap pixmap = header->icon.pixmap(proxy()->pixelMetric(PM_SmallIconSize), mode); QRect pixr = header->rect; pixr.setY(header->rect.center().y() - (pixmap.height() - 1) / 2); proxy()->drawItemPixmap(p, pixr, Qt::AlignVCenter, pixmap); textr.translate(pixmap.width() + 2, 0); } proxy()->drawItemText(p, textr, header->textAlignment | Qt::AlignVCenter, header->palette, header->state & State_Enabled, header->text, QPalette::ButtonText); } break; case CE_ToolButtonLabel: if (const QStyleOptionToolButton *tb = qstyleoption_cast(opt)) { QStyleOptionToolButton myTb = *tb; myTb.state &= ~State_AutoRaise; if (w && qobject_cast(w->parentWidget())) { QRect cr = tb->rect; int shiftX = 0; int shiftY = 0; bool needText = false; int alignment = 0; bool down = tb->state & (State_Sunken | State_On); if (down) { shiftX = proxy()->pixelMetric(PM_ButtonShiftHorizontal, tb, w); shiftY = proxy()->pixelMetric(PM_ButtonShiftVertical, tb, w); } // The down state is special for QToolButtons in a toolbar on the Mac // The text is a bit bolder and gets a drop shadow and the icons are also darkened. // This doesn't really fit into any particular case in QIcon, so we // do the majority of the work ourselves. if (!(tb->features & QStyleOptionToolButton::Arrow)) { Qt::ToolButtonStyle tbstyle = tb->toolButtonStyle; if (tb->icon.isNull() && !tb->text.isEmpty()) tbstyle = Qt::ToolButtonTextOnly; switch (tbstyle) { case Qt::ToolButtonTextOnly: { needText = true; alignment = Qt::AlignCenter; break; } case Qt::ToolButtonIconOnly: case Qt::ToolButtonTextBesideIcon: case Qt::ToolButtonTextUnderIcon: { QRect pr = cr; QIcon::Mode iconMode = (tb->state & State_Enabled) ? QIcon::Normal : QIcon::Disabled; QIcon::State iconState = (tb->state & State_On) ? QIcon::On : QIcon::Off; QPixmap pixmap = tb->icon.pixmap(tb->rect.size().boundedTo(tb->iconSize), iconMode, iconState); // Draw the text if it's needed. if (tb->toolButtonStyle != Qt::ToolButtonIconOnly) { needText = true; if (tb->toolButtonStyle == Qt::ToolButtonTextUnderIcon) { QMainWindow *mw = qobject_cast(w->window()); if (mw && mw->unifiedTitleAndToolBarOnMac()) { pr.setHeight(pixmap.size().height()); cr.adjust(0, pr.bottom() + 1, 0, 1); } else { pr.setHeight(pixmap.size().height() + 6); cr.adjust(0, pr.bottom(), 0, -3); } alignment |= Qt::AlignCenter; } else { pr.setWidth(pixmap.width() + 8); cr.adjust(pr.right(), 0, 0, 0); alignment |= Qt::AlignLeft | Qt::AlignVCenter; } } if (opt->state & State_Sunken) { pr.translate(shiftX, shiftY); pixmap = darkenPixmap(pixmap); } proxy()->drawItemPixmap(p, pr, Qt::AlignCenter, pixmap); break; } default: Q_ASSERT(false); break; } if (needText) { QPalette pal = tb->palette; QPalette::ColorRole role = QPalette::NoRole; if (down) cr.translate(shiftX, shiftY); if (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_5 && (tbstyle == Qt::ToolButtonTextOnly || (tbstyle != Qt::ToolButtonTextOnly && !down))) { QPen pen = p->pen(); QColor light = down ? Qt::black : Qt::white; light.setAlphaF(0.375f); p->setPen(light); p->drawText(cr.adjusted(0, 1, 0, 1), alignment, tb->text); p->setPen(pen); if (down && tbstyle == Qt::ToolButtonTextOnly) { pal = QApplication::palette("QMenu"); pal.setCurrentColorGroup(tb->palette.currentColorGroup()); role = QPalette::HighlightedText; } } drawItemText(p, cr, alignment, pal, tb->state & State_Enabled, tb->text, role); if (QSysInfo::MacintoshVersion < QSysInfo::MV_10_5 && (tb->state & State_Sunken)) { // Draw a "drop shadow" in earlier versions. drawItemText(p, cr.adjusted(0, 1, 0, 1), alignment, tb->palette, tb->state & State_Enabled, tb->text); } } } else { QWindowsStyle::drawControl(ce, &myTb, p, w); } } else { QWindowsStyle::drawControl(ce, &myTb, p, w); } } break; case CE_ToolBoxTabShape: QCommonStyle::drawControl(ce, opt, p, w); break; case CE_PushButtonBevel: if (const QStyleOptionButton *btn = ::qstyleoption_cast(opt)) { if (!(btn->state & (State_Raised | State_Sunken | State_On))) break; if (btn->features & QStyleOptionButton::CommandLinkButton) { QWindowsStyle::drawControl(ce, opt, p, w); break; } HIThemeButtonDrawInfo bdi; d->initHIThemePushButton(btn, w, tds, &bdi); if (btn->features & QStyleOptionButton::DefaultButton && d->animatable(QMacStylePrivate::AquaPushButton, w)) { bdi.adornment |= kThemeAdornmentDefault; bdi.animation.time.start = d->defaultButtonStart; bdi.animation.time.current = CFAbsoluteTimeGetCurrent(); if (d->timerID <= -1) QMetaObject::invokeMethod(d, "startAnimationTimer", Qt::QueuedConnection); } // Unlike Carbon, we want the button to always be drawn inside its bounds. // Therefore, make the button a bit smaller, so that even if it got focus, // the focus 'shadow' will be inside. HIRect newRect = qt_hirectForQRect(btn->rect); if (bdi.kind == kThemePushButton || bdi.kind == kThemePushButtonSmall) { newRect.origin.x += PushButtonLeftOffset; newRect.origin.y += PushButtonTopOffset; newRect.size.width -= PushButtonRightOffset; newRect.size.height -= PushButtonBottomOffset; } else if (bdi.kind == kThemePushButtonMini) { newRect.origin.x += PushButtonLeftOffset - 2; newRect.origin.y += PushButtonTopOffset; newRect.size.width -= PushButtonRightOffset - 4; } HIThemeDrawButton(&newRect, &bdi, cg, kHIThemeOrientationNormal, 0); if (btn->features & QStyleOptionButton::HasMenu) { int mbi = proxy()->pixelMetric(QStyle::PM_MenuButtonIndicator, btn, w); QRect ir = btn->rect; HIRect arrowRect = CGRectMake(ir.right() - mbi - PushButtonRightOffset, ir.height() / 2 - 4, mbi, ir.height() / 2); bool drawColorless = btn->palette.currentColorGroup() == QPalette::Active; if (drawColorless && tds == kThemeStateInactive) tds = kThemeStateActive; HIThemePopupArrowDrawInfo pdi; pdi.version = qt_mac_hitheme_version; pdi.state = tds; pdi.orientation = kThemeArrowDown; if (arrowRect.size.width < 8.) pdi.size = kThemeArrow5pt; else pdi.size = kThemeArrow9pt; HIThemeDrawPopupArrow(&arrowRect, &pdi, cg, kHIThemeOrientationNormal); } } break; case CE_PushButtonLabel: if (const QStyleOptionButton *btn = qstyleoption_cast(opt)) { // We really don't want the label to be drawn the same as on // windows style if it has an icon and text, then it should be more like a // tab. So, cheat a little here. However, if it *is* only an icon // the windows style works great, so just use that implementation. bool hasMenu = btn->features & QStyleOptionButton::HasMenu; bool hasIcon = !btn->icon.isNull(); bool hasText = !btn->text.isEmpty(); if (!hasIcon && !hasMenu) { // ### this is really overly difficult, simplify. // It basically tries to get the right font for "small" and "mini" icons. QFont oldFont = p->font(); QFont newFont = qt_app_fonts_hash()->value("QPushButton", QFont()); ThemeFontID themeId = kThemePushButtonFont; if (oldFont == newFont) { // Yes, use HITheme to draw the text for small sizes. switch (d->aquaSizeConstrain(opt, w)) { default: break; case QAquaSizeSmall: themeId = kThemeSmallSystemFont; break; case QAquaSizeMini: themeId = kThemeMiniSystemFont; break; } } if (themeId == kThemePushButtonFont) { QWindowsStyle::drawControl(ce, btn, p, w); } else { p->save(); CGContextSetShouldAntialias(cg, true); CGContextSetShouldSmoothFonts(cg, true); HIThemeTextInfo tti; tti.version = qt_mac_hitheme_version; tti.state = tds; QColor textColor = btn->palette.buttonText().color(); CGFloat colorComp[] = { textColor.redF(), textColor.greenF(), textColor.blueF(), textColor.alphaF() }; CGContextSetFillColorSpace(cg, QCoreGraphicsPaintEngine::macGenericColorSpace()); CGContextSetFillColor(cg, colorComp); tti.fontID = themeId; tti.horizontalFlushness = kHIThemeTextHorizontalFlushCenter; tti.verticalFlushness = kHIThemeTextVerticalFlushCenter; tti.options = kHIThemeTextBoxOptionNone; tti.truncationPosition = kHIThemeTextTruncationNone; tti.truncationMaxLines = 1 + btn->text.count(QLatin1Char('\n')); QCFString buttonText = qt_mac_removeMnemonics(btn->text); QRect r = btn->rect; HIRect bounds = qt_hirectForQRect(r); HIThemeDrawTextBox(buttonText, &bounds, &tti, cg, kHIThemeOrientationNormal); p->restore(); } } else { if (hasIcon && !hasText) { QWindowsStyle::drawControl(ce, btn, p, w); } else { QRect freeContentRect = btn->rect; QRect textRect = itemTextRect( btn->fontMetrics, freeContentRect, Qt::AlignCenter, btn->state & State_Enabled, btn->text); if (hasMenu) textRect.adjust(-1, 0, -1, 0); // Draw the icon: if (hasIcon) { int contentW = textRect.width(); if (hasMenu) contentW += proxy()->pixelMetric(PM_MenuButtonIndicator) + 4; QIcon::Mode mode = btn->state & State_Enabled ? QIcon::Normal : QIcon::Disabled; if (mode == QIcon::Normal && btn->state & State_HasFocus) mode = QIcon::Active; // Decide if the icon is should be on or off: QIcon::State state = QIcon::Off; if (btn->state & State_On) state = QIcon::On; QPixmap pixmap = btn->icon.pixmap(btn->iconSize, mode, state); contentW += pixmap.width() + PushButtonContentPadding; int iconLeftOffset = freeContentRect.x() + (freeContentRect.width() - contentW) / 2; int iconTopOffset = freeContentRect.y() + (freeContentRect.height() - pixmap.height()) / 2; QRect iconDestRect(iconLeftOffset, iconTopOffset, pixmap.width(), pixmap.height()); QRect visualIconDestRect = visualRect(btn->direction, freeContentRect, iconDestRect); proxy()->drawItemPixmap(p, visualIconDestRect, Qt::AlignLeft | Qt::AlignVCenter, pixmap); int newOffset = iconDestRect.x() + iconDestRect.width() + PushButtonContentPadding - textRect.x(); textRect.adjust(newOffset, 0, newOffset, 0); } // Draw the text: if (hasText) { textRect = visualRect(btn->direction, freeContentRect, textRect); proxy()->drawItemText(p, textRect, Qt::AlignLeft | Qt::AlignVCenter | Qt::TextShowMnemonic, btn->palette, (btn->state & State_Enabled), btn->text, QPalette::ButtonText); } } } } break; case CE_ComboBoxLabel: if (const QStyleOptionComboBox *cb = qstyleoption_cast(opt)) { QStyleOptionComboBox comboCopy = *cb; comboCopy.direction = Qt::LeftToRight; QWindowsStyle::drawControl(CE_ComboBoxLabel, &comboCopy, p, w); } break; case CE_TabBarTabShape: if (const QStyleOptionTab *tabOpt = qstyleoption_cast(opt)) { if (const QStyleOptionTabV3 *tabOptV3 = qstyleoption_cast(opt)) { if (tabOptV3->documentMode) { p->save(); QRect tabRect = tabOptV3->rect; drawTabShape(p, tabOptV3); p->restore(); return; } } HIThemeTabDrawInfo tdi; tdi.version = 1; tdi.style = kThemeTabNonFront; tdi.direction = getTabDirection(tabOpt->shape); switch (d->aquaSizeConstrain(opt, w)) { default: case QAquaSizeUnknown: case QAquaSizeLarge: tdi.size = kHIThemeTabSizeNormal; break; case QAquaSizeSmall: tdi.size = kHIThemeTabSizeSmall; break; case QAquaSizeMini: tdi.size = kHIThemeTabSizeMini; break; } bool verticalTabs = tdi.direction == kThemeTabWest || tdi.direction == kThemeTabEast; QRect tabRect = tabOpt->rect; bool selected = tabOpt->state & State_Selected; if (selected) { if (!(tabOpt->state & State_Active)) tdi.style = kThemeTabFrontUnavailable; else if (!(tabOpt->state & State_Enabled)) tdi.style = kThemeTabFrontInactive; else tdi.style = kThemeTabFront; } else if (!(tabOpt->state & State_Active)) { tdi.style = kThemeTabNonFrontUnavailable; } else if (!(tabOpt->state & State_Enabled)) { tdi.style = kThemeTabNonFrontInactive; } else if (tabOpt->state & State_Sunken) { tdi.style = kThemeTabFrontInactive; // (should be kThemeTabNonFrontPressed) } if (tabOpt->state & State_HasFocus) tdi.adornment = kHIThemeTabAdornmentFocus; else tdi.adornment = kHIThemeTabAdornmentNone; tdi.kind = kHIThemeTabKindNormal; if (!verticalTabs) tabRect.setY(tabRect.y() - 1); else tabRect.setX(tabRect.x() - 1); QStyleOptionTab::TabPosition tp = tabOpt->position; QStyleOptionTab::SelectedPosition sp = tabOpt->selectedPosition; if (tabOpt->direction == Qt::RightToLeft && !verticalTabs) { if (sp == QStyleOptionTab::NextIsSelected) sp = QStyleOptionTab::PreviousIsSelected; else if (sp == QStyleOptionTab::PreviousIsSelected) sp = QStyleOptionTab::NextIsSelected; switch (tp) { case QStyleOptionTab::Beginning: tp = QStyleOptionTab::End; break; case QStyleOptionTab::End: tp = QStyleOptionTab::Beginning; break; default: break; } } bool stretchTabs = (!verticalTabs && tabRect.height() > 22 || verticalTabs && tabRect.width() > 22); switch (tp) { case QStyleOptionTab::Beginning: tdi.position = kHIThemeTabPositionFirst; if (sp != QStyleOptionTab::NextIsSelected || stretchTabs) tdi.adornment |= kHIThemeTabAdornmentTrailingSeparator; break; case QStyleOptionTab::Middle: tdi.position = kHIThemeTabPositionMiddle; if (selected) tdi.adornment |= kHIThemeTabAdornmentLeadingSeparator; if (sp != QStyleOptionTab::NextIsSelected || stretchTabs) // Also when we're selected. tdi.adornment |= kHIThemeTabAdornmentTrailingSeparator; break; case QStyleOptionTab::End: tdi.position = kHIThemeTabPositionLast; if (selected) tdi.adornment |= kHIThemeTabAdornmentLeadingSeparator; break; case QStyleOptionTab::OnlyOneTab: tdi.position = kHIThemeTabPositionOnly; break; } // HITheme doesn't stretch its tabs. Therefore we have to cheat and do the job ourselves. if (stretchTabs) { HIRect hirect = CGRectMake(0, 0, 23, 23); QPixmap pm(23, 23); pm.fill(Qt::transparent); { QMacCGContext pmcg(&pm); HIThemeDrawTab(&hirect, &tdi, pmcg, kHIThemeOrientationNormal, 0); } QStyleHelper::drawBorderPixmap(pm, p, tabRect, 7, 7, 7, 7); } else { HIRect hirect = qt_hirectForQRect(tabRect); HIThemeDrawTab(&hirect, &tdi, cg, kHIThemeOrientationNormal, 0); } } break; case CE_TabBarTabLabel: if (const QStyleOptionTab *tab = qstyleoption_cast(opt)) { QStyleOptionTabV3 myTab = *tab; ThemeTabDirection ttd = getTabDirection(myTab.shape); bool verticalTabs = ttd == kThemeTabWest || ttd == kThemeTabEast; // Check to see if we use have the same as the system font // (QComboMenuItem is internal and should never be seen by the // outside world, unless they read the source, in which case, it's // their own fault). bool nonDefaultFont = p->font() != qt_app_fonts_hash()->value("QComboMenuItem"); if (verticalTabs || nonDefaultFont || !tab->icon.isNull() || !myTab.leftButtonSize.isNull() || !myTab.rightButtonSize.isNull()) { int heightOffset = 0; if (verticalTabs) { heightOffset = -1; } else if (nonDefaultFont) { if (p->fontMetrics().height() == myTab.rect.height()) heightOffset = 2; } myTab.rect.setHeight(myTab.rect.height() + heightOffset); if (myTab.documentMode) { p->save(); rotateTabPainter(p, myTab.shape, myTab.rect); QPalette np = tab->palette; np.setColor(QPalette::WindowText, QColor(255, 255, 255, 75)); QRect nr = subElementRect(SE_TabBarTabText, opt, w); nr.moveTop(-1); int alignment = Qt::AlignCenter | Qt::TextShowMnemonic | Qt::TextHideMnemonic; proxy()->drawItemText(p, nr, alignment, np, tab->state & State_Enabled, tab->text, QPalette::WindowText); p->restore(); } QCommonStyle::drawControl(ce, &myTab, p, w); } else { p->save(); CGContextSetShouldAntialias(cg, true); CGContextSetShouldSmoothFonts(cg, true); HIThemeTextInfo tti; tti.version = qt_mac_hitheme_version; tti.state = tds; QColor textColor = myTab.palette.windowText().color(); CGFloat colorComp[] = { textColor.redF(), textColor.greenF(), textColor.blueF(), textColor.alphaF() }; CGContextSetFillColorSpace(cg, QCoreGraphicsPaintEngine::macGenericColorSpace()); CGContextSetFillColor(cg, colorComp); switch (d->aquaSizeConstrain(opt, w)) { default: case QAquaSizeUnknown: case QAquaSizeLarge: tti.fontID = kThemeSystemFont; break; case QAquaSizeSmall: tti.fontID = kThemeSmallSystemFont; break; case QAquaSizeMini: tti.fontID = kThemeMiniSystemFont; break; } tti.horizontalFlushness = kHIThemeTextHorizontalFlushCenter; tti.verticalFlushness = kHIThemeTextVerticalFlushCenter; tti.options = verticalTabs ? kHIThemeTextBoxOptionStronglyVertical : kHIThemeTextBoxOptionNone; tti.truncationPosition = kHIThemeTextTruncationNone; tti.truncationMaxLines = 1 + myTab.text.count(QLatin1Char('\n')); QCFString tabText = qt_mac_removeMnemonics(myTab.text); QRect r = myTab.rect.adjusted(0, 0, 0, -1); HIRect bounds = qt_hirectForQRect(r); HIThemeDrawTextBox(tabText, &bounds, &tti, cg, kHIThemeOrientationNormal); p->restore(); } } break; case CE_DockWidgetTitle: if (const QDockWidget *dockWidget = qobject_cast(w)) { bool floating = dockWidget->isFloating(); if (floating) { ThemeDrawState tds = d->getDrawState(opt->state); HIThemeWindowDrawInfo wdi; wdi.version = qt_mac_hitheme_version; wdi.state = tds; wdi.windowType = kThemeMovableDialogWindow; wdi.titleHeight = opt->rect.height(); wdi.titleWidth = opt->rect.width(); wdi.attributes = 0; HIRect titleBarRect; HIRect tmpRect = qt_hirectForQRect(opt->rect); { QCFType titleRegion; QRect newr = opt->rect.adjusted(0, 0, 2, 0); HIThemeGetWindowShape(&tmpRect, &wdi, kWindowTitleBarRgn, &titleRegion); ptrHIShapeGetBounds(titleRegion, &tmpRect); newr.translate(newr.x() - int(tmpRect.origin.x), newr.y() - int(tmpRect.origin.y)); titleBarRect = qt_hirectForQRect(newr); } QMacCGContext cg(p); HIThemeDrawWindowFrame(&titleBarRect, &wdi, cg, kHIThemeOrientationNormal, 0); } else { // fill title bar background QLinearGradient linearGrad(0, opt->rect.top(), 0, opt->rect.bottom()); linearGrad.setColorAt(0, mainWindowGradientBegin); linearGrad.setColorAt(1, mainWindowGradientEnd); p->fillRect(opt->rect, linearGrad); // draw horizontal lines at top and bottom p->save(); p->setPen(mainWindowGradientBegin.lighter(114)); p->drawLine(opt->rect.topLeft(), opt->rect.topRight()); p->setPen(mainWindowGradientEnd.darker(114)); p->drawLine(opt->rect.bottomLeft(), opt->rect.bottomRight()); p->restore(); } } // Draw the text... if (const QStyleOptionDockWidget *dwOpt = qstyleoption_cast(opt)) { if (!dwOpt->title.isEmpty()) { const QStyleOptionDockWidgetV2 *v2 = qstyleoption_cast(dwOpt); bool verticalTitleBar = v2 == 0 ? false : v2->verticalTitleBar; QRect titleRect = subElementRect(SE_DockWidgetTitleBarText, opt, w); if (verticalTitleBar) { QRect rect = dwOpt->rect; QRect r = rect; QSize s = r.size(); s.transpose(); r.setSize(s); titleRect = QRect(r.left() + rect.bottom() - titleRect.bottom(), r.top() + titleRect.left() - rect.left(), titleRect.height(), titleRect.width()); p->translate(r.left(), r.top() + r.width()); p->rotate(-90); p->translate(-r.left(), -r.top()); } QFont oldFont = p->font(); p->setFont(qt_app_fonts_hash()->value("QToolButton", p->font())); QString text = p->fontMetrics().elidedText(dwOpt->title, Qt::ElideRight, titleRect.width()); drawItemText(p, titleRect, Qt::AlignCenter | Qt::TextShowMnemonic, dwOpt->palette, dwOpt->state & State_Enabled, text, QPalette::WindowText); p->setFont(oldFont); } } break; case CE_FocusFrame: { int xOff = proxy()->pixelMetric(PM_FocusFrameHMargin, opt, w) + 1; int yOff = proxy()->pixelMetric(PM_FocusFrameVMargin, opt, w) + 1; HIRect hirect = CGRectMake(xOff+opt->rect.x(), yOff+opt->rect.y(), opt->rect.width() - 2 * xOff, opt->rect.height() - 2 * yOff); HIThemeDrawFocusRect(&hirect, true, QMacCGContext(p), kHIThemeOrientationNormal); break; } case CE_MenuItem: case CE_MenuEmptyArea: if (const QStyleOptionMenuItem *mi = qstyleoption_cast(opt)) { p->fillRect(mi->rect, opt->palette.background()); QAquaWidgetSize widgetSize = d->aquaSizeConstrain(opt, w); int tabwidth = mi->tabWidth; int maxpmw = mi->maxIconWidth; bool active = mi->state & State_Selected; bool enabled = mi->state & State_Enabled; HIRect menuRect = qt_hirectForQRect(mi->menuRect); HIRect itemRect = qt_hirectForQRect(mi->rect); HIThemeMenuItemDrawInfo mdi; mdi.version = qt_mac_hitheme_version; mdi.itemType = kThemeMenuItemPlain; if (!mi->icon.isNull()) mdi.itemType |= kThemeMenuItemHasIcon; if (mi->menuItemType == QStyleOptionMenuItem::SubMenu) mdi.itemType |= kThemeMenuItemHierarchical | kThemeMenuItemHierBackground; else mdi.itemType |= kThemeMenuItemPopUpBackground; if (enabled) mdi.state = kThemeMenuActive; else mdi.state = kThemeMenuDisabled; if (active) mdi.state |= kThemeMenuSelected; QRect contentRect; if (mi->menuItemType == QStyleOptionMenuItem::Separator) { // First arg should be &menurect, but wacky stuff happens then. HIThemeDrawMenuSeparator(&itemRect, &itemRect, &mdi, cg, kHIThemeOrientationNormal); break; } else { HIRect cr; bool needAlpha = mi->palette.color(QPalette::Button) == Qt::transparent; if (needAlpha) { needAlpha = true; CGContextSaveGState(cg); CGContextSetAlpha(cg, 0.0); } HIThemeDrawMenuItem(&menuRect, &itemRect, &mdi, cg, kHIThemeOrientationNormal, &cr); if (needAlpha) CGContextRestoreGState(cg); if (ce == CE_MenuEmptyArea) break; contentRect = qt_qrectForHIRect(cr); } int xpos = contentRect.x() + 18; int checkcol = maxpmw; if (!enabled) p->setPen(mi->palette.text().color()); else if (active) p->setPen(mi->palette.highlightedText().color()); else p->setPen(mi->palette.buttonText().color()); if (mi->checked) { // Use the HIThemeTextInfo foo to draw the check mark correctly, if we do it, // we somehow need to use a special encoding as it doesn't look right with our // drawText(). p->save(); CGContextSetShouldAntialias(cg, true); CGContextSetShouldSmoothFonts(cg, true); QColor textColor = p->pen().color(); CGFloat colorComp[] = { textColor.redF(), textColor.greenF(), textColor.blueF(), textColor.alphaF() }; CGContextSetFillColorSpace(cg, QCoreGraphicsPaintEngine::macGenericColorSpace()); CGContextSetFillColor(cg, colorComp); HIThemeTextInfo tti; tti.version = qt_mac_hitheme_version; tti.state = tds; if (active && enabled) tti.state = kThemeStatePressed; switch (widgetSize) { case QAquaSizeUnknown: case QAquaSizeLarge: tti.fontID = kThemeMenuItemMarkFont; break; case QAquaSizeSmall: tti.fontID = kThemeSmallSystemFont; break; case QAquaSizeMini: tti.fontID = kThemeMiniSystemFont; break; } tti.horizontalFlushness = kHIThemeTextHorizontalFlushLeft; tti.verticalFlushness = kHIThemeTextVerticalFlushCenter; tti.options = kHIThemeTextBoxOptionNone; tti.truncationPosition = kHIThemeTextTruncationNone; tti.truncationMaxLines = 1; QCFString checkmark; #if 0 if (mi->checkType == QStyleOptionMenuItem::Exclusive) checkmark = QString(QChar(kDiamondUnicode)); else #endif checkmark = QString(QChar(kCheckUnicode)); int mw = checkcol + macItemFrame; int mh = contentRect.height() - 2 * macItemFrame; int xp = contentRect.x(); xp += macItemFrame; CGFloat outWidth, outHeight, outBaseline; HIThemeGetTextDimensions(checkmark, 0, &tti, &outWidth, &outHeight, &outBaseline); if (widgetSize == QAquaSizeMini) outBaseline += 1; QRect r(xp, contentRect.y(), mw, mh); r.translate(0, p->fontMetrics().ascent() - int(outBaseline) + 1); HIRect bounds = qt_hirectForQRect(r); HIThemeDrawTextBox(checkmark, &bounds, &tti, cg, kHIThemeOrientationNormal); p->restore(); } if (!mi->icon.isNull()) { QIcon::Mode mode = (mi->state & State_Enabled) ? QIcon::Normal : QIcon::Disabled; // Always be normal or disabled to follow the Mac style. int smallIconSize = proxy()->pixelMetric(PM_SmallIconSize); QSize iconSize(smallIconSize, smallIconSize); if (const QComboBox *comboBox = qobject_cast(w)) { iconSize = comboBox->iconSize(); } QPixmap pixmap = mi->icon.pixmap(iconSize, mode); int pixw = pixmap.width(); int pixh = pixmap.height(); QRect cr(xpos, contentRect.y(), checkcol, contentRect.height()); QRect pmr(0, 0, pixw, pixh); pmr.moveCenter(cr.center()); p->drawPixmap(pmr.topLeft(), pixmap); xpos += pixw + 6; } QString s = mi->text; if (!s.isEmpty()) { int t = s.indexOf(QLatin1Char('\t')); int text_flags = Qt::AlignRight | Qt::AlignVCenter | Qt::TextHideMnemonic | Qt::TextSingleLine | Qt::AlignAbsolute; int yPos = contentRect.y(); if (widgetSize == QAquaSizeMini) yPos += 1; p->save(); if (t >= 0) { p->setFont(qt_app_fonts_hash()->value("QMenuItem", p->font())); int xp = contentRect.right() - tabwidth - macRightBorder - macItemHMargin - macItemFrame + 1; p->drawText(xp, yPos, tabwidth, contentRect.height(), text_flags, s.mid(t + 1)); s = s.left(t); } const int xm = macItemFrame + maxpmw + macItemHMargin; QFont myFont = mi->font; // myFont may not have any "hard" flags set. We override // the point size so that when it is resolved against the device, this font will win. // This is mainly to handle cases where someone sets the font on the window // and then the combo inherits it and passes it onward. At that point the resolve mask // is very, very weak. This makes it stonger. myFont.setPointSizeF(QFontInfo(mi->font).pointSizeF()); p->setFont(myFont); p->drawText(xpos, yPos, contentRect.width() - xm - tabwidth + 1, contentRect.height(), text_flags ^ Qt::AlignRight, s); p->restore(); } } break; case CE_MenuHMargin: case CE_MenuVMargin: case CE_MenuTearoff: case CE_MenuScroller: if (const QStyleOptionMenuItem *mi = qstyleoption_cast(opt)) { p->fillRect(mi->rect, opt->palette.background()); HIRect menuRect = qt_hirectForQRect(mi->menuRect); HIRect itemRect = qt_hirectForQRect(mi->rect); HIThemeMenuItemDrawInfo mdi; mdi.version = qt_mac_hitheme_version; if (!(opt->state & State_Enabled)) mdi.state = kThemeMenuDisabled; else if (opt->state & State_Selected) mdi.state = kThemeMenuSelected; else mdi.state = kThemeMenuActive; if (ce == CE_MenuScroller) { if (opt->state & State_DownArrow) mdi.itemType = kThemeMenuItemScrollDownArrow; else mdi.itemType = kThemeMenuItemScrollUpArrow; } else { mdi.itemType = kThemeMenuItemPlain; } HIThemeDrawMenuItem(&menuRect, &itemRect, &mdi, cg, kHIThemeOrientationNormal, 0); if (ce == CE_MenuTearoff) { p->setPen(QPen(mi->palette.dark().color(), 1, Qt::DashLine)); p->drawLine(mi->rect.x() + 2, mi->rect.y() + mi->rect.height() / 2 - 1, mi->rect.x() + mi->rect.width() - 4, mi->rect.y() + mi->rect.height() / 2 - 1); p->setPen(QPen(mi->palette.light().color(), 1, Qt::DashLine)); p->drawLine(mi->rect.x() + 2, mi->rect.y() + mi->rect.height() / 2, mi->rect.x() + mi->rect.width() - 4, mi->rect.y() + mi->rect.height() / 2); } } break; case CE_MenuBarItem: if (const QStyleOptionMenuItem *mi = qstyleoption_cast(opt)) { HIRect menuRect = qt_hirectForQRect(mi->menuRect); HIRect itemRect = qt_hirectForQRect(mi->rect); if ((opt->state & State_Selected) && (opt->state & State_Enabled) && (opt->state & State_Sunken)){ // Draw a selected menu item background: HIThemeMenuItemDrawInfo mdi; mdi.version = qt_mac_hitheme_version; mdi.state = kThemeMenuSelected; mdi.itemType = kThemeMenuItemPlain; HIThemeDrawMenuItem(&menuRect, &itemRect, &mdi, cg, kHIThemeOrientationNormal, 0); } else { // Draw the toolbar background: HIThemeMenuBarDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = kThemeMenuBarNormal; bdi.attributes = 0; HIRect hirect = qt_hirectForQRect(mi->rect); HIThemeDrawMenuBarBackground(&menuRect, &bdi, cg, kHIThemeOrientationNormal); } if (!mi->icon.isNull()) { drawItemPixmap(p, mi->rect, Qt::AlignCenter | Qt::TextHideMnemonic | Qt::TextDontClip | Qt::TextSingleLine, mi->icon.pixmap(proxy()->pixelMetric(PM_SmallIconSize), (mi->state & State_Enabled) ? QIcon::Normal : QIcon::Disabled)); } else { drawItemText(p, mi->rect, Qt::AlignCenter | Qt::TextHideMnemonic | Qt::TextDontClip | Qt::TextSingleLine, mi->palette, mi->state & State_Enabled, mi->text, QPalette::ButtonText); } } break; case CE_MenuBarEmptyArea: if (const QStyleOptionMenuItem *mi = qstyleoption_cast(opt)) { HIThemeMenuBarDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = kThemeMenuBarNormal; bdi.attributes = 0; HIRect hirect = qt_hirectForQRect(mi->rect); HIThemeDrawMenuBarBackground(&hirect, &bdi, cg, kHIThemeOrientationNormal); break; } case CE_ProgressBarContents: if (const QStyleOptionProgressBar *pb = qstyleoption_cast(opt)) { HIThemeTrackDrawInfo tdi; tdi.version = qt_mac_hitheme_version; tdi.reserved = 0; bool isIndeterminate = (pb->minimum == 0 && pb->maximum == 0); bool vertical = false; bool inverted = false; if (const QStyleOptionProgressBarV2 *pb2 = qstyleoption_cast(opt)) { vertical = (pb2->orientation == Qt::Vertical); inverted = pb2->invertedAppearance; } bool reverse = (!vertical && (pb->direction == Qt::RightToLeft)); if (inverted) reverse = !reverse; switch (d->aquaSizeConstrain(opt, w)) { case QAquaSizeUnknown: case QAquaSizeLarge: tdi.kind = !isIndeterminate ? kThemeLargeProgressBar : kThemeLargeIndeterminateBar; break; case QAquaSizeMini: case QAquaSizeSmall: tdi.kind = !isIndeterminate ? kThemeProgressBar : kThemeIndeterminateBar; break; } tdi.bounds = qt_hirectForQRect(pb->rect); tdi.max = pb->maximum; tdi.min = pb->minimum; tdi.value = pb->progress; tdi.attributes = vertical ? 0 : kThemeTrackHorizontal; tdi.trackInfo.progress.phase = d->progressFrame; if (!(pb->state & State_Active)) tdi.enableState = kThemeTrackInactive; else if (!(pb->state & State_Enabled)) tdi.enableState = kThemeTrackDisabled; else tdi.enableState = kThemeTrackActive; HIThemeOrientation drawOrientation = kHIThemeOrientationNormal; if (reverse) { if (vertical) { drawOrientation = kHIThemeOrientationInverted; } else { CGContextSaveGState(cg); CGContextTranslateCTM(cg, pb->rect.width(), 0); CGContextScaleCTM(cg, -1, 1); } } HIThemeDrawTrack(&tdi, 0, cg, drawOrientation); if (reverse && !vertical) CGContextRestoreGState(cg); } break; case CE_ProgressBarLabel: case CE_ProgressBarGroove: break; case CE_SizeGrip: { if (w && w->testAttribute(Qt::WA_MacOpaqueSizeGrip)) { HIThemeGrowBoxDrawInfo gdi; gdi.version = qt_mac_hitheme_version; gdi.state = tds; gdi.kind = kHIThemeGrowBoxKindNormal; gdi.direction = kThemeGrowRight | kThemeGrowDown; gdi.size = kHIThemeGrowBoxSizeNormal; HIPoint pt = CGPointMake(opt->rect.x(), opt->rect.y()); HIThemeDrawGrowBox(&pt, &gdi, cg, kHIThemeOrientationNormal); } else { // It isn't possible to draw a transparent size grip with the // native API, so we do it ourselves here. const bool metal = qt_mac_is_metal(w); QPen lineColor = metal ? QColor(236, 236, 236) : QColor(82, 82, 82, 192); QPen metalHighlight = QColor(5, 5, 5, 192); lineColor.setWidth(1); p->save(); p->setRenderHint(QPainter::Antialiasing); p->setPen(lineColor); const Qt::LayoutDirection layoutDirection = w ? w->layoutDirection() : qApp->layoutDirection(); const int NumLines = metal ? 4 : 3; for (int l = 0; l < NumLines; ++l) { const int offset = (l * 4 + (metal ? 2 : 3)); QPoint start, end; if (layoutDirection == Qt::LeftToRight) { start = QPoint(opt->rect.width() - offset, opt->rect.height() - 1); end = QPoint(opt->rect.width() - 1, opt->rect.height() - offset); } else { start = QPoint(offset, opt->rect.height() - 1); end = QPoint(1, opt->rect.height() - offset); } p->drawLine(start, end); if (metal) { p->setPen(metalHighlight); p->setRenderHint(QPainter::Antialiasing, false); p->drawLine(start + QPoint(0, -1), end + QPoint(0, -1)); p->setRenderHint(QPainter::Antialiasing, true); p->setPen(lineColor); } } p->restore(); } break; } case CE_Splitter: { HIThemeSplitterDrawInfo sdi; sdi.version = qt_mac_hitheme_version; sdi.state = tds; sdi.adornment = qt_mac_is_metal(w) ? kHIThemeSplitterAdornmentMetal : kHIThemeSplitterAdornmentNone; HIRect hirect = qt_hirectForQRect(opt->rect); HIThemeDrawPaneSplitter(&hirect, &sdi, cg, kHIThemeOrientationNormal); break; } case CE_RubberBand: if (const QStyleOptionRubberBand *rubber = qstyleoption_cast(opt)) { QColor fillColor(opt->palette.color(QPalette::Disabled, QPalette::Highlight)); if (!rubber->opaque) { QColor strokeColor; // I retrieved these colors from the Carbon-Dev mailing list strokeColor.setHsvF(0, 0, 0.86, 1.0); fillColor.setHsvF(0, 0, 0.53, 0.25); if (opt->rect.width() * opt->rect.height() <= 3) { p->fillRect(opt->rect, strokeColor); } else { QPen oldPen = p->pen(); QBrush oldBrush = p->brush(); QPen pen(strokeColor); p->setPen(pen); p->setBrush(fillColor); p->drawRect(opt->rect.adjusted(0, 0, -1, -1)); p->setPen(oldPen); p->setBrush(oldBrush); } } else { p->fillRect(opt->rect, fillColor); } } break; case CE_ToolBar: { // For unified tool bars, draw nothing. if (w) { if (QMainWindow * mainWindow = qobject_cast(w->window())) { if (mainWindow->unifiedTitleAndToolBarOnMac()) break; } } // draw background gradient QLinearGradient linearGrad; if (opt->state & State_Horizontal) linearGrad = QLinearGradient(0, opt->rect.top(), 0, opt->rect.bottom()); else linearGrad = QLinearGradient(opt->rect.left(), 0, opt->rect.right(), 0); linearGrad.setColorAt(0, mainWindowGradientBegin); linearGrad.setColorAt(1, mainWindowGradientEnd); p->fillRect(opt->rect, linearGrad); p->save(); if (opt->state & State_Horizontal) { p->setPen(mainWindowGradientBegin.lighter(114)); p->drawLine(opt->rect.topLeft(), opt->rect.topRight()); p->setPen(mainWindowGradientEnd.darker(114)); p->drawLine(opt->rect.bottomLeft(), opt->rect.bottomRight()); } else { p->setPen(mainWindowGradientBegin.lighter(114)); p->drawLine(opt->rect.topLeft(), opt->rect.bottomLeft()); p->setPen(mainWindowGradientEnd.darker(114)); p->drawLine(opt->rect.topRight(), opt->rect.bottomRight()); } p->restore(); } break; default: QWindowsStyle::drawControl(ce, opt, p, w); break; } } static void setLayoutItemMargins(int left, int top, int right, int bottom, QRect *rect, Qt::LayoutDirection dir) { if (dir == Qt::RightToLeft) { rect->adjust(-right, top, -left, bottom); } else { rect->adjust(left, top, right, bottom); } } QRect QMacStyle::subElementRect(SubElement sr, const QStyleOption *opt, const QWidget *widget) const { QRect rect; int controlSize = getControlSize(opt, widget); switch (sr) { case SE_ToolBoxTabContents: rect = QCommonStyle::subElementRect(sr, opt, widget); break; case SE_PushButtonContents: if (const QStyleOptionButton *btn = qstyleoption_cast(opt)) { // Unlike Carbon, we want the button to always be drawn inside its bounds. // Therefore, the button is a bit smaller, so that even if it got focus, // the focus 'shadow' will be inside. Adjust the content rect likewise. HIThemeButtonDrawInfo bdi; d->initHIThemePushButton(btn, widget, d->getDrawState(opt->state), &bdi); HIRect contentRect = d->pushButtonContentBounds(btn, &bdi); rect = qt_qrectForHIRect(contentRect); } break; case SE_HeaderLabel: if (qstyleoption_cast(opt)) { rect = QWindowsStyle::subElementRect(sr, opt, widget); if (widget && widget->height() <= qt_mac_aqua_get_metric(kThemeMetricListHeaderHeight)){ // We need to allow the text a bit more space when the header is as // small as kThemeMetricListHeaderHeight, otherwise it gets clipped: rect.setY(0); rect.setHeight(widget->height()); } } break; case SE_ProgressBarGroove: case SE_ProgressBarLabel: break; case SE_ProgressBarContents: rect = opt->rect; break; case SE_TreeViewDisclosureItem: { HIRect inRect = CGRectMake(opt->rect.x(), opt->rect.y(), opt->rect.width(), opt->rect.height()); HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = kThemeStateActive; bdi.kind = kThemeDisclosureButton; bdi.value = kThemeDisclosureRight; bdi.adornment = kThemeAdornmentNone; HIRect contentRect; HIThemeGetButtonContentBounds(&inRect, &bdi, &contentRect); QCFType shape; HIRect outRect; HIThemeGetButtonShape(&inRect, &bdi, &shape); ptrHIShapeGetBounds(shape, &outRect); rect = QRect(int(outRect.origin.x), int(outRect.origin.y), int(contentRect.origin.x - outRect.origin.x), int(outRect.size.height)); break; } case SE_TabWidgetLeftCorner: if (const QStyleOptionTabWidgetFrame *twf = qstyleoption_cast(opt)) { switch (twf->shape) { case QTabBar::RoundedNorth: case QTabBar::TriangularNorth: rect = QRect(QPoint(0, 0), twf->leftCornerWidgetSize); break; case QTabBar::RoundedSouth: case QTabBar::TriangularSouth: rect = QRect(QPoint(0, twf->rect.height() - twf->leftCornerWidgetSize.height()), twf->leftCornerWidgetSize); break; default: break; } rect = visualRect(twf->direction, twf->rect, rect); } break; case SE_TabWidgetRightCorner: if (const QStyleOptionTabWidgetFrame *twf = qstyleoption_cast(opt)) { switch (twf->shape) { case QTabBar::RoundedNorth: case QTabBar::TriangularNorth: rect = QRect(QPoint(twf->rect.width() - twf->rightCornerWidgetSize.width(), 0), twf->rightCornerWidgetSize); break; case QTabBar::RoundedSouth: case QTabBar::TriangularSouth: rect = QRect(QPoint(twf->rect.width() - twf->rightCornerWidgetSize.width(), twf->rect.height() - twf->rightCornerWidgetSize.height()), twf->rightCornerWidgetSize); break; default: break; } rect = visualRect(twf->direction, twf->rect, rect); } break; case SE_TabWidgetTabContents: rect = QWindowsStyle::subElementRect(sr, opt, widget); if (const QStyleOptionTabWidgetFrame *twf = qstyleoption_cast(opt)) { if (twf->lineWidth != 0) { switch (getTabDirection(twf->shape)) { case kThemeTabNorth: rect.adjust(+1, +14, -1, -1); break; case kThemeTabSouth: rect.adjust(+1, +1, -1, -14); break; case kThemeTabWest: rect.adjust(+14, +1, -1, -1); break; case kThemeTabEast: rect.adjust(+1, +1, -14, -1); } } } break; case SE_LineEditContents: rect = QWindowsStyle::subElementRect(sr, opt, widget); if(widget->parentWidget() && qobject_cast(widget->parentWidget())) rect.adjust(-1, -2, 0, 0); else rect.adjust(-1, 0, 0, +1); break; case SE_CheckBoxLayoutItem: rect = opt->rect; if (controlSize == QAquaSizeLarge) { setLayoutItemMargins(+2, +3, -9, -4, &rect, opt->direction); } else if (controlSize == QAquaSizeSmall) { setLayoutItemMargins(+1, +5, 0 /* fix */, -6, &rect, opt->direction); } else { setLayoutItemMargins(0, +7, 0 /* fix */, -6, &rect, opt->direction); } break; case SE_ComboBoxLayoutItem: if (widget && qobject_cast(widget->parentWidget())) { // Do nothing, because QToolbar needs the entire widget rect. // Otherwise it will be clipped. Equivalent to // widget->setAttribute(Qt::WA_LayoutUsesWidgetRect), but without // all the hassle. } else { rect = opt->rect; if (controlSize == QAquaSizeLarge) { rect.adjust(+3, +2, -3, -4); } else if (controlSize == QAquaSizeSmall) { setLayoutItemMargins(+2, +1, -3, -4, &rect, opt->direction); } else { setLayoutItemMargins(+1, 0, -2, 0, &rect, opt->direction); } } break; case SE_LabelLayoutItem: rect = opt->rect; setLayoutItemMargins(+1, 0 /* SHOULD be -1, done for alignment */, 0, 0 /* SHOULD be -1, done for alignment */, &rect, opt->direction); break; case SE_ProgressBarLayoutItem: { rect = opt->rect; int bottom = SIZE(3, 8, 8); if (opt->state & State_Horizontal) { rect.adjust(0, +1, 0, -bottom); } else { setLayoutItemMargins(+1, 0, -bottom, 0, &rect, opt->direction); } break; } case SE_PushButtonLayoutItem: if (const QStyleOptionButton *buttonOpt = qstyleoption_cast(opt)) { if ((buttonOpt->features & QStyleOptionButton::Flat)) break; // leave rect alone } rect = opt->rect; if (controlSize == QAquaSizeLarge) { rect.adjust(+6, +4, -6, -8); } else if (controlSize == QAquaSizeSmall) { rect.adjust(+5, +4, -5, -6); } else { rect.adjust(+1, 0, -1, -2); } break; case SE_RadioButtonLayoutItem: rect = opt->rect; if (controlSize == QAquaSizeLarge) { setLayoutItemMargins(+2, +2 /* SHOULD BE +3, done for alignment */, 0, -4 /* SHOULD BE -3, done for alignment */, &rect, opt->direction); } else if (controlSize == QAquaSizeSmall) { rect.adjust(0, +6, 0 /* fix */, -5); } else { rect.adjust(0, +6, 0 /* fix */, -7); } break; case SE_SliderLayoutItem: if (const QStyleOptionSlider *sliderOpt = qstyleoption_cast(opt)) { rect = opt->rect; if (sliderOpt->tickPosition == QSlider::NoTicks) { int above = SIZE(3, 0, 2); int below = SIZE(4, 3, 0); if (sliderOpt->orientation == Qt::Horizontal) { rect.adjust(0, +above, 0, -below); } else { rect.adjust(+above, 0, -below, 0); //### Seems that QSlider flip the position of the ticks in reverse mode. } } else if (sliderOpt->tickPosition == QSlider::TicksAbove) { int below = SIZE(3, 2, 0); if (sliderOpt->orientation == Qt::Horizontal) { rect.setHeight(rect.height() - below); } else { rect.setWidth(rect.width() - below); } } else if (sliderOpt->tickPosition == QSlider::TicksBelow) { int above = SIZE(3, 2, 0); if (sliderOpt->orientation == Qt::Horizontal) { rect.setTop(rect.top() + above); } else { rect.setLeft(rect.left() + above); } } } break; case SE_FrameLayoutItem: // hack because QStyleOptionFrameV2 doesn't have a frameStyle member if (const QFrame *frame = qobject_cast(widget)) { rect = opt->rect; switch (frame->frameStyle() & QFrame::Shape_Mask) { case QFrame::HLine: rect.adjust(0, +1, 0, -1); break; case QFrame::VLine: rect.adjust(+1, 0, -1, 0); break; default: ; } } break; case SE_GroupBoxLayoutItem: rect = opt->rect; if (const QStyleOptionGroupBox *groupBoxOpt = qstyleoption_cast(opt)) { /* AHIG is very inconsistent when it comes to group boxes. Basically, we make sure that (non-checkable) group boxes and tab widgets look good when laid out side by side. */ if (groupBoxOpt->subControls & (QStyle::SC_GroupBoxCheckBox | QStyle::SC_GroupBoxLabel)) { int delta; if (groupBoxOpt->subControls & QStyle::SC_GroupBoxCheckBox) { delta = SIZE(8, 4, 4); // guess } else { delta = SIZE(15, 12, 12); // guess } rect.setTop(rect.top() + delta); } } rect.setBottom(rect.bottom() - 1); break; case SE_TabWidgetLayoutItem: if (const QStyleOptionTabWidgetFrame *tabWidgetOpt = qstyleoption_cast(opt)) { /* AHIG specifies "12 or 14" as the distance from the window edge. We choose 14 and since the default top margin is 20, the overlap is 6. */ rect = tabWidgetOpt->rect; if (tabWidgetOpt->shape == QTabBar::RoundedNorth) rect.setTop(rect.top() + SIZE(6 /* AHIG */, 3 /* guess */, 2 /* AHIG */)); } break; default: rect = QWindowsStyle::subElementRect(sr, opt, widget); break; } return rect; } static inline void drawToolbarButtonArrow(const QRect &toolButtonRect, ThemeDrawState tds, CGContextRef cg) { QRect arrowRect = QRect(toolButtonRect.right() - 9, toolButtonRect.bottom() - 9, 7, 5); HIThemePopupArrowDrawInfo padi; padi.version = qt_mac_hitheme_version; padi.state = tds; padi.orientation = kThemeArrowDown; padi.size = kThemeArrow7pt; HIRect hirect = qt_hirectForQRect(arrowRect); HIThemeDrawPopupArrow(&hirect, &padi, cg, kHIThemeOrientationNormal); } void QMacStyle::drawComplexControl(ComplexControl cc, const QStyleOptionComplex *opt, QPainter *p, const QWidget *widget) const { ThemeDrawState tds = d->getDrawState(opt->state); QMacCGContext cg(p); switch (cc) { case CC_Slider: case CC_ScrollBar: if (const QStyleOptionSlider *slider = qstyleoption_cast(opt)) { HIThemeTrackDrawInfo tdi; d->getSliderInfo(cc, slider, &tdi, widget); if (slider->state & State_Sunken) { if (cc == CC_Slider) { if (slider->activeSubControls == SC_SliderHandle) tdi.trackInfo.slider.pressState = kThemeThumbPressed; else if (slider->activeSubControls == SC_SliderGroove) tdi.trackInfo.slider.pressState = kThemeLeftTrackPressed; } else { if (slider->activeSubControls == SC_ScrollBarSubLine || slider->activeSubControls == SC_ScrollBarAddLine) { // This test looks complex but it basically boils down // to the following: The "RTL look" on the mac also // changed the directions of the controls, that's not // what people expect (an arrow is an arrow), so we // kind of fake and say the opposite button is hit. // This works great, up until 10.4 which broke the // scroll bars, so I also have actually do something // similar when I have an upside down scroll bar // because on Tiger I only "fake" the reverse stuff. bool reverseHorizontal = (slider->direction == Qt::RightToLeft && slider->orientation == Qt::Horizontal); if ((reverseHorizontal && slider->activeSubControls == SC_ScrollBarAddLine) || (!reverseHorizontal && slider->activeSubControls == SC_ScrollBarSubLine)) { tdi.trackInfo.scrollbar.pressState = kThemeRightInsideArrowPressed | kThemeLeftOutsideArrowPressed; } else { tdi.trackInfo.scrollbar.pressState = kThemeLeftInsideArrowPressed | kThemeRightOutsideArrowPressed; } } else if (slider->activeSubControls == SC_ScrollBarAddPage) { tdi.trackInfo.scrollbar.pressState = kThemeRightTrackPressed; } else if (slider->activeSubControls == SC_ScrollBarSubPage) { tdi.trackInfo.scrollbar.pressState = kThemeLeftTrackPressed; } else if (slider->activeSubControls == SC_ScrollBarSlider) { tdi.trackInfo.scrollbar.pressState = kThemeThumbPressed; } } } HIRect macRect; bool tracking = slider->sliderPosition == slider->sliderValue; if (!tracking) { // Small optimization, the same as q->subControlRect QCFType shape; HIThemeGetTrackThumbShape(&tdi, &shape); ptrHIShapeGetBounds(shape, &macRect); tdi.value = slider->sliderValue; } // Remove controls from the scroll bar if it is to short to draw them correctly. // This is done in two stages: first the thumb indicator is removed when it is // no longer possible to move it, second the up/down buttons are removed when // there is not enough space for them. if (cc == CC_ScrollBar) { const int scrollBarLength = (slider->orientation == Qt::Horizontal) ? slider->rect.width() : slider->rect.height(); const QMacStyle::WidgetSizePolicy sizePolicy = widgetSizePolicy(widget); if (scrollBarLength < scrollButtonsCutoffSize(thumbIndicatorCutoff, sizePolicy)) tdi.attributes &= ~kThemeTrackShowThumb; if (scrollBarLength < scrollButtonsCutoffSize(scrollButtonsCutoff, sizePolicy)) tdi.enableState = kThemeTrackNothingToScroll; } HIThemeDrawTrack(&tdi, tracking ? 0 : &macRect, cg, kHIThemeOrientationNormal); if (cc == CC_Slider && slider->subControls & SC_SliderTickmarks) { if (qt_mac_is_metal(widget)) { if (tdi.enableState == kThemeTrackInactive) tdi.enableState = kThemeTrackActive; // Looks more Cocoa-like } int interval = slider->tickInterval; if (interval == 0) { interval = slider->pageStep; if (interval == 0) interval = slider->singleStep; if (interval == 0) interval = 1; } int numMarks = 1 + ((slider->maximum - slider->minimum) / interval); if (tdi.trackInfo.slider.thumbDir == kThemeThumbPlain) { // They asked for both, so we'll give it to them. tdi.trackInfo.slider.thumbDir = kThemeThumbDownward; HIThemeDrawTrackTickMarks(&tdi, numMarks, cg, kHIThemeOrientationNormal); tdi.trackInfo.slider.thumbDir = kThemeThumbUpward; HIThemeDrawTrackTickMarks(&tdi, numMarks, cg, kHIThemeOrientationNormal); } else { HIThemeDrawTrackTickMarks(&tdi, numMarks, cg, kHIThemeOrientationNormal); } } } break; case CC_Q3ListView: if (const QStyleOptionQ3ListView *lv = qstyleoption_cast(opt)) { if (lv->subControls & SC_Q3ListView) QWindowsStyle::drawComplexControl(cc, lv, p, widget); if (lv->subControls & (SC_Q3ListViewBranch | SC_Q3ListViewExpand)) { int y = lv->rect.y(); int h = lv->rect.height(); int x = lv->rect.right() - 10; for (int i = 1; i < lv->items.size() && y < h; ++i) { QStyleOptionQ3ListViewItem item = lv->items.at(i); if (y + item.height > 0 && (item.childCount > 0 || (item.features & (QStyleOptionQ3ListViewItem::Expandable | QStyleOptionQ3ListViewItem::Visible)) == (QStyleOptionQ3ListViewItem::Expandable | QStyleOptionQ3ListViewItem::Visible))) { QStyleOption treeOpt(0); treeOpt.rect.setRect(x, y + item.height / 2 - 4, 9, 9); treeOpt.palette = lv->palette; treeOpt.state = lv->state; treeOpt.state |= State_Children; if (item.state & State_Open) treeOpt.state |= State_Open; proxy()->drawPrimitive(PE_IndicatorBranch, &treeOpt, p, widget); } y += item.totalHeight; } } } break; case CC_SpinBox: if (const QStyleOptionSpinBox *sb = qstyleoption_cast(opt)) { QStyleOptionSpinBox newSB = *sb; if (sb->frame && (sb->subControls & SC_SpinBoxFrame)) { SInt32 frame_size; GetThemeMetric(kThemeMetricEditTextFrameOutset, &frame_size); QRect lineeditRect = proxy()->subControlRect(CC_SpinBox, sb, SC_SpinBoxEditField, widget); lineeditRect.adjust(-frame_size, -frame_size, +frame_size, +frame_size); HIThemeFrameDrawInfo fdi; fdi.version = qt_mac_hitheme_version; fdi.state = kThemeStateInactive; fdi.kind = kHIThemeFrameTextFieldSquare; fdi.isFocused = false; HIRect hirect = qt_hirectForQRect(lineeditRect); HIThemeDrawFrame(&hirect, &fdi, cg, kHIThemeOrientationNormal); } if (sb->subControls & (SC_SpinBoxUp | SC_SpinBoxDown)) { HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; QAquaWidgetSize aquaSize = d->aquaSizeConstrain(opt, widget); switch (aquaSize) { case QAquaSizeUnknown: case QAquaSizeLarge: bdi.kind = kThemeIncDecButton; break; case QAquaSizeMini: bdi.kind = kThemeIncDecButtonMini; break; case QAquaSizeSmall: bdi.kind = kThemeIncDecButtonSmall; break; } if (!(sb->stepEnabled & (QAbstractSpinBox::StepUpEnabled | QAbstractSpinBox::StepDownEnabled))) tds = kThemeStateUnavailable; if (sb->activeSubControls == SC_SpinBoxDown && (sb->state & State_Sunken)) tds = kThemeStatePressedDown; else if (sb->activeSubControls == SC_SpinBoxUp && (sb->state & State_Sunken)) tds = kThemeStatePressedUp; bdi.state = tds; if (!(sb->state & State_Active) && sb->palette.currentColorGroup() == QPalette::Active && tds == kThemeStateInactive) bdi.state = kThemeStateActive; bdi.value = kThemeButtonOff; bdi.adornment = kThemeAdornmentNone; QRect updown = proxy()->subControlRect(CC_SpinBox, sb, SC_SpinBoxUp, widget); updown |= proxy()->subControlRect(CC_SpinBox, sb, SC_SpinBoxDown, widget); HIRect newRect = qt_hirectForQRect(updown); QRect off_rct; HIRect outRect; HIThemeGetButtonBackgroundBounds(&newRect, &bdi, &outRect); off_rct.setRect(int(newRect.origin.x - outRect.origin.x), int(newRect.origin.y - outRect.origin.y), int(outRect.size.width - newRect.size.width), int(outRect.size.height - newRect.size.height)); newRect = qt_hirectForQRect(updown, off_rct); HIThemeDrawButton(&newRect, &bdi, cg, kHIThemeOrientationNormal, 0); } } break; case CC_ComboBox: if (const QStyleOptionComboBox *combo = qstyleoption_cast(opt)){ HIThemeButtonDrawInfo bdi; d->initComboboxBdi(combo, &bdi, widget, d->getDrawState(opt->state)); bool drawColorless = combo->palette.currentColorGroup() == QPalette::Active && tds == kThemeStateInactive; if (!drawColorless) QMacStylePrivate::drawCombobox(qt_hirectForQRect(combo->rect), bdi, p); else d->drawColorlessButton(qt_hirectForQRect(combo->rect), &bdi, p, opt); } break; case CC_TitleBar: if (const QStyleOptionTitleBar *titlebar = qstyleoption_cast(opt)) { if (titlebar->state & State_Active) { if (titlebar->titleBarState & State_Active) tds = kThemeStateActive; else tds = kThemeStateInactive; } else { tds = kThemeStateInactive; } HIThemeWindowDrawInfo wdi; wdi.version = qt_mac_hitheme_version; wdi.state = tds; wdi.windowType = QtWinType; wdi.titleHeight = titlebar->rect.height(); wdi.titleWidth = titlebar->rect.width(); wdi.attributes = kThemeWindowHasTitleText; // It seems HIThemeDrawTitleBarWidget is not able to draw a dirty // close button, so use HIThemeDrawWindowFrame instead. if (widget && widget->isWindowModified() && titlebar->subControls & SC_TitleBarCloseButton) wdi.attributes |= kThemeWindowHasCloseBox | kThemeWindowHasDirty; HIRect titleBarRect; HIRect tmpRect = qt_hirectForQRect(titlebar->rect); { QCFType titleRegion; QRect newr = titlebar->rect.adjusted(0, 0, 2, 0); HIThemeGetWindowShape(&tmpRect, &wdi, kWindowTitleBarRgn, &titleRegion); ptrHIShapeGetBounds(titleRegion, &tmpRect); newr.translate(newr.x() - int(tmpRect.origin.x), newr.y() - int(tmpRect.origin.y)); titleBarRect = qt_hirectForQRect(newr); } HIThemeDrawWindowFrame(&titleBarRect, &wdi, cg, kHIThemeOrientationNormal, 0); if (titlebar->subControls & (SC_TitleBarCloseButton | SC_TitleBarMaxButton | SC_TitleBarMinButton | SC_TitleBarNormalButton)) { HIThemeWindowWidgetDrawInfo wwdi; wwdi.version = qt_mac_hitheme_version; wwdi.widgetState = tds; if (titlebar->state & State_MouseOver) wwdi.widgetState = kThemeStateRollover; wwdi.windowType = QtWinType; wwdi.attributes = wdi.attributes | kThemeWindowHasFullZoom | kThemeWindowHasCloseBox | kThemeWindowHasCollapseBox; wwdi.windowState = wdi.state; wwdi.titleHeight = wdi.titleHeight; wwdi.titleWidth = wdi.titleWidth; ThemeDrawState savedControlState = wwdi.widgetState; uint sc = SC_TitleBarMinButton; ThemeTitleBarWidget tbw = kThemeWidgetCollapseBox; bool active = titlebar->state & State_Active; if (qMacVersion() < QSysInfo::MV_10_6) { int border = 2; titleBarRect.origin.x += border; titleBarRect.origin.y -= border; } while (sc <= SC_TitleBarCloseButton) { if (sc & titlebar->subControls) { uint tmp = sc; wwdi.widgetState = savedControlState; wwdi.widgetType = tbw; if (sc == SC_TitleBarMinButton) tmp |= SC_TitleBarNormalButton; if (active && (titlebar->activeSubControls & tmp) && (titlebar->state & State_Sunken)) wwdi.widgetState = kThemeStatePressed; // Draw all sub controllers except the dirty close button // (it is already handled by HIThemeDrawWindowFrame). if (!(widget && widget->isWindowModified() && tbw == kThemeWidgetCloseBox)) { HIThemeDrawTitleBarWidget(&titleBarRect, &wwdi, cg, kHIThemeOrientationNormal); p->paintEngine()->syncState(); } } sc = sc << 1; tbw = tbw >> 1; } } p->paintEngine()->syncState(); if (titlebar->subControls & SC_TitleBarLabel) { int iw = 0; if (!titlebar->icon.isNull()) { QCFType titleRegion2; HIThemeGetWindowShape(&titleBarRect, &wdi, kWindowTitleProxyIconRgn, &titleRegion2); ptrHIShapeGetBounds(titleRegion2, &tmpRect); if (tmpRect.size.width != 1) { int iconExtent = proxy()->pixelMetric(PM_SmallIconSize); iw = titlebar->icon.actualSize(QSize(iconExtent, iconExtent)).width(); } } if (!titlebar->text.isEmpty()) { p->save(); QCFType titleRegion3; HIThemeGetWindowShape(&titleBarRect, &wdi, kWindowTitleTextRgn, &titleRegion3); ptrHIShapeGetBounds(titleRegion3, &tmpRect); p->setClipRect(qt_qrectForHIRect(tmpRect)); QRect br = p->clipRegion().boundingRect(); int x = br.x(), y = br.y() + (titlebar->rect.height() / 2 - p->fontMetrics().height() / 2); if (br.width() <= (p->fontMetrics().width(titlebar->text) + iw * 2)) x += iw; else x += br.width() / 2 - p->fontMetrics().width(titlebar->text) / 2; if (iw) p->drawPixmap(x - iw, y, titlebar->icon.pixmap(proxy()->pixelMetric(PM_SmallIconSize), QIcon::Normal)); drawItemText(p, br, Qt::AlignCenter, opt->palette, tds == kThemeStateActive, titlebar->text, QPalette::Text); p->restore(); } } } break; case CC_GroupBox: if (const QStyleOptionGroupBox *groupBox = qstyleoption_cast(opt)) { QStyleOptionGroupBox groupBoxCopy(*groupBox); if ((widget && !widget->testAttribute(Qt::WA_SetFont)) && QApplication::desktopSettingsAware()) groupBoxCopy.subControls = groupBoxCopy.subControls & ~SC_GroupBoxLabel; QWindowsStyle::drawComplexControl(cc, &groupBoxCopy, p, widget); if (groupBoxCopy.subControls != groupBox->subControls) { bool checkable = groupBox->subControls & SC_GroupBoxCheckBox; p->save(); CGContextSetShouldAntialias(cg, true); CGContextSetShouldSmoothFonts(cg, true); HIThemeTextInfo tti; tti.version = qt_mac_hitheme_version; tti.state = tds; QColor textColor = groupBox->palette.windowText().color(); CGFloat colorComp[] = { textColor.redF(), textColor.greenF(), textColor.blueF(), textColor.alphaF() }; CGContextSetFillColorSpace(cg, QCoreGraphicsPaintEngine::macGenericColorSpace()); CGContextSetFillColor(cg, colorComp); tti.fontID = checkable ? kThemeSystemFont : kThemeSmallSystemFont; tti.horizontalFlushness = kHIThemeTextHorizontalFlushCenter; tti.verticalFlushness = kHIThemeTextVerticalFlushCenter; tti.options = kHIThemeTextBoxOptionNone; tti.truncationPosition = kHIThemeTextTruncationNone; tti.truncationMaxLines = 1 + groupBox->text.count(QLatin1Char('\n')); QCFString groupText = qt_mac_removeMnemonics(groupBox->text); QRect r = proxy()->subControlRect(CC_GroupBox, groupBox, SC_GroupBoxLabel, widget); HIRect bounds = qt_hirectForQRect(r); HIThemeDrawTextBox(groupText, &bounds, &tti, cg, kHIThemeOrientationNormal); p->restore(); } } break; case CC_ToolButton: if (const QStyleOptionToolButton *tb = qstyleoption_cast(opt)) { if (widget && qobject_cast(widget->parentWidget())) { if (tb->subControls & SC_ToolButtonMenu) { QStyleOption arrowOpt(0); arrowOpt.rect = proxy()->subControlRect(cc, tb, SC_ToolButtonMenu, widget); arrowOpt.rect.setY(arrowOpt.rect.y() + arrowOpt.rect.height() / 2); arrowOpt.rect.setHeight(arrowOpt.rect.height() / 2); arrowOpt.state = tb->state; arrowOpt.palette = tb->palette; proxy()->drawPrimitive(PE_IndicatorArrowDown, &arrowOpt, p, widget); } else if ((tb->features & QStyleOptionToolButton::HasMenu) && (tb->toolButtonStyle != Qt::ToolButtonTextOnly && !tb->icon.isNull())) { drawToolbarButtonArrow(tb->rect, tds, cg); } if (tb->state & State_On) { if (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_5) { static QPixmap pm(QLatin1String(":/trolltech/mac/style/images/leopard-unified-toolbar-on.png")); p->setRenderHint(QPainter::SmoothPixmapTransform); QStyleHelper::drawBorderPixmap(pm, p, tb->rect, 2, 2, 2, 2); } else { QPen oldPen = p->pen(); p->setPen(QColor(0, 0, 0, 0x3a)); p->fillRect(tb->rect.adjusted(1, 1, -1, -1), QColor(0, 0, 0, 0x12)); p->drawLine(tb->rect.left() + 1, tb->rect.top(), tb->rect.right() - 1, tb->rect.top()); p->drawLine(tb->rect.left() + 1, tb->rect.bottom(), tb->rect.right() - 1, tb->rect.bottom()); p->drawLine(tb->rect.topLeft(), tb->rect.bottomLeft()); p->drawLine(tb->rect.topRight(), tb->rect.bottomRight()); p->setPen(oldPen); } } proxy()->drawControl(CE_ToolButtonLabel, opt, p, widget); } else { ThemeButtonKind bkind = kThemeBevelButton; switch (d->aquaSizeConstrain(opt, widget)) { case QAquaSizeUnknown: case QAquaSizeLarge: bkind = kThemeBevelButton; break; case QAquaSizeMini: case QAquaSizeSmall: bkind = kThemeSmallBevelButton; break; } QRect button, menuarea; button = proxy()->subControlRect(cc, tb, SC_ToolButton, widget); menuarea = proxy()->subControlRect(cc, tb, SC_ToolButtonMenu, widget); State bflags = tb->state, mflags = tb->state; if (tb->subControls & SC_ToolButton) bflags |= State_Sunken; if (tb->subControls & SC_ToolButtonMenu) mflags |= State_Sunken; if (tb->subControls & SC_ToolButton) { if (bflags & (State_Sunken | State_On | State_Raised)) { HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = tds; bdi.adornment = kThemeAdornmentNone; bdi.kind = bkind; bdi.value = kThemeButtonOff; if (tb->state & State_HasFocus) bdi.adornment = kThemeAdornmentFocus; if (tb->state & State_Sunken) bdi.state = kThemeStatePressed; if (tb->state & State_On) bdi.value = kThemeButtonOn; QRect off_rct(0, 0, 0, 0); HIRect myRect, macRect; myRect = CGRectMake(tb->rect.x(), tb->rect.y(), tb->rect.width(), tb->rect.height()); HIThemeGetButtonBackgroundBounds(&myRect, &bdi, &macRect); off_rct.setRect(int(myRect.origin.x - macRect.origin.x), int(myRect.origin.y - macRect.origin.y), int(macRect.size.width - myRect.size.width), int(macRect.size.height - myRect.size.height)); myRect = qt_hirectForQRect(button, off_rct); HIThemeDrawButton(&myRect, &bdi, cg, kHIThemeOrientationNormal, 0); } } if (tb->subControls & SC_ToolButtonMenu) { HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = tds; bdi.value = kThemeButtonOff; bdi.adornment = kThemeAdornmentNone; bdi.kind = bkind; if (tb->state & State_HasFocus) bdi.adornment = kThemeAdornmentFocus; if (tb->state & (State_On | State_Sunken) || (tb->activeSubControls & SC_ToolButtonMenu)) bdi.state = kThemeStatePressed; HIRect hirect = qt_hirectForQRect(menuarea); HIThemeDrawButton(&hirect, &bdi, cg, kHIThemeOrientationNormal, 0); QRect r(menuarea.x() + ((menuarea.width() / 2) - 3), menuarea.height() - 8, 8, 8); HIThemePopupArrowDrawInfo padi; padi.version = qt_mac_hitheme_version; padi.state = tds; padi.orientation = kThemeArrowDown; padi.size = kThemeArrow7pt; hirect = qt_hirectForQRect(r); HIThemeDrawPopupArrow(&hirect, &padi, cg, kHIThemeOrientationNormal); } else if (tb->features & QStyleOptionToolButton::HasMenu) { drawToolbarButtonArrow(tb->rect, tds, cg); } QRect buttonRect = proxy()->subControlRect(CC_ToolButton, tb, SC_ToolButton, widget); int fw = proxy()->pixelMetric(PM_DefaultFrameWidth, opt, widget); QStyleOptionToolButton label = *tb; label.rect = buttonRect.adjusted(fw, fw, -fw, -fw); proxy()->drawControl(CE_ToolButtonLabel, &label, p, widget); } } break; case CC_Dial: if (const QStyleOptionSlider *dial = qstyleoption_cast(opt)) QStyleHelper::drawDial(dial, p); break; default: QWindowsStyle::drawComplexControl(cc, opt, p, widget); break; } } QStyle::SubControl QMacStyle::hitTestComplexControl(ComplexControl cc, const QStyleOptionComplex *opt, const QPoint &pt, const QWidget *widget) const { SubControl sc = QStyle::SC_None; switch (cc) { case CC_ComboBox: if (const QStyleOptionComboBox *cmb = qstyleoption_cast(opt)) { sc = QWindowsStyle::hitTestComplexControl(cc, cmb, pt, widget); if (!cmb->editable && sc != QStyle::SC_None) sc = SC_ComboBoxArrow; // A bit of a lie, but what we want } break; case CC_Slider: if (const QStyleOptionSlider *slider = qstyleoption_cast(opt)) { HIThemeTrackDrawInfo tdi; d->getSliderInfo(cc, slider, &tdi, widget); ControlPartCode part; HIPoint pos = CGPointMake(pt.x(), pt.y()); if (HIThemeHitTestTrack(&tdi, &pos, &part)) { if (part == kControlPageUpPart || part == kControlPageDownPart) sc = SC_SliderGroove; else sc = SC_SliderHandle; } } break; case CC_ScrollBar: if (const QStyleOptionSlider *sb = qstyleoption_cast(opt)) { HIScrollBarTrackInfo sbi; sbi.version = qt_mac_hitheme_version; if (!(sb->state & State_Active)) sbi.enableState = kThemeTrackInactive; else if (sb->state & State_Enabled) sbi.enableState = kThemeTrackActive; else sbi.enableState = kThemeTrackDisabled; // The arrow buttons are not drawn if the scroll bar is to short, // exclude them from the hit test. const int scrollBarLength = (sb->orientation == Qt::Horizontal) ? sb->rect.width() : sb->rect.height(); if (scrollBarLength < scrollButtonsCutoffSize(scrollButtonsCutoff, widgetSizePolicy(widget))) sbi.enableState = kThemeTrackNothingToScroll; sbi.viewsize = sb->pageStep; HIPoint pos = CGPointMake(pt.x(), pt.y()); HIRect macSBRect = qt_hirectForQRect(sb->rect); ControlPartCode part; bool reverseHorizontal = (sb->direction == Qt::RightToLeft && sb->orientation == Qt::Horizontal && (!sb->upsideDown || (QSysInfo::MacintoshVersion >= QSysInfo::MV_10_4 && sb->upsideDown))); if (HIThemeHitTestScrollBarArrows(&macSBRect, &sbi, sb->orientation == Qt::Horizontal, &pos, 0, &part)) { if (part == kControlUpButtonPart) sc = reverseHorizontal ? SC_ScrollBarAddLine : SC_ScrollBarSubLine; else if (part == kControlDownButtonPart) sc = reverseHorizontal ? SC_ScrollBarSubLine : SC_ScrollBarAddLine; } else { HIThemeTrackDrawInfo tdi; d->getSliderInfo(cc, sb, &tdi, widget); if(tdi.enableState == kThemeTrackInactive) tdi.enableState = kThemeTrackActive; if (HIThemeHitTestTrack(&tdi, &pos, &part)) { if (part == kControlPageUpPart) sc = reverseHorizontal ? SC_ScrollBarAddPage : SC_ScrollBarSubPage; else if (part == kControlPageDownPart) sc = reverseHorizontal ? SC_ScrollBarSubPage : SC_ScrollBarAddPage; else sc = SC_ScrollBarSlider; } } } break; /* I don't know why, but we only get kWindowContentRgn here, which isn't what we want at all. It would be very nice if this would work. case QStyle::CC_TitleBar: if (const QStyleOptionTitleBar *tbar = qstyleoption_cast(opt)) { HIThemeWindowDrawInfo wdi; memset(&wdi, 0, sizeof(wdi)); wdi.version = qt_mac_hitheme_version; wdi.state = kThemeStateActive; wdi.windowType = QtWinType; wdi.titleWidth = tbar->rect.width(); wdi.titleHeight = tbar->rect.height(); if (tbar->titleBarState) wdi.attributes |= kThemeWindowHasFullZoom | kThemeWindowHasCloseBox | kThemeWindowHasCollapseBox; else if (tbar->titleBarFlags & Qt::WindowSystemMenuHint) wdi.attributes |= kThemeWindowHasCloseBox; QRect tmpRect = tbar->rect; tmpRect.setHeight(tmpRect.height() + 100); HIRect hirect = qt_hirectForQRect(tmpRect); WindowRegionCode hit; HIPoint hipt = CGPointMake(pt.x(), pt.y()); if (HIThemeGetWindowRegionHit(&hirect, &wdi, &hipt, &hit)) { switch (hit) { case kWindowCloseBoxRgn: sc = QStyle::SC_TitleBarCloseButton; break; case kWindowCollapseBoxRgn: sc = QStyle::SC_TitleBarMinButton; break; case kWindowZoomBoxRgn: sc = QStyle::SC_TitleBarMaxButton; break; case kWindowTitleTextRgn: sc = QStyle::SC_TitleBarLabel; break; default: qDebug("got something else %d", hit); break; } } } break; */ default: sc = QWindowsStyle::hitTestComplexControl(cc, opt, pt, widget); break; } return sc; } QRect QMacStyle::subControlRect(ComplexControl cc, const QStyleOptionComplex *opt, SubControl sc, const QWidget *widget) const { QRect ret; switch (cc) { case CC_Slider: case CC_ScrollBar: if (const QStyleOptionSlider *slider = qstyleoption_cast(opt)) { HIThemeTrackDrawInfo tdi; d->getSliderInfo(cc, slider, &tdi, widget); HIRect macRect; QCFType shape; bool scrollBar = cc == CC_ScrollBar; if ((scrollBar && sc == SC_ScrollBarSlider) || (!scrollBar && sc == SC_SliderHandle)) { HIThemeGetTrackThumbShape(&tdi, &shape); ptrHIShapeGetBounds(shape, &macRect); } else if (!scrollBar && sc == SC_SliderGroove) { HIThemeGetTrackBounds(&tdi, &macRect); } else if (sc == SC_ScrollBarGroove) { // Only scroll bar parts available... HIThemeGetTrackDragRect(&tdi, &macRect); } else { ControlPartCode cpc; if (sc == SC_ScrollBarSubPage || sc == SC_ScrollBarAddPage) { cpc = sc == SC_ScrollBarSubPage ? kControlPageDownPart : kControlPageUpPart; } else { cpc = sc == SC_ScrollBarSubLine ? kControlUpButtonPart : kControlDownButtonPart; if (slider->direction == Qt::RightToLeft && slider->orientation == Qt::Horizontal) { if (cpc == kControlDownButtonPart) cpc = kControlUpButtonPart; else if (cpc == kControlUpButtonPart) cpc = kControlDownButtonPart; } } HIThemeGetTrackPartBounds(&tdi, cpc, &macRect); } ret = qt_qrectForHIRect(macRect); // Tweak: the dark line between the sub/add line buttons belong to only one of the buttons // when doing hit-testing, but both of them have to repaint it. Extend the rect to cover // the line in the cases where HIThemeGetTrackPartBounds returns a rect that doesn't. if (slider->orientation == Qt::Horizontal) { if (slider->direction == Qt::LeftToRight && sc == SC_ScrollBarSubLine) ret.adjust(0, 0, 1, 0); else if (slider->direction == Qt::RightToLeft && sc == SC_ScrollBarAddLine) ret.adjust(-1, 0, 1, 0); } else if (sc == SC_ScrollBarAddLine) { ret.adjust(0, -1, 0, 1); } } break; case CC_TitleBar: if (const QStyleOptionTitleBar *titlebar = qstyleoption_cast(opt)) { HIThemeWindowDrawInfo wdi; memset(&wdi, 0, sizeof(wdi)); wdi.version = qt_mac_hitheme_version; wdi.state = kThemeStateActive; wdi.windowType = QtWinType; wdi.titleHeight = titlebar->rect.height(); wdi.titleWidth = titlebar->rect.width(); wdi.attributes = kThemeWindowHasTitleText; if (titlebar->subControls & SC_TitleBarCloseButton) wdi.attributes |= kThemeWindowHasCloseBox; if (titlebar->subControls & SC_TitleBarMaxButton | SC_TitleBarNormalButton) wdi.attributes |= kThemeWindowHasFullZoom; if (titlebar->subControls & SC_TitleBarMinButton) wdi.attributes |= kThemeWindowHasCollapseBox; WindowRegionCode wrc = kWindowGlobalPortRgn; if (sc == SC_TitleBarCloseButton) wrc = kWindowCloseBoxRgn; else if (sc == SC_TitleBarMinButton) wrc = kWindowCollapseBoxRgn; else if (sc == SC_TitleBarMaxButton) wrc = kWindowZoomBoxRgn; else if (sc == SC_TitleBarLabel) wrc = kWindowTitleTextRgn; else if (sc == SC_TitleBarSysMenu) ret.setRect(-1024, -1024, 10, proxy()->pixelMetric(PM_TitleBarHeight, titlebar, widget)); if (wrc != kWindowGlobalPortRgn) { QCFType region; QRect tmpRect = titlebar->rect; HIRect titleRect = qt_hirectForQRect(tmpRect); HIThemeGetWindowShape(&titleRect, &wdi, kWindowTitleBarRgn, ®ion); ptrHIShapeGetBounds(region, &titleRect); CFRelease(region); tmpRect.translate(tmpRect.x() - int(titleRect.origin.x), tmpRect.y() - int(titleRect.origin.y)); titleRect = qt_hirectForQRect(tmpRect); HIThemeGetWindowShape(&titleRect, &wdi, wrc, ®ion); ptrHIShapeGetBounds(region, &titleRect); ret = qt_qrectForHIRect(titleRect); } } break; case CC_ComboBox: if (const QStyleOptionComboBox *combo = qstyleoption_cast(opt)) { HIThemeButtonDrawInfo bdi; d->initComboboxBdi(combo, &bdi, widget, d->getDrawState(opt->state)); switch (sc) { case SC_ComboBoxEditField:{ ret = QMacStylePrivate::comboboxEditBounds(combo->rect, bdi); // hack to posistion the edit feld correctly for QDateTimeEdits // in calendarPopup mode. if (qobject_cast(widget)) { ret.moveTop(ret.top() - 2); ret.setHeight(ret.height() +1); } break; } case SC_ComboBoxArrow:{ ret = QMacStylePrivate::comboboxEditBounds(combo->rect, bdi); ret.setX(ret.x() + ret.width()); ret.setWidth(combo->rect.right() - ret.right()); break; } case SC_ComboBoxListBoxPopup:{ if (combo->editable) { HIRect inner = QMacStylePrivate::comboboxInnerBounds(qt_hirectForQRect(combo->rect), bdi.kind); QRect editRect = QMacStylePrivate::comboboxEditBounds(combo->rect, bdi); const int comboTop = combo->rect.top(); ret = QRect(qRound(inner.origin.x), comboTop, qRound(inner.origin.x - combo->rect.left() + inner.size.width), editRect.bottom() - comboTop + 2); } else { QRect editRect = QMacStylePrivate::comboboxEditBounds(combo->rect, bdi); ret = QRect(combo->rect.x() + 4 - 11, combo->rect.y() + 1, editRect.width() + 10 + 11, 1); } break; } default: break; } } break; case CC_GroupBox: if (const QStyleOptionGroupBox *groupBox = qstyleoption_cast(opt)) { bool checkable = groupBox->subControls & SC_GroupBoxCheckBox; bool flat = (groupBox->features & QStyleOptionFrameV2::Flat); bool hasNoText = !checkable && groupBox->text.isEmpty(); switch (sc) { case SC_GroupBoxLabel: case SC_GroupBoxCheckBox: { // Cheat and use the smaller font if we need to bool checkable = groupBox->subControls & SC_GroupBoxCheckBox; bool fontIsSet = (widget && widget->testAttribute(Qt::WA_SetFont) || !QApplication::desktopSettingsAware()); int tw; int h; int margin = flat || hasNoText ? 0 : 12; ret = groupBox->rect.adjusted(margin, 0, -margin, 0); if (!fontIsSet) { HIThemeTextInfo tti; tti.version = qt_mac_hitheme_version; tti.state = kThemeStateActive; tti.fontID = checkable ? kThemeSystemFont : kThemeSmallSystemFont; tti.horizontalFlushness = kHIThemeTextHorizontalFlushCenter; tti.verticalFlushness = kHIThemeTextVerticalFlushCenter; tti.options = kHIThemeTextBoxOptionNone; tti.truncationPosition = kHIThemeTextTruncationNone; tti.truncationMaxLines = 1 + groupBox->text.count(QLatin1Char('\n')); CGFloat width; CGFloat height; QCFString groupText = qt_mac_removeMnemonics(groupBox->text); HIThemeGetTextDimensions(groupText, 0, &tti, &width, &height, 0); tw = int(width); h = int(height); } else { QFontMetrics fm = groupBox->fontMetrics; if (!checkable && !fontIsSet) fm = QFontMetrics(qt_app_fonts_hash()->value("QSmallFont", QFont())); h = fm.height(); tw = fm.size(Qt::TextShowMnemonic, groupBox->text).width(); } ret.setHeight(h); QRect labelRect = alignedRect(groupBox->direction, groupBox->textAlignment, QSize(tw, h), ret); int indicatorWidth = proxy()->pixelMetric(PM_IndicatorWidth, opt, widget); bool rtl = groupBox->direction == Qt::RightToLeft; if (sc == SC_GroupBoxLabel) { if (checkable) { int newSum = indicatorWidth + 1; int newLeft = labelRect.left() + (rtl ? -newSum : newSum); labelRect.moveLeft(newLeft); } else if (flat) { int newLeft = labelRect.left() - (rtl ? 3 : -3); labelRect.moveLeft(newLeft); labelRect.moveTop(labelRect.top() + 3); } else { int newLeft = labelRect.left() - (rtl ? 3 : 2); labelRect.moveLeft(newLeft); labelRect.moveTop(labelRect.top() + 5); } ret = labelRect; } if (sc == SC_GroupBoxCheckBox) { int left = rtl ? labelRect.right() - indicatorWidth : labelRect.left(); ret.setRect(left, ret.top(), indicatorWidth, proxy()->pixelMetric(PM_IndicatorHeight, opt, widget)); } break; } case SC_GroupBoxContents: case SC_GroupBoxFrame: { if (flat) { ret = QWindowsStyle::subControlRect(cc, groupBox, sc, widget); break; } QFontMetrics fm = groupBox->fontMetrics; bool checkable = groupBox->subControls & SC_GroupBoxCheckBox; int yOffset = 3; if (!checkable) { if (widget && !widget->testAttribute(Qt::WA_SetFont) && QApplication::desktopSettingsAware()) fm = QFontMetrics(qt_app_fonts_hash()->value("QSmallFont", QFont())); yOffset = 5; if (hasNoText) yOffset = -fm.height(); } ret = opt->rect.adjusted(0, fm.height() + yOffset, 0, 0); if (sc == SC_GroupBoxContents) ret.adjust(3, 3, -3, -4); // guess } break; default: ret = QWindowsStyle::subControlRect(cc, groupBox, sc, widget); break; } } break; case CC_SpinBox: if (const QStyleOptionSpinBox *spin = qstyleoption_cast(opt)) { QAquaWidgetSize aquaSize = d->aquaSizeConstrain(spin, widget); int spinner_w; int spinBoxSep; int fw = proxy()->pixelMetric(PM_SpinBoxFrameWidth, spin, widget); switch (aquaSize) { default: case QAquaSizeUnknown: case QAquaSizeLarge: spinner_w = 14; spinBoxSep = 2; break; case QAquaSizeSmall: spinner_w = 12; spinBoxSep = 2; break; case QAquaSizeMini: spinner_w = 10; spinBoxSep = 1; break; } switch (sc) { case SC_SpinBoxUp: case SC_SpinBoxDown: { if (spin->buttonSymbols == QAbstractSpinBox::NoButtons) break; const int y = fw; const int x = spin->rect.width() - spinner_w; ret.setRect(x + spin->rect.x(), y + spin->rect.y(), spinner_w, spin->rect.height() - y * 2); HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.kind = kThemeIncDecButton; int hackTranslateX; switch (aquaSize) { default: case QAquaSizeUnknown: case QAquaSizeLarge: bdi.kind = kThemeIncDecButton; hackTranslateX = 0; break; case QAquaSizeSmall: bdi.kind = kThemeIncDecButtonSmall; hackTranslateX = -2; break; case QAquaSizeMini: bdi.kind = kThemeIncDecButtonMini; hackTranslateX = -1; break; } bdi.state = kThemeStateActive; bdi.value = kThemeButtonOff; bdi.adornment = kThemeAdornmentNone; HIRect hirect = qt_hirectForQRect(ret); HIRect outRect; HIThemeGetButtonBackgroundBounds(&hirect, &bdi, &outRect); ret = qt_qrectForHIRect(outRect); switch (sc) { case SC_SpinBoxUp: ret.setHeight(ret.height() / 2); break; case SC_SpinBoxDown: ret.setY(ret.y() + ret.height() / 2); break; default: Q_ASSERT(0); break; } ret.translate(hackTranslateX, 0); // hack: position the buttons correctly (weird that we need this) ret = visualRect(spin->direction, spin->rect, ret); break; } case SC_SpinBoxEditField: if (spin->buttonSymbols == QAbstractSpinBox::NoButtons) { ret.setRect(fw, fw, spin->rect.width() - fw * 2, spin->rect.height() - fw * 2); } else { ret.setRect(fw, fw, spin->rect.width() - fw * 2 - spinBoxSep - spinner_w, spin->rect.height() - fw * 2); } ret = visualRect(spin->direction, spin->rect, ret); break; default: ret = QWindowsStyle::subControlRect(cc, spin, sc, widget); break; } } break; case CC_ToolButton: ret = QWindowsStyle::subControlRect(cc, opt, sc, widget); if (sc == SC_ToolButtonMenu && widget && !qobject_cast(widget->parentWidget())) { ret.adjust(-1, 0, 0, 0); } break; default: ret = QWindowsStyle::subControlRect(cc, opt, sc, widget); break; } return ret; } QSize QMacStyle::sizeFromContents(ContentsType ct, const QStyleOption *opt, const QSize &csz, const QWidget *widget) const { QSize sz(csz); bool useAquaGuideline = true; switch (ct) { case QStyle::CT_SpinBox: // hack to work around horrible sizeHint() code in QAbstractSpinBox sz.setHeight(sz.height() - 3); break; case QStyle::CT_TabBarTab: if (const QStyleOptionTabV3 *tab = qstyleoption_cast(opt)) { const QAquaWidgetSize AquaSize = d->aquaSizeConstrain(opt, widget); const bool differentFont = (widget && widget->testAttribute(Qt::WA_SetFont)) || !QApplication::desktopSettingsAware(); ThemeTabDirection ttd = getTabDirection(tab->shape); bool vertTabs = ttd == kThemeTabWest || ttd == kThemeTabEast; if (vertTabs) sz.transpose(); int defaultTabHeight; int defaultExtraSpace = proxy()->pixelMetric(PM_TabBarTabHSpace, tab, widget); // Remove spurious gcc warning (AFAIK) QFontMetrics fm = opt->fontMetrics; switch (AquaSize) { case QAquaSizeUnknown: case QAquaSizeLarge: if (tab->documentMode) defaultTabHeight = 23; else defaultTabHeight = 21; break; case QAquaSizeSmall: defaultTabHeight = 18; break; case QAquaSizeMini: defaultTabHeight = 16; break; } bool setWidth = false; if (differentFont || !tab->icon.isNull()) { sz.rheight() = qMax(defaultTabHeight, sz.height()); } else { QSize textSize = fm.size(Qt::TextShowMnemonic, tab->text); sz.rheight() = qMax(defaultTabHeight, textSize.height()); sz.rwidth() = textSize.width() + defaultExtraSpace; setWidth = true; } if (vertTabs) sz.transpose(); int maxWidgetHeight = qMax(tab->leftButtonSize.height(), tab->rightButtonSize.height()); int maxWidgetWidth = qMax(tab->leftButtonSize.width(), tab->rightButtonSize.width()); int widgetWidth = 0; int widgetHeight = 0; int padding = 0; if (tab->leftButtonSize.isValid()) { padding += 8; widgetWidth += tab->leftButtonSize.width(); widgetHeight += tab->leftButtonSize.height(); } if (tab->rightButtonSize.isValid()) { padding += 8; widgetWidth += tab->rightButtonSize.width(); widgetHeight += tab->rightButtonSize.height(); } if (vertTabs) { sz.setHeight(sz.height() + widgetHeight + padding); sz.setWidth(qMax(sz.width(), maxWidgetWidth)); } else { if (setWidth) sz.setWidth(sz.width() + widgetWidth + padding); sz.setHeight(qMax(sz.height(), maxWidgetHeight)); } } break; case QStyle::CT_PushButton: // By default, we fit the contents inside a normal rounded push button. // Do this by add enough space around the contents so that rounded // borders (including highlighting when active) will show. sz.rwidth() += PushButtonLeftOffset + PushButtonRightOffset + 12; sz.rheight() += PushButtonTopOffset + PushButtonBottomOffset; break; case QStyle::CT_MenuItem: if (const QStyleOptionMenuItem *mi = qstyleoption_cast(opt)) { int maxpmw = mi->maxIconWidth; const QComboBox *comboBox = qobject_cast(widget); int w = sz.width(), h = sz.height(); if (mi->menuItemType == QStyleOptionMenuItem::Separator) { w = 10; SInt16 ash; GetThemeMenuSeparatorHeight(&ash); h = ash; } else { h = mi->fontMetrics.height() + 2; if (!mi->icon.isNull()) { if (comboBox) { const QSize &iconSize = comboBox->iconSize(); h = qMax(h, iconSize.height() + 4); maxpmw = qMax(maxpmw, iconSize.width()); } else { int iconExtent = proxy()->pixelMetric(PM_SmallIconSize); h = qMax(h, mi->icon.actualSize(QSize(iconExtent, iconExtent)).height() + 4); } } } if (mi->text.contains(QLatin1Char('\t'))) w += 12; if (mi->menuItemType == QStyleOptionMenuItem::SubMenu) w += 20; if (maxpmw) w += maxpmw + 6; // add space for a check. All items have place for a check too. w += 20; if (comboBox && comboBox->isVisible()) { QStyleOptionComboBox cmb; cmb.initFrom(comboBox); cmb.editable = false; cmb.subControls = QStyle::SC_ComboBoxEditField; cmb.activeSubControls = QStyle::SC_None; w = qMax(w, subControlRect(QStyle::CC_ComboBox, &cmb, QStyle::SC_ComboBoxEditField, comboBox).width()); } else { w += 12; } sz = QSize(w, h); } break; case CT_ToolButton: if (widget && qobject_cast(widget->parentWidget())) { if (QMainWindow * mainWindow = qobject_cast(widget->parent())) { if (mainWindow->unifiedTitleAndToolBarOnMac()) { sz.rwidth() += 4; if (sz.height() <= 32) { // Workaround strange HIToolBar bug when getting constraints. sz.rheight() += 1; } return sz; } } } sz.rwidth() += 10; sz.rheight() += 10; return sz; case CT_ComboBox: sz.rwidth() += 50; break; case CT_Menu: { QStyleHintReturnMask menuMask; QStyleOption myOption = *opt; myOption.rect.setSize(sz); if (proxy()->styleHint(SH_Menu_Mask, &myOption, widget, &menuMask)) { sz = menuMask.region.boundingRect().size(); } break; } case CT_HeaderSection:{ const QStyleOptionHeader *header = qstyleoption_cast(opt); sz = QWindowsStyle::sizeFromContents(ct, opt, csz, widget); if (header->text.contains(QLatin1Char('\n'))) useAquaGuideline = false; break; } case CT_ScrollBar : // Make sure that the scroll bar is large enough to display the thumb indicator. if (const QStyleOptionSlider *slider = qstyleoption_cast(opt)) { const int minimumSize = scrollButtonsCutoffSize(thumbIndicatorCutoff, widgetSizePolicy(widget)); if (slider->orientation == Qt::Horizontal) sz = sz.expandedTo(QSize(minimumSize, sz.height())); else sz = sz.expandedTo(QSize(sz.width(), minimumSize)); } break; default: sz = QWindowsStyle::sizeFromContents(ct, opt, csz, widget); } if (useAquaGuideline){ QSize macsz; if (d->aquaSizeConstrain(opt, widget, ct, sz, &macsz) != QAquaSizeUnknown) { if (macsz.width() != -1) sz.setWidth(macsz.width()); if (macsz.height() != -1) sz.setHeight(macsz.height()); } } // The sizes that Carbon and the guidelines gives us excludes the focus frame. // We compensate for this by adding some extra space here to make room for the frame when drawing: if (const QStyleOptionComboBox *combo = qstyleoption_cast(opt)){ QAquaWidgetSize widgetSize = d->aquaSizeConstrain(opt, widget); int bkind = 0; switch (widgetSize) { default: case QAquaSizeLarge: bkind = combo->editable ? kThemeComboBox : kThemePopupButton; break; case QAquaSizeSmall: bkind = combo->editable ? int(kThemeComboBoxSmall) : int(kThemePopupButtonSmall); break; case QAquaSizeMini: bkind = combo->editable ? kThemeComboBoxMini : kThemePopupButtonMini; break; } HIRect tmpRect = {{0, 0}, {0, 0}}; HIRect diffRect = QMacStylePrivate::comboboxInnerBounds(tmpRect, bkind); sz.rwidth() -= qRound(diffRect.size.width); sz.rheight() -= qRound(diffRect.size.height); } else if (ct == CT_PushButton || ct == CT_ToolButton){ ThemeButtonKind bkind; QAquaWidgetSize widgetSize = d->aquaSizeConstrain(opt, widget); switch (ct) { default: case CT_PushButton: if (const QStyleOptionButton *btn = qstyleoption_cast(opt)) { if (btn->features & QStyleOptionButton::CommandLinkButton) { return QWindowsStyle::sizeFromContents(ct, opt, sz, widget); } } switch (widgetSize) { default: case QAquaSizeLarge: bkind = kThemePushButton; break; case QAquaSizeSmall: bkind = kThemePushButtonSmall; break; case QAquaSizeMini: bkind = kThemePushButtonMini; break; } break; case CT_ToolButton: switch (widgetSize) { default: case QAquaSizeLarge: bkind = kThemeLargeBevelButton; break; case QAquaSizeMini: case QAquaSizeSmall: bkind = kThemeSmallBevelButton; } break; } HIThemeButtonDrawInfo bdi; bdi.version = qt_mac_hitheme_version; bdi.state = kThemeStateActive; bdi.kind = bkind; bdi.value = kThemeButtonOff; bdi.adornment = kThemeAdornmentNone; HIRect macRect, myRect; myRect = CGRectMake(0, 0, sz.width(), sz.height()); HIThemeGetButtonBackgroundBounds(&myRect, &bdi, &macRect); // Mini buttons only return their actual size in HIThemeGetButtonBackgroundBounds, so help them out a bit (guess), if (bkind == kThemePushButtonMini) macRect.size.height += 8.; else if (bkind == kThemePushButtonSmall) macRect.size.height -= 10; sz.setWidth(sz.width() + int(macRect.size.width - myRect.size.width)); sz.setHeight(sz.height() + int(macRect.size.height - myRect.size.height)); } return sz; } void QMacStyle::drawItemText(QPainter *p, const QRect &r, int flags, const QPalette &pal, bool enabled, const QString &text, QPalette::ColorRole textRole) const { if(flags & Qt::TextShowMnemonic) flags |= Qt::TextHideMnemonic; QWindowsStyle::drawItemText(p, r, flags, pal, enabled, text, textRole); } bool QMacStyle::event(QEvent *e) { if(e->type() == QEvent::FocusIn) { QWidget *f = 0; QWidget *focusWidget = QApplication::focusWidget(); #ifndef QT_NO_GRAPHICSVIEW if (QGraphicsView *graphicsView = qobject_cast(focusWidget)) { QGraphicsItem *focusItem = graphicsView->scene() ? graphicsView->scene()->focusItem() : 0; if (focusItem && focusItem->type() == QGraphicsProxyWidget::Type) { QGraphicsProxyWidget *proxy = static_cast(focusItem); if (proxy->widget()) focusWidget = proxy->widget()->focusWidget(); } } #endif if (focusWidget && focusWidget->testAttribute(Qt::WA_MacShowFocusRect)) { f = focusWidget; QWidget *top = f->parentWidget(); while (top && !top->isWindow() && !(top->windowType() == Qt::SubWindow)) top = top->parentWidget(); #ifndef QT_NO_MAINWINDOW if (qobject_cast(top)) { QWidget *central = static_cast(top)->centralWidget(); for (const QWidget *par = f; par; par = par->parentWidget()) { if (par == central) { top = central; break; } if (par->isWindow()) break; } } #endif } if (f) { if(!d->focusWidget) d->focusWidget = new QFocusFrame(f); d->focusWidget->setWidget(f); } else if(d->focusWidget) { d->focusWidget->setWidget(0); } } else if(e->type() == QEvent::FocusOut) { if(d->focusWidget) d->focusWidget->setWidget(0); } return false; } QIcon QMacStyle::standardIconImplementation(StandardPixmap standardIcon, const QStyleOption *opt, const QWidget *widget) const { switch (standardIcon) { default: return QWindowsStyle::standardIconImplementation(standardIcon, opt, widget); case SP_ToolBarHorizontalExtensionButton: case SP_ToolBarVerticalExtensionButton: { QPixmap pixmap(qt_mac_toolbar_ext); if (standardIcon == SP_ToolBarVerticalExtensionButton) { QPixmap pix2(pixmap.height(), pixmap.width()); pix2.fill(Qt::transparent); QPainter p(&pix2); p.translate(pix2.width(), 0); p.rotate(90); p.drawPixmap(0, 0, pixmap); return pix2; } return pixmap; } } } int QMacStyle::layoutSpacingImplementation(QSizePolicy::ControlType control1, QSizePolicy::ControlType control2, Qt::Orientation orientation, const QStyleOption *option, const QWidget *widget) const { const int ButtonMask = QSizePolicy::ButtonBox | QSizePolicy::PushButton; bool isMetal = (widget && widget->testAttribute(Qt::WA_MacBrushedMetal)); int controlSize = getControlSize(option, widget); if (control2 == QSizePolicy::ButtonBox) { /* AHIG seems to prefer a 12-pixel margin between group boxes and the row of buttons. The 20 pixel comes from Builder. */ if (isMetal // (AHIG, guess, guess) || (control1 & (QSizePolicy::Frame // guess | QSizePolicy::GroupBox // (AHIG, guess, guess) | QSizePolicy::TabWidget // guess | ButtonMask))) { // AHIG return_SIZE(14, 8, 8); } else if (control1 == QSizePolicy::LineEdit) { return_SIZE(8, 8, 8); // Interface Builder } else { return_SIZE(20, 7, 7); // Interface Builder } } if ((control1 | control2) & ButtonMask) { if (control1 == QSizePolicy::LineEdit) return_SIZE(8, 8, 8); // Interface Builder else if (control2 == QSizePolicy::LineEdit) { if (orientation == Qt::Vertical) return_SIZE(20, 7, 7); // Interface Builder else return_SIZE(20, 8, 8); } return_SIZE(14, 8, 8); // Interface Builder } switch (CT2(control1, control2)) { case CT1(QSizePolicy::Label): // guess case CT2(QSizePolicy::Label, QSizePolicy::DefaultType): // guess case CT2(QSizePolicy::Label, QSizePolicy::CheckBox): // AHIG case CT2(QSizePolicy::Label, QSizePolicy::ComboBox): // AHIG case CT2(QSizePolicy::Label, QSizePolicy::LineEdit): // guess case CT2(QSizePolicy::Label, QSizePolicy::RadioButton): // AHIG case CT2(QSizePolicy::Label, QSizePolicy::Slider): // guess case CT2(QSizePolicy::Label, QSizePolicy::SpinBox): // guess case CT2(QSizePolicy::Label, QSizePolicy::ToolButton): // guess return_SIZE(8, 6, 5); case CT1(QSizePolicy::ToolButton): return 8; // AHIG case CT1(QSizePolicy::CheckBox): case CT2(QSizePolicy::CheckBox, QSizePolicy::RadioButton): case CT2(QSizePolicy::RadioButton, QSizePolicy::CheckBox): if (orientation == Qt::Vertical) return_SIZE(8, 8, 7); // AHIG and Builder break; case CT1(QSizePolicy::RadioButton): if (orientation == Qt::Vertical) return 5; // (Builder, guess, AHIG) } if (orientation == Qt::Horizontal && (control2 & (QSizePolicy::CheckBox | QSizePolicy::RadioButton))) return_SIZE(12, 10, 8); // guess if ((control1 | control2) & (QSizePolicy::Frame | QSizePolicy::GroupBox | QSizePolicy::TabWidget)) { /* These values were chosen so that nested container widgets look good side by side. Builder uses 8, which looks way too small, and AHIG doesn't say anything. */ return_SIZE(16, 10, 10); // guess } if ((control1 | control2) & (QSizePolicy::Line | QSizePolicy::Slider)) return_SIZE(12, 10, 8); // AHIG if ((control1 | control2) & QSizePolicy::LineEdit) return_SIZE(10, 8, 8); // AHIG /* AHIG and Builder differ by up to 4 pixels for stacked editable comboboxes. We use some values that work fairly well in all cases. */ if ((control1 | control2) & QSizePolicy::ComboBox) return_SIZE(10, 8, 7); // guess /* Builder defaults to 8, 6, 5 in lots of cases, but most of the time the result looks too cramped. */ return_SIZE(10, 8, 6); // guess } QT_END_NAMESPACE