/* * Copyright 2019 Michail Vourlakos * * This file is part of Latte-Dock * * Latte-Dock is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * Latte-Dock is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * */ #include "generictools.h" // Qt #include #include #include #include namespace Latte { const int ICONMARGIN = 1; const int INDICATORCHANGESLENGTH = 6; const int INDICATORCHANGESMARGIN = 5; const int MARGIN = 2; bool isEnabled(const QStyleOption &option) { if (option.state & QStyle::State_Enabled) { return true; } return false; } bool isActive(const QStyleOption &option) { if (option.state & QStyle::State_Active) { return true; } return false; } bool isSelected(const QStyleOption &option) { if (option.state & QStyle::State_Selected) { return true; } return false; } bool isHovered(const QStyleOption &option) { if (option.state & QStyle::State_MouseOver) { return true; } return false; } bool isFocused(const QStyleOption &option) { if (option.state & QStyle::State_HasFocus) { return true; } return false; } bool isTextCentered(const QStyleOptionViewItem &option) { if (option.displayAlignment & Qt::AlignHCenter) { return true; } return false; } QPalette::ColorGroup colorGroup(const QStyleOption &option) { if (!isEnabled(option)) { return QPalette::Disabled; } if (isActive(option) || isFocused(option)) { return QPalette::Active; } if (!isActive(option) && isSelected(option)) { return QPalette::Inactive; } return QPalette::Normal; } QStringList subtracted(const QStringList &original, const QStringList ¤t) { QStringList subtract; for(int i=0; isave(); QPalette::ColorRole applyColor = Latte::isSelected(option) ? QPalette::HighlightedText : QPalette::Text; QBrush nBrush = option.palette.brush(Latte::colorGroup(option), applyColor); QColor brushColor = nBrush.color(); brushColor.setAlphaF(textOpacity); QString css = QString("body { color : %1;}").arg(brushColor.name(QColor::HexArgb)); QTextDocument doc; doc.setDefaultStyleSheet(css); doc.setHtml("" + text + ""); //we need an offset to be in the same vertical center of TextEdit int offsetY = ((option.rect.height() - doc.size().height()) / 2); int textWidth = doc.size().width(); int textY = option.rect.top() + offsetY + 1; if (isTextCentered) { int textX = qMax(0, (option.rect.width() / 2) - (textWidth/2)); painter->translate(option.rect.left() + textX, textY); } else if (qApp->layoutDirection() == Qt::RightToLeft) { painter->translate(qMax(option.rect.left(), option.rect.right() - textWidth), textY); } else { painter->translate(option.rect.left(), textY); } QRect clip(0, 0, option.rect.width(), option.rect.height()); doc.drawContents(painter, clip); painter->restore(); } void drawBackground(QPainter *painter, const QStyleOptionViewItem &option) { QStyleOptionViewItem backOption = option; backOption.text = ""; //! Remove the focus dotted lines backOption.state = (option.state & ~QStyle::State_HasFocus); option.widget->style()->drawControl(QStyle::CE_ItemViewItem, &backOption, painter); } void drawBackground(QPainter *painter, const QStyle *style, const QStyleOptionMenuItem &option) { QStyleOptionMenuItem backOption = option; backOption.text = ""; //! Remove the focus dotted lines // iconOption.state = (option.state & ~QStyle::State_HasFocus); style->drawControl(QStyle::CE_MenuItem, &backOption, painter); } QRect remainedFromLayoutIcon(const QStyleOption &option, Qt::AlignmentFlag alignment, int lengthMargin, int thickMargin) { if (alignment == Qt::AlignHCenter) { return option.rect; } return remainedFromIcon(option, alignment, lengthMargin, thickMargin); } void drawLayoutIcon(QPainter *painter, const QStyleOption &option, const bool &isBackgroundFile, const QString &iconName, Qt::AlignmentFlag alignment, int lengthMargin, int thickMargin) { bool active = Latte::isActive(option); bool selected = Latte::isSelected(option); bool focused = Latte::isFocused(option); int lenmargin = (lengthMargin == -1 ? ICONMARGIN + MARGIN : lengthMargin); int thickmargin = (thickMargin == -1 ? ICONMARGIN : thickMargin); int iconsize = option.rect.height() - 2*thickMargin; int total = iconsize + 2*lenmargin; Qt::AlignmentFlag curalign = alignment; if (qApp->layoutDirection() == Qt::LeftToRight || alignment == Qt::AlignHCenter) { curalign = alignment; } else { curalign = alignment == Qt::AlignLeft ? Qt::AlignRight : Qt::AlignLeft; } QRect target; if (curalign == Qt::AlignLeft) { target = QRect(option.rect.x() + lenmargin, option.rect.y() + thickmargin, iconsize, iconsize); } else if (curalign == Qt::AlignRight) { target = QRect(option.rect.x() + option.rect.width() - total + lenmargin, option.rect.y() + thickmargin, iconsize, iconsize); } else { //! centered target = QRect(option.rect.x() + ((option.rect.width() - total)/2) + lenmargin, option.rect.y() + thickmargin, iconsize, iconsize); } painter->save(); painter->setRenderHint(QPainter::Antialiasing, true); if (isBackgroundFile) { int backImageMargin = 1; //most icon themes provide 1-2px. padding around icons //OLD CALCS: ICONMARGIN; //qMin(target.height()/4, ICONMARGIN+1); QRect backTarget(target.x() + backImageMargin, target.y() + backImageMargin, target.width() - 2*backImageMargin, target.height() - 2*backImageMargin); QPixmap backImage(iconName); backImage = backImage.copy(backTarget); QPalette::ColorRole textColorRole = selected ? QPalette::HighlightedText : QPalette::Text; QBrush imageBrush(backImage); QPen pen; pen.setWidth(1); pen.setColor(option.palette.color(Latte::colorGroup(option), textColorRole)); painter->setBrush(imageBrush); painter->setPen(pen); painter->drawEllipse(backTarget); } else { QIcon::Mode mode = ((active && (selected || focused)) ? QIcon::Selected : QIcon::Normal); painter->drawPixmap(target, QIcon::fromTheme(iconName).pixmap(target.height(), target.height(), mode)); } painter->restore(); } QRect remainedFromIcon(const QStyleOption &option, Qt::AlignmentFlag alignment, int lengthMargin, int thickMargin) { int lenmargin = (lengthMargin == -1 ? ICONMARGIN + MARGIN : lengthMargin); int thickmargin = (thickMargin == -1 ? ICONMARGIN : thickMargin); int iconsize = option.rect.height() - 2*thickMargin; int total = iconsize + 2*lenmargin; Qt::AlignmentFlag curalign = alignment; if (qApp->layoutDirection() == Qt::LeftToRight) { curalign = alignment; } else { curalign = alignment == Qt::AlignLeft ? Qt::AlignRight : Qt::AlignLeft; } QRect optionRemainedRect = (curalign == Qt::AlignLeft) ? QRect(option.rect.x() + total, option.rect.y(), option.rect.width() - total, option.rect.height()) : QRect(option.rect.x(), option.rect.y(), option.rect.width() - total, option.rect.height()); return optionRemainedRect; } void drawIcon(QPainter *painter, const QStyleOption &option, const QString &icon, Qt::AlignmentFlag alignment, int lengthMargin, int thickMargin) { int lenmargin = (lengthMargin == -1 ? ICONMARGIN + MARGIN : lengthMargin); int thickmargin = (thickMargin == -1 ? ICONMARGIN : thickMargin); int iconsize = option.rect.height() - 2*thickMargin; int total = iconsize + 2*lenmargin; bool active = Latte::isActive(option); bool selected = Latte::isSelected(option); bool focused = Latte::isFocused(option); QIcon::Mode mode = ((active && (selected || focused)) ? QIcon::Selected : QIcon::Normal); Qt::AlignmentFlag curalign = alignment; if (qApp->layoutDirection() == Qt::LeftToRight) { curalign = alignment; } else { curalign = alignment == Qt::AlignLeft ? Qt::AlignRight : Qt::AlignLeft; } QRect target; if (curalign == Qt::AlignLeft) { target = QRect(option.rect.x() + lenmargin, option.rect.y() + thickmargin, iconsize, iconsize); } else { target = QRect(option.rect.x() + option.rect.width() - total + lenmargin, option.rect.y() + thickmargin, iconsize, iconsize); } painter->drawPixmap(target, QIcon::fromTheme(icon).pixmap(target.height(), target.height(), mode)); } QRect remainedFromChangesIndicator(const QStyleOptionViewItem &option) { int tsize{INDICATORCHANGESLENGTH + INDICATORCHANGESMARGIN*2}; QRect optionRemainedRect = (qApp->layoutDirection() == Qt::RightToLeft) ? QRect(option.rect.x() + tsize, option.rect.y(), option.rect.width() - tsize, option.rect.height()) : QRect(option.rect.x(), option.rect.y(), option.rect.width() - tsize, option.rect.height()); return optionRemainedRect; } void drawChangesIndicator(QPainter *painter, const QStyleOptionViewItem &option) { //! draw changes circle indicator int csize{INDICATORCHANGESLENGTH}; int tsize{INDICATORCHANGESLENGTH + INDICATORCHANGESMARGIN*2}; painter->save(); QRect changesRect = (qApp->layoutDirection() == Qt::RightToLeft) ? QRect(option.rect.x() + INDICATORCHANGESMARGIN, option.rect.y() + option.rect.height()/2 - csize/2, csize, csize) : QRect(option.rect.x() + option.rect.width() - csize - INDICATORCHANGESMARGIN, option.rect.y() + option.rect.height()/2 - csize/2, csize, csize); QColor plasmaOrange(246, 116, 0); //orangish color used from plasma systemsettings #f67400 QBrush backBrush(plasmaOrange); QPen pen; pen.setWidth(1); pen.setColor(plasmaOrange); painter->setBrush(backBrush); painter->setPen(pen); painter->drawEllipse(changesRect); painter->restore(); } int screenMaxLength(const QStyleOption &option, const int &maxIconSize) { int icon_length = maxIconSize >= 0 ? qMin(option.rect.height(), maxIconSize) : option.rect.height(); int scr_maxlength = icon_length * 1.7; //! provide odd screen_maxlength if (scr_maxlength % 2 == 0) { scr_maxlength--; } return scr_maxlength; } QRect remainedFromScreenDrawing(const QStyleOption &option, const int &maxIconSize) { int total_length = screenMaxLength(option, maxIconSize) + MARGIN * 2 + 1; QRect optionRemainedRect = (qApp->layoutDirection() == Qt::RightToLeft) ? QRect(option.rect.x(), option.rect.y(), option.rect.width() - total_length, option.rect.height()) : QRect(option.rect.x() + total_length, option.rect.y(), option.rect.width() - total_length, option.rect.height()); return optionRemainedRect; } QRect drawScreen(QPainter *painter, const QStyleOption &option, QRect screenGeometry, const int &maxIconSize, const float brushOpacity) { float scr_ratio = (float)screenGeometry.width() / (float)screenGeometry.height(); bool isVertical = (scr_ratio < 1.0); int scr_maxlength = screenMaxLength(option, maxIconSize); int scr_maxthickness = maxIconSize >= 0 ? qMin(maxIconSize, option.rect.height() - MARGIN * 2) : option.rect.height() - MARGIN * 2; int total_length = scr_maxlength + MARGIN * 2; int pen_width = 2; painter->save(); painter->setRenderHint(QPainter::Antialiasing, true); float scr_maxratio = ((float)scr_maxlength) / (float)(scr_maxthickness); scr_ratio = qMin(qMax((float)0.75, scr_ratio), (float)scr_maxratio); int scr_height = (!isVertical ? scr_maxthickness - MARGIN * 4 : scr_maxthickness - MARGIN * 2); int scr_width = scr_ratio * scr_height; //! provide even screen width and height if (scr_width % 2 == 1) { scr_width++; } //! provide even screen width and height if (scr_height % 2 == 0) { scr_height++; } int topmargin = (option.rect.height() - scr_maxthickness) / 2; QRect screenMaximumRect = (qApp->layoutDirection() == Qt::RightToLeft) ? QRect(option.rect.x() + option.rect.width() - scr_maxlength - MARGIN, option.rect.y() + topmargin, scr_maxlength, scr_maxthickness - 1) : QRect(option.rect.x() + MARGIN , option.rect.y() + topmargin, scr_maxlength, scr_maxthickness - 1); int topScreenMargin = (screenMaximumRect.height() - scr_height) / 2; int leftScreenMargin = (screenMaximumRect.width() - scr_width) / 2; QRect screenRect(screenMaximumRect.x() + leftScreenMargin + MARGIN/2, screenMaximumRect.y() + topScreenMargin, scr_width, scr_height); QRect screenAvailableRect(screenRect.x() + pen_width - 1, screenRect.y() + pen_width - 1, screenRect.width() - pen_width - 1, screenRect.height() - pen_width - 1); bool selected = Latte::isSelected(option); QPalette::ColorRole textColorRole = selected ? QPalette::HighlightedText : QPalette::Text; QPen pen; pen.setWidth(pen_width); QColor pencolor = option.palette.color(Latte::colorGroup(option), textColorRole); pencolor.setAlphaF(brushOpacity); pen.setColor(pencolor); painter->setPen(pen); painter->drawRect(screenRect); pen.setWidth(1); painter->setPen(pen); int basex = screenRect.x() + (screenRect.width()/2) - 4; int basey = screenRect.y() + screenRect.height() + 2; painter->setRenderHint(QPainter::Antialiasing, false); painter->drawLine(basex , basey, basex + 8, basey); // debug screen maximum available rect //painter->drawRect(screenMaximumRect); painter->restore(); return screenAvailableRect; } }