/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "miniprojecttargetselector.h" #include "kit.h" #include "kitconfigwidget.h" #include "kitmanager.h" #include "target.h" #include "projectexplorericons.h" #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 static QIcon createCenteredIcon(const QIcon &icon, const QIcon &overlay) { QPixmap targetPixmap; const qreal appDevicePixelRatio = qApp->devicePixelRatio(); int deviceSpaceIconSize = Core::Constants::TARGET_ICON_SIZE * appDevicePixelRatio; targetPixmap = QPixmap(deviceSpaceIconSize, deviceSpaceIconSize); targetPixmap.setDevicePixelRatio(appDevicePixelRatio); targetPixmap.fill(Qt::transparent); QPainter painter(&targetPixmap); // painter in user space QPixmap pixmap = icon.pixmap(Core::Constants::TARGET_ICON_SIZE); // already takes app devicePixelRatio into account qreal pixmapDevicePixelRatio = pixmap.devicePixelRatio(); painter.drawPixmap((Core::Constants::TARGET_ICON_SIZE - pixmap.width() / pixmapDevicePixelRatio) / 2, (Core::Constants::TARGET_ICON_SIZE - pixmap.height() / pixmapDevicePixelRatio) / 2, pixmap); if (!overlay.isNull()) { pixmap = overlay.pixmap(Core::Constants::TARGET_ICON_SIZE); // already takes app devicePixelRatio into account pixmapDevicePixelRatio = pixmap.devicePixelRatio(); painter.drawPixmap((Core::Constants::TARGET_ICON_SIZE - pixmap.width() / pixmapDevicePixelRatio) / 2, (Core::Constants::TARGET_ICON_SIZE - pixmap.height() / pixmapDevicePixelRatio) / 2, pixmap); } return QIcon(targetPixmap); } using namespace ProjectExplorer; using namespace ProjectExplorer::Internal; using namespace Utils; static bool projectLesserThan(Project *p1, Project *p2) { int result = caseFriendlyCompare(p1->displayName(), p2->displayName()); if (result != 0) return result < 0; else return p1 < p2; } //////// // TargetSelectorDelegate //////// class TargetSelectorDelegate : public QItemDelegate { public: TargetSelectorDelegate(ListWidget *parent) : QItemDelegate(parent), m_listWidget(parent) { } private: QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const; void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const; mutable QImage selectionGradient; ListWidget *m_listWidget; }; QSize TargetSelectorDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const { Q_UNUSED(option) Q_UNUSED(index) return QSize(m_listWidget->size().width(), 30); } void TargetSelectorDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { painter->save(); painter->setClipping(false); if (selectionGradient.isNull()) selectionGradient.load(QLatin1String(":/projectexplorer/images/targetpanel_gradient.png")); if (option.state & QStyle::State_Selected) { const QColor color = (option.state & QStyle::State_HasFocus) ? option.palette.highlight().color() : option.palette.dark().color(); if (creatorTheme()->widgetStyle() == Theme::StyleFlat) { painter->fillRect(option.rect, color); } else { painter->fillRect(option.rect, color.darker(140)); StyleHelper::drawCornerImage(selectionGradient, painter, option.rect.adjusted(0, 0, 0, -1), 5, 5, 5, 5); const QRectF borderRect = QRectF(option.rect).adjusted(0.5, 0.5, -0.5, -0.5); painter->setPen(QColor(255, 255, 255, 60)); painter->drawLine(borderRect.topLeft(), borderRect.topRight()); painter->setPen(QColor(255, 255, 255, 30)); painter->drawLine(borderRect.bottomLeft() - QPointF(0, 1), borderRect.bottomRight() - QPointF(0, 1)); painter->setPen(QColor(0, 0, 0, 80)); painter->drawLine(borderRect.bottomLeft(), borderRect.bottomRight()); } } QFontMetrics fm(option.font); QString text = index.data(Qt::DisplayRole).toString(); painter->setPen(QColor(255, 255, 255, 160)); QString elidedText = fm.elidedText(text, Qt::ElideMiddle, option.rect.width() - 12); if (elidedText != text) const_cast(index.model())->setData(index, text, Qt::ToolTipRole); else const_cast(index.model())->setData(index, QString(), Qt::ToolTipRole); painter->drawText(option.rect.left() + 6, option.rect.top() + (option.rect.height() - fm.height()) / 2 + fm.ascent(), elidedText); painter->restore(); } //////// // ListWidget //////// ListWidget::ListWidget(QWidget *parent) : QListWidget(parent), m_maxCount(0), m_optimalWidth(0) { setFocusPolicy(Qt::NoFocus); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setAlternatingRowColors(false); setFocusPolicy(Qt::WheelFocus); setItemDelegate(new TargetSelectorDelegate(this)); setAttribute(Qt::WA_MacShowFocusRect, false); setStyleSheet(QString::fromLatin1("QListWidget { background: #464646; border-style: none; }")); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); } void ListWidget::keyPressEvent(QKeyEvent *event) { if (event->key() == Qt::Key_Left) focusPreviousChild(); else if (event->key() == Qt::Key_Right) focusNextChild(); else QListWidget::keyPressEvent(event); } void ListWidget::keyReleaseEvent(QKeyEvent *event) { if (event->key() != Qt::Key_Left && event->key() != Qt::Key_Right) QListWidget::keyReleaseEvent(event); } void ListWidget::setMaxCount(int maxCount) { m_maxCount = maxCount; updateGeometry(); } int ListWidget::maxCount() { return m_maxCount; } int ListWidget::optimalWidth() const { return m_optimalWidth; } void ListWidget::setOptimalWidth(int width) { m_optimalWidth = width; updateGeometry(); } int ListWidget::padding() { // there needs to be enough extra pixels to show a scrollbar return 2 * style()->pixelMetric(QStyle::PM_FocusFrameHMargin, 0, this) + style()->pixelMetric(QStyle::PM_ScrollBarExtent, 0, this) + 10; } //////// // ProjectListWidget //////// ProjectListWidget::ProjectListWidget(QWidget *parent) : ListWidget(parent), m_ignoreIndexChange(false) { SessionManager *sessionManager = SessionManager::instance(); connect(sessionManager, &SessionManager::projectAdded, this, &ProjectListWidget::addProject); connect(sessionManager, &SessionManager::aboutToRemoveProject, this, &ProjectListWidget::removeProject); connect(sessionManager, &SessionManager::startupProjectChanged, this, &ProjectListWidget::changeStartupProject); connect(sessionManager, &SessionManager::projectDisplayNameChanged, this, &ProjectListWidget::projectDisplayNameChanged); connect(this, &QListWidget::currentRowChanged, this, &ProjectListWidget::setProject); } QListWidgetItem *ProjectListWidget::itemForProject(Project *project) { for (int i = 0; i < count(); ++i) { QListWidgetItem *currentItem = item(i); if (currentItem->data(Qt::UserRole).value() == project) return currentItem; } return 0; } QString ProjectListWidget::fullName(Project *project) { return tr("%1 (%2)").arg(project->displayName(), project->projectFilePath().toUserOutput()); } void ProjectListWidget::addProject(Project *project) { m_ignoreIndexChange = true; int pos = count(); for (int i=0; i < count(); ++i) { Project *p = item(i)->data(Qt::UserRole).value(); if (projectLesserThan(project, p)) { pos = i; break; } } bool useFullName = false; for (int i = 0; i < count(); ++i) { Project *p = item(i)->data(Qt::UserRole).value(); if (p->displayName() == project->displayName()) { useFullName = true; item(i)->setText(fullName(p)); } } QString displayName = useFullName ? fullName(project) : project->displayName(); QListWidgetItem *item = new QListWidgetItem(); item->setData(Qt::UserRole, QVariant::fromValue(project)); item->setText(displayName); insertItem(pos, item); if (project == SessionManager::startupProject()) setCurrentItem(item); QFontMetrics fn(font()); int width = fn.width(displayName) + padding(); if (width > optimalWidth()) setOptimalWidth(width); m_ignoreIndexChange = false; } void ProjectListWidget::removeProject(Project *project) { m_ignoreIndexChange = true; QListWidgetItem *listItem = itemForProject(project); delete listItem; // Update display names QString name = project->displayName(); int countDisplayName = 0; int otherIndex = -1; for (int i = 0; i < count(); ++i) { Project *p = item(i)->data(Qt::UserRole).value(); if (p->displayName() == name) { ++countDisplayName; otherIndex = i; } } if (countDisplayName == 1) { Project *p = item(otherIndex)->data(Qt::UserRole).value(); item(otherIndex)->setText(p->displayName()); } QFontMetrics fn(font()); // recheck optimal width int width = 0; for (int i = 0; i < count(); ++i) width = qMax(fn.width(item(i)->text()) + padding(), width); setOptimalWidth(width); m_ignoreIndexChange = false; } void ProjectListWidget::projectDisplayNameChanged(Project *project) { m_ignoreIndexChange = true; int oldPos = 0; bool useFullName = false; for (int i = 0; i < count(); ++i) { Project *p = item(i)->data(Qt::UserRole).value(); if (p == project) { oldPos = i; } else if (p->displayName() == project->displayName()) { useFullName = true; item(i)->setText(fullName(p)); } } bool isCurrentItem = (oldPos == currentRow()); QListWidgetItem *projectItem = takeItem(oldPos); int pos = count(); for (int i = 0; i < count(); ++i) { Project *p = item(i)->data(Qt::UserRole).value(); if (projectLesserThan(project, p)) { pos = i; break; } } QString displayName = useFullName ? fullName(project) : project->displayName(); projectItem->setText(displayName); insertItem(pos, projectItem); if (isCurrentItem) setCurrentRow(pos); // recheck optimal width QFontMetrics fn(font()); int width = 0; for (int i = 0; i < count(); ++i) width = qMax(fn.width(item(i)->text()) + padding(), width); setOptimalWidth(width); m_ignoreIndexChange = false; } void ProjectListWidget::setProject(int index) { if (m_ignoreIndexChange) return; if (index < 0) return; Project *p = item(index)->data(Qt::UserRole).value(); SessionManager::setStartupProject(p); } void ProjectListWidget::changeStartupProject(Project *project) { setCurrentItem(itemForProject(project)); } ///////// // GenericListWidget ///////// GenericListWidget::GenericListWidget(QWidget *parent) : ListWidget(parent), m_ignoreIndexChange(false) { connect(this, &QListWidget::currentRowChanged, this, &GenericListWidget::rowChanged); } void GenericListWidget::setProjectConfigurations(const QList &list, ProjectConfiguration *active) { m_ignoreIndexChange = true; clear(); for (int i = 0; i < count(); ++i) { ProjectConfiguration *p = item(i)->data(Qt::UserRole).value(); disconnect(p, &ProjectConfiguration::displayNameChanged, this, &GenericListWidget::displayNameChanged); } QFontMetrics fn(font()); int width = 0; foreach (ProjectConfiguration *pc, list) { addProjectConfiguration(pc); width = qMax(width, fn.width(pc->displayName()) + padding()); } setOptimalWidth(width); setActiveProjectConfiguration(active); m_ignoreIndexChange = false; } void GenericListWidget::setActiveProjectConfiguration(ProjectConfiguration *active) { QListWidgetItem *item = itemForProjectConfiguration(active); setCurrentItem(item); } void GenericListWidget::addProjectConfiguration(ProjectConfiguration *pc) { m_ignoreIndexChange = true; QListWidgetItem *lwi = new QListWidgetItem(); lwi->setText(pc->displayName()); lwi->setData(Qt::UserRole, QVariant::fromValue(pc)); // Figure out pos int pos = count(); for (int i = 0; i < count(); ++i) { ProjectConfiguration *p = item(i)->data(Qt::UserRole).value(); if (caseFriendlyCompare(pc->displayName(), p->displayName()) < 0) { pos = i; break; } } insertItem(pos, lwi); connect(pc, &ProjectConfiguration::displayNameChanged, this, &GenericListWidget::displayNameChanged); QFontMetrics fn(font()); int width = fn.width(pc->displayName()) + padding(); if (width > optimalWidth()) setOptimalWidth(width); m_ignoreIndexChange = false; } void GenericListWidget::removeProjectConfiguration(ProjectConfiguration *pc) { m_ignoreIndexChange = true; disconnect(pc, &ProjectConfiguration::displayNameChanged, this, &GenericListWidget::displayNameChanged); delete itemForProjectConfiguration(pc); QFontMetrics fn(font()); int width = 0; for (int i = 0; i < count(); ++i) { ProjectConfiguration *p = item(i)->data(Qt::UserRole).value(); width = qMax(width, fn.width(p->displayName()) + padding()); } setOptimalWidth(width); m_ignoreIndexChange = false; } void GenericListWidget::rowChanged(int index) { if (m_ignoreIndexChange) return; if (index < 0) return; emit changeActiveProjectConfiguration(item(index)->data(Qt::UserRole).value()); } void GenericListWidget::displayNameChanged() { m_ignoreIndexChange = true; ProjectConfiguration *activeProjectConfiguration = 0; if (currentItem()) activeProjectConfiguration = currentItem()->data(Qt::UserRole).value(); ProjectConfiguration *pc = qobject_cast(sender()); int index = -1; int i = 0; for (; i < count(); ++i) { QListWidgetItem *lwi = item(i); if (lwi->data(Qt::UserRole).value() == pc) { index = i; break; } } if (index == -1) return; QListWidgetItem *lwi = takeItem(i); lwi->setText(pc->displayName()); int pos = count(); for (int i = 0; i < count(); ++i) { ProjectConfiguration *p = item(i)->data(Qt::UserRole).value(); if (caseFriendlyCompare(pc->displayName(), p->displayName()) < 0) { pos = i; break; } } insertItem(pos, lwi); if (activeProjectConfiguration) setCurrentItem(itemForProjectConfiguration(activeProjectConfiguration)); QFontMetrics fn(font()); int width = 0; for (int i = 0; i < count(); ++i) { ProjectConfiguration *p = item(i)->data(Qt::UserRole).value(); width = qMax(width, fn.width(p->displayName()) + padding()); } setOptimalWidth(width); m_ignoreIndexChange = false; } QListWidgetItem *GenericListWidget::itemForProjectConfiguration(ProjectConfiguration *pc) { for (int i = 0; i < count(); ++i) { QListWidgetItem *lwi = item(i); if (lwi->data(Qt::UserRole).value() == pc) return lwi; } return 0; } ///////// // KitAreaWidget ///////// KitAreaWidget::KitAreaWidget(QWidget *parent) : QWidget(parent), m_layout(new QGridLayout(this)), m_kit(0) { m_layout->setMargin(3); setAutoFillBackground(true); connect(KitManager::instance(), &KitManager::kitUpdated, this, &KitAreaWidget::updateKit); } KitAreaWidget::~KitAreaWidget() { setKit(0); } void KitAreaWidget::setKit(Kit *k) { foreach (KitConfigWidget *w, m_widgets) delete(w); m_widgets.clear(); if (!k) return; foreach (QLabel *l, m_labels) l->deleteLater(); m_labels.clear(); int row = 0; foreach (KitInformation *ki, KitManager::kitInformation()) { if (k && k->isMutable(ki->id())) { KitConfigWidget *widget = ki->createConfigWidget(k); m_widgets << widget; QLabel *label = new QLabel(widget->displayName()); m_labels << label; widget->setStyle(QStyleFactory::create(QLatin1String("fusion"))); widget->setPalette(palette()); m_layout->addWidget(label, row, 0); m_layout->addWidget(widget->mainWidget(), row, 1); m_layout->addWidget(widget->buttonWidget(), row, 2); ++row; } } m_kit = k; setHidden(m_widgets.isEmpty()); } void KitAreaWidget::updateKit(Kit *k) { if (!m_kit || m_kit != k) return; bool addedMutables = false; QList knownIdList = Utils::transform(m_widgets, &KitConfigWidget::kitInformationId); foreach (KitInformation *ki, KitManager::kitInformation()) { Core::Id currentId = ki->id(); if (m_kit->isMutable(currentId) && !knownIdList.removeOne(currentId)) { addedMutables = true; break; } } const bool removedMutables = !knownIdList.isEmpty(); if (addedMutables || removedMutables) { // Redo whole setup if the number of mutable settings did change setKit(m_kit); } else { // Refresh all widgets if the number of mutable settings did not change foreach (KitConfigWidget *w, m_widgets) w->refresh(); } } ///////// // MiniProjectTargetSelector ///////// QWidget *MiniProjectTargetSelector::createTitleLabel(const QString &text) { StyledBar *bar = new StyledBar(this); bar->setSingleRow(true); QVBoxLayout *toolLayout = new QVBoxLayout(bar); toolLayout->setContentsMargins(6, 0, 6, 0); toolLayout->setSpacing(0); QLabel *l = new QLabel(text); QFont f = l->font(); f.setBold(true); l->setFont(f); toolLayout->addWidget(l); int panelHeight = l->fontMetrics().height() + 12; bar->ensurePolished(); // Required since manhattanstyle overrides height bar->setFixedHeight(panelHeight); return bar; } MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorAction, QWidget *parent) : QWidget(parent), m_projectAction(targetSelectorAction), m_project(0), m_target(0), m_buildConfiguration(0), m_deployConfiguration(0), m_runConfiguration(0), m_hideOnRelease(false) { QPalette p; p.setColor(QPalette::Foreground, creatorTheme()->color(Theme::MiniProjectTargetSelectorTextColor)); p.setColor(QPalette::Text, creatorTheme()->color(Theme::MiniProjectTargetSelectorTextColor)); p.setColor(QPalette::ButtonText, creatorTheme()->color(Theme::MiniProjectTargetSelectorTextColor)); p.setColor(QPalette::Background, creatorTheme()->color(Theme::MiniProjectTargetSelectorSummaryBackgroundColor)); p.setColor(QPalette::Base, creatorTheme()->color(Theme::MiniProjectTargetSelectorSummaryBackgroundColor)); p.setColor(QPalette::Button, creatorTheme()->color(Theme::MiniProjectTargetSelectorSummaryBackgroundColor).name()); setPalette(p); setProperty("panelwidget", true); setContentsMargins(QMargins(0, 1, 1, 8)); setWindowFlags(Qt::Popup); targetSelectorAction->setIcon(creatorTheme()->flag(Theme::FlatSideBarIcons) ? Icons::DESKTOP_DEVICE.icon() : style()->standardIcon(QStyle::SP_ComputerIcon)); targetSelectorAction->setProperty("titledAction", true); m_kitAreaWidget = new KitAreaWidget(this); m_summaryLabel = new QLabel(this); m_summaryLabel->setMargin(3); m_summaryLabel->setAlignment(Qt::AlignLeft | Qt::AlignTop); m_summaryLabel->setStyleSheet(QString::fromLatin1("background: %1;") .arg(creatorTheme()->color(Theme::MiniProjectTargetSelectorSummaryBackgroundColor).name())); m_summaryLabel->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); m_summaryLabel->setTextInteractionFlags(m_summaryLabel->textInteractionFlags() | Qt::LinksAccessibleByMouse); m_listWidgets.resize(LAST); m_titleWidgets.resize(LAST); m_listWidgets[PROJECT] = 0; //project is not a generic list widget m_titleWidgets[PROJECT] = createTitleLabel(tr("Project")); m_projectListWidget = new ProjectListWidget(this); QStringList titles; titles << tr("Kit") << tr("Build") << tr("Deploy") << tr("Run"); for (int i = TARGET; i < LAST; ++i) { m_titleWidgets[i] = createTitleLabel(titles.at(i -1)); m_listWidgets[i] = new GenericListWidget(this); } Project *startup = SessionManager::startupProject(); changeStartupProject(startup); if (startup) activeTargetChanged(startup->activeTarget()); connect(m_summaryLabel, &QLabel::linkActivated, this, &MiniProjectTargetSelector::switchToProjectsMode); SessionManager *sessionManager = SessionManager::instance(); connect(sessionManager, &SessionManager::startupProjectChanged, this, &MiniProjectTargetSelector::changeStartupProject); connect(sessionManager, &SessionManager::projectAdded, this, &MiniProjectTargetSelector::projectAdded); connect(sessionManager, &SessionManager::projectRemoved, this, &MiniProjectTargetSelector::projectRemoved); connect(sessionManager, &SessionManager::projectDisplayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); // for icon changes: connect(ProjectExplorer::KitManager::instance(), &KitManager::kitUpdated, this, &MiniProjectTargetSelector::kitChanged); connect(m_listWidgets[TARGET], &GenericListWidget::changeActiveProjectConfiguration, this, &MiniProjectTargetSelector::setActiveTarget); connect(m_listWidgets[BUILD], &GenericListWidget::changeActiveProjectConfiguration, this, &MiniProjectTargetSelector::setActiveBuildConfiguration); connect(m_listWidgets[DEPLOY], &GenericListWidget::changeActiveProjectConfiguration, this, &MiniProjectTargetSelector::setActiveDeployConfiguration); connect(m_listWidgets[RUN], &GenericListWidget::changeActiveProjectConfiguration, this, &MiniProjectTargetSelector::setActiveRunConfiguration); } bool MiniProjectTargetSelector::event(QEvent *event) { if (event->type() == QEvent::LayoutRequest) { doLayout(true); return true; } else if (event->type() == QEvent::ShortcutOverride) { if (static_cast(event)->key() == Qt::Key_Escape) { event->accept(); return true; } } return QWidget::event(event); } // does some fancy calculations to ensure proper widths for the list widgets QVector MiniProjectTargetSelector::listWidgetWidths(int minSize, int maxSize) { QVector result; result.resize(LAST); if (m_projectListWidget->isVisibleTo(this)) result[PROJECT] = m_projectListWidget->optimalWidth(); else result[PROJECT] = -1; for (int i = TARGET; i < LAST; ++i) { if (m_listWidgets[i]->isVisibleTo(this)) result[i] = m_listWidgets[i]->optimalWidth(); else result[i] = -1; } int totalWidth = 0; // Adjust to minimum width of title for (int i = PROJECT; i < LAST; ++i) { if (result[i] != -1) { // We want at least 100 pixels per column int width = qMax(m_titleWidgets[i]->sizeHint().width(), 100); if (result[i] < width) result[i] = width; totalWidth += result[i]; } } if (totalWidth == 0) // All hidden return result; bool tooSmall; if (totalWidth < minSize) tooSmall = true; else if (totalWidth > maxSize) tooSmall = false; else return result; int widthToDistribute = tooSmall ? (minSize - totalWidth) : (totalWidth - maxSize); QVector indexes; indexes.reserve(LAST); for (int i = PROJECT; i < LAST; ++i) if (result[i] != -1) indexes.append(i); if (tooSmall) { Utils::sort(indexes, [&result](int i, int j) { return result[i] < result[j]; }); } else { Utils::sort(indexes, [&result](int i, int j) { return result[i] > result[j]; }); } int i = 0; int first = result[indexes.first()]; // biggest or smallest // we resize the biggest columns until they are the same size as the second biggest // since it looks prettiest if all the columns are the same width while (true) { for (; i < indexes.size(); ++i) { if (result[indexes[i]] != first) break; } int next = tooSmall ? INT_MAX : 0; if (i < indexes.size()) next = result[indexes[i]]; int delta; if (tooSmall) delta = qMin(next - first, widthToDistribute / i); else delta = qMin(first - next, widthToDistribute / i); if (delta == 0) return result; if (tooSmall) { for (int j = 0; j < i; ++j) result[indexes[j]] += delta; } else { for (int j = 0; j < i; ++j) result[indexes[j]] -= delta; } widthToDistribute -= delta * i; if (widthToDistribute == 0) return result; first = result[indexes.first()]; i = 0; // TODO can we do better? } } void MiniProjectTargetSelector::doLayout(bool keepSize) { // An unconfigured project shows empty build/deploy/run sections // if there's a configured project in the seesion // that could be improved static QStatusBar *statusBar = Core::ICore::statusBar(); static QWidget *actionBar = Core::ICore::mainWindow()->findChild(QLatin1String("actionbar")); Q_ASSERT(actionBar); m_kitAreaWidget->move(0, 0); int oldSummaryLabelY = m_summaryLabel->y(); int kitAreaHeight = m_kitAreaWidget->isVisibleTo(this) ? m_kitAreaWidget->sizeHint().height() : 0; // 1. Calculate the summary label height int summaryLabelY = 1 + kitAreaHeight; int summaryLabelHeight = 0; int oldSummaryLabelHeight = m_summaryLabel->height(); bool onlySummary = false; // Count the number of lines int visibleLineCount = m_projectListWidget->isVisibleTo(this) ? 0 : 1; for (int i = TARGET; i < LAST; ++i) visibleLineCount += m_listWidgets[i]->isVisibleTo(this) ? 0 : 1; if (visibleLineCount == LAST) { summaryLabelHeight = visibleLineCount * QFontMetrics(m_summaryLabel->font()).height() + m_summaryLabel->margin() *2; onlySummary = true; } else { if (visibleLineCount < 3) { if (Utils::anyOf(SessionManager::projects(), &Project::needsConfiguration)) visibleLineCount = 3; } if (visibleLineCount) summaryLabelHeight = visibleLineCount * QFontMetrics(m_summaryLabel->font()).height() + m_summaryLabel->margin() *2; } if (keepSize && oldSummaryLabelHeight > summaryLabelHeight) summaryLabelHeight = oldSummaryLabelHeight; m_summaryLabel->move(0, summaryLabelY); // Height to be aligned with side bar button int alignedWithActionHeight = 210; if (actionBar->isVisible()) alignedWithActionHeight = actionBar->height() - statusBar->height(); int bottomMargin = 9; int heightWithoutKitArea = 0; if (!onlySummary) { // list widget height int maxItemCount = m_projectListWidget->maxCount(); for (int i = TARGET; i < LAST; ++i) maxItemCount = qMax(maxItemCount, m_listWidgets[i]->maxCount()); int titleWidgetsHeight = m_titleWidgets.first()->height(); if (keepSize) { heightWithoutKitArea = height() - oldSummaryLabelY + 1; } else { // Clamp the size of the listwidgets to be // at least as high as the sidebar button // and at most twice as high heightWithoutKitArea = summaryLabelHeight + qBound(alignedWithActionHeight, maxItemCount * 30 + bottomMargin + titleWidgetsHeight, alignedWithActionHeight * 2); } int titleY = summaryLabelY + summaryLabelHeight; int listY = titleY + titleWidgetsHeight; int listHeight = heightWithoutKitArea + kitAreaHeight - bottomMargin - listY + 1; // list widget widths int minWidth = qMax(m_summaryLabel->sizeHint().width(), 250); minWidth = qMax(minWidth, m_kitAreaWidget->sizeHint().width()); if (keepSize) { // Do not make the widget smaller then it was before int oldTotalListWidgetWidth = m_projectListWidget->isVisibleTo(this) ? m_projectListWidget->width() : 0; for (int i = TARGET; i < LAST; ++i) oldTotalListWidgetWidth += m_listWidgets[i]->width(); minWidth = qMax(minWidth, oldTotalListWidgetWidth); } QVector widths = listWidgetWidths(minWidth, 1000); int x = 0; for (int i = PROJECT; i < LAST; ++i) { int optimalWidth = widths[i]; if (i == PROJECT) { m_projectListWidget->resize(optimalWidth, listHeight); m_projectListWidget->move(x, listY); } else { m_listWidgets[i]->resize(optimalWidth, listHeight); m_listWidgets[i]->move(x, listY); } m_titleWidgets[i]->resize(optimalWidth, titleWidgetsHeight); m_titleWidgets[i]->move(x, titleY); x += optimalWidth + 1; //1 extra pixel for the separators or the right border } m_summaryLabel->resize(x - 1, summaryLabelHeight); m_kitAreaWidget->resize(x - 1, kitAreaHeight); setFixedSize(x, heightWithoutKitArea + kitAreaHeight); } else { if (keepSize) heightWithoutKitArea = height() - oldSummaryLabelY + 1; else heightWithoutKitArea = qMax(summaryLabelHeight + bottomMargin, alignedWithActionHeight); m_summaryLabel->resize(m_summaryLabel->sizeHint().width(), heightWithoutKitArea - bottomMargin); m_kitAreaWidget->resize(m_kitAreaWidget->sizeHint()); setFixedSize(m_summaryLabel->width() + 1, heightWithoutKitArea + kitAreaHeight); //1 extra pixel for the border } QPoint moveTo = statusBar->mapToGlobal(QPoint(0,0)); moveTo -= QPoint(0, height()); move(moveTo); } void MiniProjectTargetSelector::setActiveTarget(ProjectConfiguration *pc) { SessionManager::setActiveTarget(m_project, static_cast(pc), SetActive::Cascade); } void MiniProjectTargetSelector::setActiveBuildConfiguration(ProjectConfiguration *pc) { SessionManager::setActiveBuildConfiguration(m_target, static_cast(pc), SetActive::Cascade); } void MiniProjectTargetSelector::setActiveDeployConfiguration(ProjectConfiguration *pc) { SessionManager::setActiveDeployConfiguration(m_target, static_cast(pc), SetActive::Cascade); } void MiniProjectTargetSelector::setActiveRunConfiguration(ProjectConfiguration *pc) { m_target->setActiveRunConfiguration(static_cast(pc)); } void MiniProjectTargetSelector::projectAdded(Project *project) { connect(project, &Project::addedTarget, this, &MiniProjectTargetSelector::slotAddedTarget); connect(project, &Project::removedTarget, this, &MiniProjectTargetSelector::slotRemovedTarget); foreach (Target *t, project->targets()) addedTarget(t); updateProjectListVisible(); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::projectRemoved(Project *project) { disconnect(project, &Project::addedTarget, this, &MiniProjectTargetSelector::slotAddedTarget); disconnect(project, &Project::removedTarget, this, &MiniProjectTargetSelector::slotRemovedTarget); foreach (Target *t, project->targets()) removedTarget(t); updateProjectListVisible(); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::addedTarget(Target *target) { connect(target, &Target::addedBuildConfiguration, this, &MiniProjectTargetSelector::slotAddedBuildConfiguration); connect(target, &Target::removedBuildConfiguration, this, &MiniProjectTargetSelector::slotRemovedBuildConfiguration); connect(target, &Target::addedDeployConfiguration, this, &MiniProjectTargetSelector::slotAddedDeployConfiguration); connect(target, &Target::removedDeployConfiguration, this, &MiniProjectTargetSelector::slotRemovedDeployConfiguration); connect(target, &Target::addedRunConfiguration, this, &MiniProjectTargetSelector::slotAddedRunConfiguration); connect(target, &Target::removedRunConfiguration, this, &MiniProjectTargetSelector::slotRemovedRunConfiguration); if (target->project() == m_project) m_listWidgets[TARGET]->addProjectConfiguration(target); foreach (BuildConfiguration *bc, target->buildConfigurations()) addedBuildConfiguration(bc); foreach (DeployConfiguration *dc, target->deployConfigurations()) addedDeployConfiguration(dc); foreach (RunConfiguration *rc, target->runConfigurations()) addedRunConfiguration(rc); } void MiniProjectTargetSelector::slotAddedTarget(Target *target) { addedTarget(target); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::removedTarget(Target *target) { disconnect(target, &Target::addedBuildConfiguration, this, &MiniProjectTargetSelector::slotAddedBuildConfiguration); disconnect(target, &Target::removedBuildConfiguration, this, &MiniProjectTargetSelector::slotRemovedBuildConfiguration); disconnect(target, &Target::addedDeployConfiguration, this, &MiniProjectTargetSelector::slotAddedDeployConfiguration); disconnect(target, &Target::removedDeployConfiguration, this, &MiniProjectTargetSelector::slotRemovedDeployConfiguration); disconnect(target, &Target::addedRunConfiguration, this, &MiniProjectTargetSelector::slotAddedRunConfiguration); disconnect(target, &Target::removedRunConfiguration, this, &MiniProjectTargetSelector::slotRemovedRunConfiguration); if (target->project() == m_project) m_listWidgets[TARGET]->removeProjectConfiguration(target); foreach (BuildConfiguration *bc, target->buildConfigurations()) removedBuildConfiguration(bc); foreach (DeployConfiguration *dc, target->deployConfigurations()) removedDeployConfiguration(dc); foreach (RunConfiguration *rc, target->runConfigurations()) removedRunConfiguration(rc); } void MiniProjectTargetSelector::slotRemovedTarget(Target *target) { removedTarget(target); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::addedBuildConfiguration(BuildConfiguration *bc) { if (bc->target() == m_target) m_listWidgets[BUILD]->addProjectConfiguration(bc); } void MiniProjectTargetSelector::slotAddedBuildConfiguration(BuildConfiguration *bc) { if (bc->target() == m_target) m_listWidgets[BUILD]->addProjectConfiguration(bc); updateBuildListVisible(); } void MiniProjectTargetSelector::removedBuildConfiguration(BuildConfiguration *bc) { if (bc->target() == m_target) m_listWidgets[BUILD]->removeProjectConfiguration(bc); } void MiniProjectTargetSelector::slotRemovedBuildConfiguration(BuildConfiguration *bc) { if (bc->target() == m_target) m_listWidgets[BUILD]->removeProjectConfiguration(bc); updateBuildListVisible(); } void MiniProjectTargetSelector::addedDeployConfiguration(DeployConfiguration *dc) { if (dc->target() == m_target) m_listWidgets[DEPLOY]->addProjectConfiguration(dc); } void MiniProjectTargetSelector::slotAddedDeployConfiguration(DeployConfiguration *dc) { if (dc->target() == m_target) m_listWidgets[DEPLOY]->addProjectConfiguration(dc); updateDeployListVisible(); } void MiniProjectTargetSelector::removedDeployConfiguration(DeployConfiguration *dc) { if (dc->target() == m_target) m_listWidgets[DEPLOY]->removeProjectConfiguration(dc); } void MiniProjectTargetSelector::slotRemovedDeployConfiguration(DeployConfiguration *dc) { if (dc->target() == m_target) m_listWidgets[DEPLOY]->removeProjectConfiguration(dc); updateDeployListVisible(); } void MiniProjectTargetSelector::addedRunConfiguration(RunConfiguration *rc) { if (rc->target() == m_target) m_listWidgets[RUN]->addProjectConfiguration(rc); } void MiniProjectTargetSelector::slotAddedRunConfiguration(RunConfiguration *rc) { if (rc->target() == m_target) m_listWidgets[RUN]->addProjectConfiguration(rc); updateRunListVisible(); } void MiniProjectTargetSelector::removedRunConfiguration(RunConfiguration *rc) { if (rc->target() == m_target) m_listWidgets[RUN]->removeProjectConfiguration(rc); } void MiniProjectTargetSelector::slotRemovedRunConfiguration(RunConfiguration *rc) { if (rc->target() == m_target) m_listWidgets[RUN]->removeProjectConfiguration(rc); updateRunListVisible(); } void MiniProjectTargetSelector::updateProjectListVisible() { int count = SessionManager::projects().size(); bool visible = count > 1; m_projectListWidget->setVisible(visible); m_projectListWidget->setMaxCount(count); m_titleWidgets[PROJECT]->setVisible(visible); updateSummary(); } void MiniProjectTargetSelector::updateTargetListVisible() { int maxCount = 0; foreach (Project *p, SessionManager::projects()) maxCount = qMax(p->targets().size(), maxCount); bool visible = maxCount > 1; m_listWidgets[TARGET]->setVisible(visible); m_listWidgets[TARGET]->setMaxCount(maxCount); m_titleWidgets[TARGET]->setVisible(visible); updateSummary(); } void MiniProjectTargetSelector::updateBuildListVisible() { int maxCount = 0; foreach (Project *p, SessionManager::projects()) foreach (Target *t, p->targets()) maxCount = qMax(t->buildConfigurations().size(), maxCount); bool visible = maxCount > 1; m_listWidgets[BUILD]->setVisible(visible); m_listWidgets[BUILD]->setMaxCount(maxCount); m_titleWidgets[BUILD]->setVisible(visible); updateSummary(); } void MiniProjectTargetSelector::updateDeployListVisible() { int maxCount = 0; foreach (Project *p, SessionManager::projects()) foreach (Target *t, p->targets()) maxCount = qMax(t->deployConfigurations().size(), maxCount); bool visible = maxCount > 1; m_listWidgets[DEPLOY]->setVisible(visible); m_listWidgets[DEPLOY]->setMaxCount(maxCount); m_titleWidgets[DEPLOY]->setVisible(visible); updateSummary(); } void MiniProjectTargetSelector::updateRunListVisible() { int maxCount = 0; foreach (Project *p, SessionManager::projects()) foreach (Target *t, p->targets()) maxCount = qMax(t->runConfigurations().size(), maxCount); bool visible = maxCount > 1; m_listWidgets[RUN]->setVisible(visible); m_listWidgets[RUN]->setMaxCount(maxCount); m_titleWidgets[RUN]->setVisible(visible); updateSummary(); } void MiniProjectTargetSelector::changeStartupProject(Project *project) { if (m_project) { disconnect(m_project, &Project::activeTargetChanged, this, &MiniProjectTargetSelector::activeTargetChanged); } m_project = project; if (m_project) { connect(m_project, &Project::activeTargetChanged, this, &MiniProjectTargetSelector::activeTargetChanged); activeTargetChanged(m_project->activeTarget()); } else { activeTargetChanged(0); } if (project) { QList list; foreach (Target *t, project->targets()) list.append(t); m_listWidgets[TARGET]->setProjectConfigurations(list, project->activeTarget()); } else { m_listWidgets[TARGET]->setProjectConfigurations(QList(), 0); } updateActionAndSummary(); } void MiniProjectTargetSelector::activeTargetChanged(Target *target) { if (m_target) { disconnect(m_target, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); disconnect(m_target, &Target::toolTipChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); disconnect(m_target, &Target::iconChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); disconnect(m_target, &Target::activeBuildConfigurationChanged, this, &MiniProjectTargetSelector::activeBuildConfigurationChanged); disconnect(m_target, &Target::activeDeployConfigurationChanged, this, &MiniProjectTargetSelector::activeDeployConfigurationChanged); disconnect(m_target, &Target::activeRunConfigurationChanged, this, &MiniProjectTargetSelector::activeRunConfigurationChanged); } m_target = target; m_kitAreaWidget->setKit(m_target ? m_target->kit() : 0); m_listWidgets[TARGET]->setActiveProjectConfiguration(m_target); if (m_buildConfiguration) disconnect(m_buildConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); if (m_deployConfiguration) disconnect(m_deployConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); if (m_runConfiguration) disconnect(m_runConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); if (m_target) { QList bl; foreach (BuildConfiguration *bc, target->buildConfigurations()) bl.append(bc); m_listWidgets[BUILD]->setProjectConfigurations(bl, target->activeBuildConfiguration()); QList dl; foreach (DeployConfiguration *dc, target->deployConfigurations()) dl.append(dc); m_listWidgets[DEPLOY]->setProjectConfigurations(dl, target->activeDeployConfiguration()); QList rl; foreach (RunConfiguration *rc, target->runConfigurations()) rl.append(rc); m_listWidgets[RUN]->setProjectConfigurations(rl, target->activeRunConfiguration()); m_buildConfiguration = m_target->activeBuildConfiguration(); if (m_buildConfiguration) connect(m_buildConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_deployConfiguration = m_target->activeDeployConfiguration(); if (m_deployConfiguration) connect(m_deployConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_runConfiguration = m_target->activeRunConfiguration(); if (m_runConfiguration) connect(m_runConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); connect(m_target, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); connect(m_target, &Target::toolTipChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); connect(m_target, &Target::iconChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); connect(m_target, &Target::activeBuildConfigurationChanged, this, &MiniProjectTargetSelector::activeBuildConfigurationChanged); connect(m_target, &Target::activeDeployConfigurationChanged, this, &MiniProjectTargetSelector::activeDeployConfigurationChanged); connect(m_target, &Target::activeRunConfigurationChanged, this, &MiniProjectTargetSelector::activeRunConfigurationChanged); } else { m_listWidgets[BUILD]->setProjectConfigurations(QList(), 0); m_listWidgets[DEPLOY]->setProjectConfigurations(QList(), 0); m_listWidgets[RUN]->setProjectConfigurations(QList(), 0); m_buildConfiguration = 0; m_deployConfiguration = 0; m_runConfiguration = 0; } updateActionAndSummary(); } void MiniProjectTargetSelector::kitChanged(Kit *k) { if (m_target && m_target->kit() == k) updateActionAndSummary(); } void MiniProjectTargetSelector::activeBuildConfigurationChanged(BuildConfiguration *bc) { if (m_buildConfiguration) disconnect(m_buildConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_buildConfiguration = bc; if (m_buildConfiguration) connect(m_buildConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_listWidgets[BUILD]->setActiveProjectConfiguration(bc); updateActionAndSummary(); } void MiniProjectTargetSelector::activeDeployConfigurationChanged(DeployConfiguration *dc) { if (m_deployConfiguration) disconnect(m_deployConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_deployConfiguration = dc; if (m_deployConfiguration) connect(m_deployConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_listWidgets[DEPLOY]->setActiveProjectConfiguration(dc); updateActionAndSummary(); } void MiniProjectTargetSelector::activeRunConfigurationChanged(RunConfiguration *rc) { if (m_runConfiguration) disconnect(m_runConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_runConfiguration = rc; if (m_runConfiguration) connect(m_runConfiguration, &ProjectConfiguration::displayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); m_listWidgets[RUN]->setActiveProjectConfiguration(rc); updateActionAndSummary(); } void MiniProjectTargetSelector::setVisible(bool visible) { doLayout(false); QWidget::setVisible(visible); m_projectAction->setChecked(visible); if (visible) { if (!focusWidget() || !focusWidget()->isVisibleTo(this)) { // Does the second part actually work? if (m_projectListWidget->isVisibleTo(this)) m_projectListWidget->setFocus(); for (int i = TARGET; i < LAST; ++i) { if (m_listWidgets[i]->isVisibleTo(this)) { m_listWidgets[i]->setFocus(); break; } } } } } void MiniProjectTargetSelector::toggleVisible() { setVisible(!isVisible()); } void MiniProjectTargetSelector::nextOrShow() { if (!isVisible()) { show(); } else { m_hideOnRelease = true; m_earliestHidetime = QDateTime::currentDateTime().addMSecs(800); if (ListWidget *lw = qobject_cast(focusWidget())) { if (lw->currentRow() < lw->count() -1) lw->setCurrentRow(lw->currentRow() + 1); else lw->setCurrentRow(0); } } } void MiniProjectTargetSelector::keyPressEvent(QKeyEvent *ke) { if (ke->key() == Qt::Key_Return || ke->key() == Qt::Key_Enter || ke->key() == Qt::Key_Space || ke->key() == Qt::Key_Escape) { hide(); } else { QWidget::keyPressEvent(ke); } } void MiniProjectTargetSelector::keyReleaseEvent(QKeyEvent *ke) { if (m_hideOnRelease) { if (ke->modifiers() == 0 /*HACK this is to overcome some event inconsistencies between platforms*/ || (ke->modifiers() == Qt::AltModifier && (ke->key() == Qt::Key_Alt || ke->key() == -1))) { delayedHide(); m_hideOnRelease = false; } } if (ke->key() == Qt::Key_Return || ke->key() == Qt::Key_Enter || ke->key() == Qt::Key_Space || ke->key() == Qt::Key_Escape) return; QWidget::keyReleaseEvent(ke); } void MiniProjectTargetSelector::delayedHide() { QDateTime current = QDateTime::currentDateTime(); if (m_earliestHidetime > current) { // schedule for later QTimer::singleShot(current.msecsTo(m_earliestHidetime) + 50, this, &MiniProjectTargetSelector::delayedHide); } else { hide(); } } // This is a workaround for the problem that Windows // will let the mouse events through when you click // outside a popup to close it. This causes the popup // to open on mouse release if you hit the button, which // // // A similar case can be found in QComboBox void MiniProjectTargetSelector::mousePressEvent(QMouseEvent *e) { setAttribute(Qt::WA_NoMouseReplay); QWidget::mousePressEvent(e); } void MiniProjectTargetSelector::updateActionAndSummary() { QString projectName = QLatin1String(" "); QString fileName; // contains the path if projectName is not unique QString targetName; QString targetToolTipText; QString buildConfig; QString deployConfig; QString runConfig; QIcon targetIcon = creatorTheme()->flag(Theme::FlatSideBarIcons) ? Icons::DESKTOP_DEVICE.icon() : style()->standardIcon(QStyle::SP_ComputerIcon); Project *project = SessionManager::startupProject(); if (project) { projectName = project->displayName(); foreach (Project *p, SessionManager::projects()) { if (p != project && p->displayName() == projectName) { fileName = project->projectFilePath().toUserOutput(); break; } } if (Target *target = project->activeTarget()) { targetName = project->activeTarget()->displayName(); if (BuildConfiguration *bc = target->activeBuildConfiguration()) buildConfig = bc->displayName(); if (DeployConfiguration *dc = target->activeDeployConfiguration()) deployConfig = dc->displayName(); if (RunConfiguration *rc = target->activeRunConfiguration()) runConfig = rc->displayName(); targetToolTipText = target->toolTip(); targetIcon = createCenteredIcon(target->icon(), target->overlayIcon()); } } m_projectAction->setProperty("heading", projectName); if (project && project->needsConfiguration()) m_projectAction->setProperty("subtitle", tr("Unconfigured")); else m_projectAction->setProperty("subtitle", buildConfig); m_projectAction->setIcon(targetIcon); QStringList lines; lines << tr("Project: %1").arg(projectName); if (!fileName.isEmpty()) lines << tr("Path: %1").arg(fileName); if (!targetName.isEmpty()) lines << tr("Kit: %1").arg(targetName); if (!buildConfig.isEmpty()) lines << tr("Build: %1").arg(buildConfig); if (!deployConfig.isEmpty()) lines << tr("Deploy: %1").arg(deployConfig); if (!runConfig.isEmpty()) lines << tr("Run: %1").arg(runConfig); if (!targetToolTipText.isEmpty()) lines << tr("%1").arg(targetToolTipText); QString toolTip = tr("%1") .arg(lines.join(QLatin1String("
"))); m_projectAction->setToolTip(toolTip); updateSummary(); } void MiniProjectTargetSelector::updateSummary() { QString summary; if (Project *startupProject = SessionManager::startupProject()) { if (!m_projectListWidget->isVisibleTo(this)) summary.append(tr("Project: %1
").arg(startupProject->displayName())); if (Target *activeTarget = startupProject->activeTarget()) { if (!m_listWidgets[TARGET]->isVisibleTo(this)) summary.append(tr("Kit: %1
").arg( activeTarget->displayName())); if (!m_listWidgets[BUILD]->isVisibleTo(this) && activeTarget->activeBuildConfiguration()) summary.append(tr("Build: %1
").arg( activeTarget->activeBuildConfiguration()->displayName())); if (!m_listWidgets[DEPLOY]->isVisibleTo(this) && activeTarget->activeDeployConfiguration()) summary.append(tr("Deploy: %1
").arg( activeTarget->activeDeployConfiguration()->displayName())); if (!m_listWidgets[RUN]->isVisibleTo(this) && activeTarget->activeRunConfiguration()) summary.append(tr("Run: %1
").arg( activeTarget->activeRunConfiguration()->displayName())); } else if (startupProject->needsConfiguration()) { summary = tr("" "The project %1 is not yet configured

" "You can configure it in the Projects mode
") .arg(startupProject->displayName()); } else { if (!m_listWidgets[TARGET]->isVisibleTo(this)) summary.append(QLatin1String("
")); if (!m_listWidgets[BUILD]->isVisibleTo(this)) summary.append(QLatin1String("
")); if (!m_listWidgets[DEPLOY]->isVisibleTo(this)) summary.append(QLatin1String("
")); if (!m_listWidgets[RUN]->isVisibleTo(this)) summary.append(QLatin1String("
")); } } m_summaryLabel->setText(summary); } void MiniProjectTargetSelector::paintEvent(QPaintEvent *) { QPainter painter(this); painter.setBrush(creatorTheme()->color(Theme::MiniProjectTargetSelectorBackgroundColor)); painter.drawRect(rect()); painter.setPen(creatorTheme()->color(Theme::MiniProjectTargetSelectorBorderColor)); // draw border on top and right QRectF borderRect = QRectF(rect()).adjusted(0.5, 0.5, -0.5, -0.5); painter.drawLine(borderRect.topLeft(), borderRect.topRight()); painter.drawLine(borderRect.topRight(), borderRect.bottomRight()); if (creatorTheme()->flag(Theme::DrawTargetSelectorBottom)) { // draw thicker border on the bottom QRect bottomRect(0, rect().height() - 8, rect().width(), 8); static QImage image(QLatin1String(":/projectexplorer/images/targetpanel_bottom.png")); StyleHelper::drawCornerImage(image, &painter, bottomRect, 1, 1, 1, 1); } } void MiniProjectTargetSelector::switchToProjectsMode() { Core::ModeManager::activateMode(Constants::MODE_SESSION); hide(); }