// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "miniprojecttargetselector.h" #include "buildconfiguration.h" #include "buildmanager.h" #include "deployconfiguration.h" #include "kit.h" #include "kitmanager.h" #include "project.h" #include "projectexplorer.h" #include "projectexplorerconstants.h" #include "projectexplorericons.h" #include "projectexplorertr.h" #include "projectmanager.h" #include "runconfiguration.h" #include "target.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 using namespace Utils; namespace ProjectExplorer::Internal { const int RunColumnWidth = 30; static QIcon createCenteredIcon(const QIcon &icon, const QIcon &overlay) { QPixmap targetPixmap; const qreal appDevicePixelRatio = qApp->devicePixelRatio(); const auto deviceSpaceIconSize = static_cast(Core::Constants::MODEBAR_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::MODEBAR_ICON_SIZE); // already takes app devicePixelRatio into account qreal pixmapDevicePixelRatio = pixmap.devicePixelRatio(); painter.drawPixmap((Core::Constants::MODEBAR_ICON_SIZE - pixmap.width() / pixmapDevicePixelRatio) / 2, (Core::Constants::MODEBAR_ICON_SIZE - pixmap.height() / pixmapDevicePixelRatio) / 2, pixmap); if (!overlay.isNull()) { pixmap = overlay.pixmap(Core::Constants::MODEBAR_ICON_SIZE); // already takes app devicePixelRatio into account pixmapDevicePixelRatio = pixmap.devicePixelRatio(); painter.drawPixmap((Core::Constants::MODEBAR_ICON_SIZE - pixmap.width() / pixmapDevicePixelRatio) / 2, (Core::Constants::MODEBAR_ICON_SIZE - pixmap.height() / pixmapDevicePixelRatio) / 2, pixmap); } return QIcon(targetPixmap); } class GenericItem : public TreeItem { public: GenericItem() = default; GenericItem(QObject *object) : m_object(object) {} QObject *object() const { return m_object; } QString rawDisplayName() const { if (const auto p = qobject_cast(object())) return p->displayName(); if (const auto t = qobject_cast(object())) return t->displayName(); return static_cast(object())->expandedDisplayName(); } QString displayName() const { if (const auto p = qobject_cast(object())) { const auto hasSameProjectName = [this](TreeItem *ti) { return ti != this && static_cast(ti)->rawDisplayName() == rawDisplayName(); }; QString displayName = p->displayName(); if (parent()->findAnyChild(hasSameProjectName)) { displayName.append(" (").append(p->projectFilePath().toUserOutput()) .append(')'); } return displayName; } return rawDisplayName(); } private: QVariant toolTip() const { if (qobject_cast(object())) return {}; if (const auto t = qobject_cast(object())) return t->toolTip(); return static_cast(object())->toolTip(); } QVariant data(int column, int role) const override { if (column == 1 && role == Qt::ToolTipRole) return Tr::tr("Run Without Deployment"); if (column != 0) return {}; switch (role) { case Qt::DisplayRole: return displayName(); case Qt::ToolTipRole: return toolTip(); default: break; } return {}; } QObject *m_object = nullptr; }; static bool compareItems(const TreeItem *ti1, const TreeItem *ti2) { return caseFriendlyCompare(static_cast(ti1)->rawDisplayName(), static_cast(ti2)->rawDisplayName()) < 0; } class GenericModel : public TreeModel { Q_OBJECT public: GenericModel(QObject *parent) : TreeModel(parent) { } void rebuild(const QList &objects) { clear(); for (QObject * const e : objects) addItemForObject(e); } const GenericItem *addItemForObject(QObject *object) { const auto item = new GenericItem(object); rootItem()->insertOrderedChild(item, &compareItems); if (const auto project = qobject_cast(object)) { connect(project, &Project::displayNameChanged, this, &GenericModel::displayNameChanged); } else if (const auto target = qobject_cast(object)) { connect(target, &Target::kitChanged, this, &GenericModel::displayNameChanged); } else { const auto pc = qobject_cast(object); QTC_CHECK(pc); connect(pc, &ProjectConfiguration::displayNameChanged, this, &GenericModel::displayNameChanged); connect(pc, &ProjectConfiguration::toolTipChanged, this, &GenericModel::updateToolTips); } return item; } GenericItem *itemForObject(const QObject *object) const { return findItemAtLevel<1>([object](const GenericItem *item) { return item->object() == object; }); } void setColumnCount(int columns) { m_columnCount = columns; } signals: void displayNameChanged(); private: void updateToolTips() { emit dataChanged(index(0, 0), index(rowCount() - 1, 0), {Qt::ToolTipRole}); } }; class SelectorView : public TreeView { Q_OBJECT public: SelectorView(QWidget *parent); void setMaxCount(int maxCount); int maxCount(); int optimalWidth() const; void setOptimalWidth(int width); int padding(); GenericModel *theModel() const { return static_cast(model()); } protected: void resetOptimalWidth() { if (m_resetScheduled) return; m_resetScheduled = true; QMetaObject::invokeMethod(this, &SelectorView::doResetOptimalWidth, Qt::QueuedConnection); } private: void keyPressEvent(QKeyEvent *event) override; void keyReleaseEvent(QKeyEvent *event) override; void doResetOptimalWidth() { m_resetScheduled = false; int width = 0; QFontMetrics fn(font()); theModel()->forItemsAtLevel<1>([this, &width, &fn](const GenericItem *item) { width = qMax(fn.horizontalAdvance(item->displayName()) + padding(), width); }); setOptimalWidth(width); } int m_maxCount = 0; int m_optimalWidth = 0; bool m_resetScheduled = false; }; class ProjectListView : public SelectorView { Q_OBJECT public: explicit ProjectListView(QWidget *parent = nullptr) : SelectorView(parent) { const auto model = new GenericModel(this); model->rebuild(transform>(ProjectManager::projects(), [](Project *p) { return p; })); connect(ProjectManager::instance(), &ProjectManager::projectAdded, this, [this, model](Project *project) { const GenericItem *projectItem = model->addItemForObject(project); QFontMetrics fn(font()); const int width = fn.horizontalAdvance(projectItem->displayName()) + padding(); if (width > optimalWidth()) setOptimalWidth(width); restoreCurrentIndex(); }); connect(ProjectManager::instance(), &ProjectManager::aboutToRemoveProject, this, [this, model](const Project *project) { GenericItem * const item = model->itemForObject(project); if (!item) return; model->destroyItem(item); resetOptimalWidth(); }); connect(ProjectManager::instance(), &ProjectManager::startupProjectChanged, this, [this, model](const Project *project) { const GenericItem * const item = model->itemForObject(project); if (item) setCurrentIndex(item->index()); }); connect(model, &GenericModel::displayNameChanged, this, [this, model] { model->rootItem()->sortChildren(&compareItems); resetOptimalWidth(); restoreCurrentIndex(); }); setModel(model); connect(selectionModel(), &QItemSelectionModel::currentChanged, this, [model](const QModelIndex &index) { const GenericItem * const item = model->itemForIndex(index); if (item && item->object()) ProjectManager::setStartupProject(qobject_cast(item->object())); }); } private: void restoreCurrentIndex() { const GenericItem * const itemForStartupProject = theModel()->itemForObject(ProjectManager::startupProject()); if (itemForStartupProject) setCurrentIndex(theModel()->indexForItem(itemForStartupProject)); } }; class GenericListWidget : public SelectorView { Q_OBJECT public: explicit GenericListWidget(QWidget *parent = nullptr) : SelectorView(parent) { const auto model = new GenericModel(this); connect(model, &GenericModel::displayNameChanged, this, [this, model] { const GenericItem * const activeItem = model->itemForIndex(currentIndex()); model->rootItem()->sortChildren(&compareItems); resetOptimalWidth(); if (activeItem) setCurrentIndex(activeItem->index()); }); setModel(model); connect(selectionModel(), &QItemSelectionModel::currentChanged, this, &GenericListWidget::rowChanged); } signals: void changeActiveProjectConfiguration(QObject *pc); public: void setProjectConfigurations(const QList &list, QObject *active) { theModel()->rebuild(list); resetOptimalWidth(); setActiveProjectConfiguration(active); } void setActiveProjectConfiguration(QObject *active) { if (const GenericItem * const item = theModel()->itemForObject(active)) setCurrentIndex(item->index()); } void addProjectConfiguration(QObject *pc) { const auto activeItem = theModel()->itemForIndex(currentIndex()); const auto item = theModel()->addItemForObject(pc); QFontMetrics fn(font()); const int width = fn.horizontalAdvance(item->displayName()) + padding(); if (width > optimalWidth()) setOptimalWidth(width); if (activeItem) setCurrentIndex(activeItem->index()); } void removeProjectConfiguration(QObject *pc) { const auto activeItem = theModel()->itemForIndex(currentIndex()); if (GenericItem * const item = theModel()->itemForObject(pc)) { theModel()->destroyItem(item); resetOptimalWidth(); if (activeItem && activeItem != item) setCurrentIndex(activeItem->index()); } } private: void mousePressEvent(QMouseEvent *event) override { const QModelIndex pressedIndex = indexAt(event->pos()); if (pressedIndex.column() == 1) { m_pressedIndex = pressedIndex; return; // Clicking on the run button should not change the current index } m_pressedIndex = QModelIndex(); TreeView::mousePressEvent(event); } void mouseReleaseEvent(QMouseEvent *event) override { const QModelIndex pressedIndex = m_pressedIndex; m_pressedIndex = QModelIndex(); if (pressedIndex.isValid() && pressedIndex == indexAt(event->pos())) { const auto rc = qobject_cast( theModel()->itemForIndex(pressedIndex)->object()); QTC_ASSERT(rc, return); if (!BuildManager::isBuilding(rc->project())) ProjectExplorerPlugin::runRunConfiguration(rc, Constants::NORMAL_RUN_MODE, true); return; } TreeView::mouseReleaseEvent(event); } void showEvent(QShowEvent* event) override { scrollTo(currentIndex()); TreeView::showEvent(event); } QObject *objectAt(const QModelIndex &index) const { return theModel()->itemForIndex(index)->object(); } void rowChanged(const QModelIndex &index) { if (index.isValid()) emit changeActiveProjectConfiguration(objectAt(index)); } QModelIndex m_pressedIndex; }; //////// // TargetSelectorDelegate //////// class TargetSelectorDelegate : public QItemDelegate { public: TargetSelectorDelegate(SelectorView *parent) : QItemDelegate(parent), m_view(parent) { } private: QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const override; void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const override; SelectorView *m_view; }; QSize TargetSelectorDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const { Q_UNUSED(option) Q_UNUSED(index) return QSize(m_view->size().width(), 30); } void TargetSelectorDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { painter->save(); painter->setClipping(false); QColor textColor = creatorTheme()->color(Theme::MiniProjectTargetSelectorTextColor); if (option.state & QStyle::State_Selected) { QColor color; if (m_view->hasFocus()) { color = option.palette.highlight().color(); textColor = option.palette.highlightedText().color(); } else { color = option.palette.dark().color(); } if (creatorTheme()->flag(Theme::FlatToolBars)) { painter->fillRect(option.rect, color); } else { painter->fillRect(option.rect, color.darker(140)); static const QImage selectionGradient(":/projectexplorer/images/targetpanel_gradient.png"); 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(textColor); 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, index.model()->data(index, Qt::UserRole + 1).toString(), Qt::ToolTipRole); painter->drawText(option.rect.left() + 6, option.rect.top() + (option.rect.height() - fm.height()) / 2 + fm.ascent(), elidedText); if (index.column() == 1 && option.state & QStyle::State_MouseOver) { const QIcon icon = Utils::Icons::RUN_SMALL_TOOLBAR.icon(); QRect iconRect(0, 0, 16, 16); iconRect.moveCenter(option.rect.center()); icon.paint(painter, iconRect); } painter->restore(); } //////// // ListWidget //////// SelectorView::SelectorView(QWidget *parent) : TreeView(parent) { setFocusPolicy(Qt::NoFocus); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setAlternatingRowColors(false); setIndentation(0); setFocusPolicy(Qt::WheelFocus); setItemDelegate(new TargetSelectorDelegate(this)); setSelectionBehavior(SelectRows); setAttribute(Qt::WA_MacShowFocusRect, false); setHeaderHidden(true); const QColor bgColor = creatorTheme()->color(Theme::MiniProjectTargetSelectorBackgroundColor); const QString bgColorName = creatorTheme()->flag(Theme::FlatToolBars) ? bgColor.lighter(120).name() : bgColor.name(); setStyleSheet(QString::fromLatin1("QAbstractItemView { background: %1; border-style: none; }").arg(bgColorName)); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); } void SelectorView::keyPressEvent(QKeyEvent *event) { if (event->key() == Qt::Key_Left) focusPreviousChild(); else if (event->key() == Qt::Key_Right) focusNextChild(); else TreeView::keyPressEvent(event); } void SelectorView::keyReleaseEvent(QKeyEvent *event) { if (event->key() != Qt::Key_Left && event->key() != Qt::Key_Right) TreeView::keyReleaseEvent(event); } void SelectorView::setMaxCount(int maxCount) { m_maxCount = maxCount; updateGeometry(); } int SelectorView::maxCount() { return m_maxCount; } int SelectorView::optimalWidth() const { return m_optimalWidth; } void SelectorView::setOptimalWidth(int width) { m_optimalWidth = width; if (model()->columnCount() == 2) m_optimalWidth += RunColumnWidth; updateGeometry(); } int SelectorView::padding() { // there needs to be enough extra pixels to show a scrollbar return 2 * style()->pixelMetric(QStyle::PM_FocusFrameHMargin, nullptr, this) + style()->pixelMetric(QStyle::PM_ScrollBarExtent, nullptr, this) + 10; } ///////// // KitAreaWidget ///////// void doLayout(KitAspect *aspect, Layouting::LayoutItem &builder) { aspect->addToLayout(builder); } class KitAreaWidget : public QWidget { public: explicit KitAreaWidget(QWidget *parent) : QWidget(parent) { connect(KitManager::instance(), &KitManager::kitUpdated, this, &KitAreaWidget::updateKit); auto layout = new QVBoxLayout; layout->setContentsMargins({}); setLayout(layout); } ~KitAreaWidget() override { setKit(nullptr); } void setKit(Kit *k) { qDeleteAll(m_kitAspects); m_kitAspects.clear(); delete m_gridWidget; m_gridWidget = nullptr; if (!k) return; Layouting::Grid grid; for (KitAspectFactory *factory : KitManager::kitAspectFactories()) { if (k && k->isMutable(factory->id())) { KitAspect *aspect = factory->createKitAspect(k); m_kitAspects << aspect; grid.addItems({aspect, Layouting::br}); } } m_gridWidget = grid.emerge(); m_gridWidget->layout()->setContentsMargins(3, 3, 3, 3); layout()->addWidget(m_gridWidget); m_kit = k; setHidden(m_kitAspects.isEmpty()); } private: void updateKit(Kit *k) { if (!m_kit || m_kit != k) return; bool addedMutables = false; QList knownList = Utils::transform(m_kitAspects, &KitAspect::factory); for (KitAspectFactory *factory : KitManager::kitAspectFactories()) { const Utils::Id currentId = factory->id(); if (m_kit->isMutable(currentId) && !knownList.removeOne(factory)) { addedMutables = true; break; } } const bool removedMutables = !knownList.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 for (KitAspect *w : std::as_const(m_kitAspects)) w->refresh(); } } Kit *m_kit = nullptr; QWidget *m_gridWidget = nullptr; QList m_kitAspects; }; ///////// // MiniProjectTargetSelector ///////// QWidget *MiniProjectTargetSelector::createTitleLabel(const QString &text) { auto *bar = new StyledBar(this); bar->setSingleRow(true); auto *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) { StyleHelper::setPanelWidget(this); 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->setContentsMargins(3, 3, 3, 3); m_summaryLabel->setAlignment(Qt::AlignLeft | Qt::AlignTop); QPalette pal = m_summaryLabel->palette(); pal.setColor(QPalette::Window, StyleHelper::baseColor()); m_summaryLabel->setPalette(pal); 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] = nullptr; //project is not a generic list widget m_titleWidgets[PROJECT] = createTitleLabel(Tr::tr("Project")); m_projectListWidget = new ProjectListView(this); connect(m_projectListWidget, &QAbstractItemView::doubleClicked, this, &MiniProjectTargetSelector::hide); QStringList titles; titles << Tr::tr("Kit") << Tr::tr("Build") << Tr::tr("Deploy") << Tr::tr("Run"); for (int i = TARGET; i < LAST; ++i) { m_titleWidgets[i] = createTitleLabel(titles.at(i -1)); m_listWidgets[i] = new GenericListWidget(this); connect(m_listWidgets[i], &QAbstractItemView::doubleClicked, this, &MiniProjectTargetSelector::hide); } m_listWidgets[RUN]->theModel()->setColumnCount(2); m_listWidgets[RUN]->viewport()->setAttribute(Qt::WA_Hover); // Validate state: At this point the session is still empty! Project *startup = ProjectManager::startupProject(); QTC_CHECK(!startup); QTC_CHECK(ProjectManager::projects().isEmpty()); connect(m_summaryLabel, &QLabel::linkActivated, this, &MiniProjectTargetSelector::switchToProjectsMode); ProjectManager *sessionManager = ProjectManager::instance(); connect(sessionManager, &ProjectManager::startupProjectChanged, this, &MiniProjectTargetSelector::changeStartupProject); connect(sessionManager, &ProjectManager::projectAdded, this, &MiniProjectTargetSelector::projectAdded); connect(sessionManager, &ProjectManager::projectRemoved, this, &MiniProjectTargetSelector::projectRemoved); connect(sessionManager, &ProjectManager::projectDisplayNameChanged, this, &MiniProjectTargetSelector::updateActionAndSummary); // for icon changes: connect(ProjectExplorer::KitManager::instance(), &KitManager::kitUpdated, this, &MiniProjectTargetSelector::kitChanged); connect(m_listWidgets[TARGET], &GenericListWidget::changeActiveProjectConfiguration, this, [this](QObject *pc) { m_project->setActiveTarget(static_cast(pc), SetActive::Cascade); }); connect(m_listWidgets[BUILD], &GenericListWidget::changeActiveProjectConfiguration, this, [this](QObject *pc) { m_project->activeTarget()->setActiveBuildConfiguration( static_cast(pc), SetActive::Cascade); }); connect(m_listWidgets[DEPLOY], &GenericListWidget::changeActiveProjectConfiguration, this, [this](QObject *pc) { m_project->activeTarget()->setActiveDeployConfiguration( static_cast(pc), SetActive::Cascade); }); connect(m_listWidgets[RUN], &GenericListWidget::changeActiveProjectConfiguration, this, [this](QObject *pc) { m_project->activeTarget()->setActiveRunConfiguration(static_cast(pc)); }); } bool MiniProjectTargetSelector::event(QEvent *event) { if (event->type() == QEvent::ShortcutOverride && 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 / qMax(i, 1)); else delta = qMin(first - next, widthToDistribute / qMax(i, 1)); 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() { // 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 auto *actionBar = Core::ICore::mainWindow()->findChild(QLatin1String("actionbar")); Q_ASSERT(actionBar); m_kitAreaWidget->move(0, 0); int kitAreaHeight = m_kitAreaWidget->isVisibleTo(this) ? m_kitAreaWidget->sizeHint().height() : 0; // 1. Calculate the summary label height int summaryLabelY = 1 + kitAreaHeight; int summaryLabelHeight = 0; 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 = m_summaryLabel->sizeHint().height(); onlySummary = true; } else { if (visibleLineCount < 3) { if (Utils::anyOf(ProjectManager::projects(), &Project::needsConfiguration)) visibleLineCount = 3; } if (visibleLineCount) summaryLabelHeight = m_summaryLabel->sizeHint().height(); } m_summaryLabel->move(0, summaryLabelY); // Height to be aligned with side bar button int alignedWithActionHeight = 210; if (actionBar->isVisible()) alignedWithActionHeight = qMax(0, actionBar->height() - statusBar->height()); int bottomMargin = 9; int heightWithoutKitArea = 0; QRect newGeometry; 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(); // Clamp the size of the listwidgets to be at least as high as the sidebar button // and at most half the height of the entire Qt Creator window. heightWithoutKitArea = summaryLabelHeight + qBound(alignedWithActionHeight, maxItemCount * 30 + bottomMargin + titleWidgetsHeight, Core::ICore::mainWindow()->height() / 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()); QVector widths = listWidgetWidths(minWidth, Core::ICore::mainWindow()->width() * 0.9); const int runColumnWidth = widths[RUN] == -1 ? 0 : RunColumnWidth; 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 { if (i == RUN) optimalWidth += runColumnWidth; 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_listWidgets[RUN]->setColumnWidth(0, m_listWidgets[RUN]->size().width() - runColumnWidth - m_listWidgets[RUN]->padding()); m_listWidgets[RUN]->setColumnWidth(1, runColumnWidth); m_summaryLabel->resize(x - 1, summaryLabelHeight); m_kitAreaWidget->resize(x - 1, kitAreaHeight); newGeometry.setSize({x, heightWithoutKitArea + kitAreaHeight}); } else { heightWithoutKitArea = qMax(summaryLabelHeight + bottomMargin, alignedWithActionHeight); m_summaryLabel->resize(m_summaryLabel->sizeHint().width(), heightWithoutKitArea - bottomMargin); m_kitAreaWidget->resize(m_kitAreaWidget->sizeHint()); newGeometry.setSize({m_summaryLabel->width() + 1, heightWithoutKitArea + kitAreaHeight}); } newGeometry.translate(statusBar->mapToGlobal(QPoint{0, 0})); newGeometry.translate(QPoint{0, -newGeometry.height()}); repaint(); setGeometry(newGeometry); } void MiniProjectTargetSelector::projectAdded(Project *project) { connect(project, &Project::addedTarget, this, &MiniProjectTargetSelector::handleNewTarget); connect(project, &Project::removedTarget, this, &MiniProjectTargetSelector::handleRemovalOfTarget); const QList targets = project->targets(); for (Target *t : targets) addedTarget(t); updateProjectListVisible(); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::projectRemoved(Project *project) { disconnect(project, &Project::addedTarget, this, &MiniProjectTargetSelector::handleNewTarget); disconnect(project, &Project::removedTarget, this, &MiniProjectTargetSelector::handleRemovalOfTarget); const QList targets = project->targets(); for (Target *t : targets) removedTarget(t); updateProjectListVisible(); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::handleNewTarget(Target *target) { addedTarget(target); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::handleRemovalOfTarget(Target *target) { removedTarget(target); updateTargetListVisible(); updateBuildListVisible(); updateDeployListVisible(); updateRunListVisible(); } void MiniProjectTargetSelector::addedTarget(Target *target) { if (target->project() != m_project) return; m_listWidgets[TARGET]->addProjectConfiguration(target); for (BuildConfiguration *bc : target->buildConfigurations()) addedBuildConfiguration(bc, false); for (DeployConfiguration *dc : target->deployConfigurations()) addedDeployConfiguration(dc, false); for (RunConfiguration *rc : target->runConfigurations()) addedRunConfiguration(rc, false); } void MiniProjectTargetSelector::removedTarget(Target *target) { if (target->project() != m_project) return; m_listWidgets[TARGET]->removeProjectConfiguration(target); for (BuildConfiguration *bc : target->buildConfigurations()) removedBuildConfiguration(bc, false); for (DeployConfiguration *dc : target->deployConfigurations()) removedDeployConfiguration(dc, false); for (RunConfiguration *rc : target->runConfigurations()) removedRunConfiguration(rc, false); } void MiniProjectTargetSelector::addedBuildConfiguration(BuildConfiguration *bc, bool update) { if (!m_project || bc->target() != m_project->activeTarget()) return; m_listWidgets[BUILD]->addProjectConfiguration(bc); if (update) updateBuildListVisible(); } void MiniProjectTargetSelector::removedBuildConfiguration(BuildConfiguration *bc, bool update) { if (!m_project || bc->target() != m_project->activeTarget()) return; m_listWidgets[BUILD]->removeProjectConfiguration(bc); if (update) updateBuildListVisible(); } void MiniProjectTargetSelector::addedDeployConfiguration(DeployConfiguration *dc, bool update) { if (!m_project || dc->target() != m_project->activeTarget()) return; m_listWidgets[DEPLOY]->addProjectConfiguration(dc); if (update) updateDeployListVisible(); } void MiniProjectTargetSelector::removedDeployConfiguration(DeployConfiguration *dc, bool update) { if (!m_project || dc->target() != m_project->activeTarget()) return; m_listWidgets[DEPLOY]->removeProjectConfiguration(dc); if (update) updateDeployListVisible(); } void MiniProjectTargetSelector::addedRunConfiguration(RunConfiguration *rc, bool update) { if (!m_project || rc->target() != m_project->activeTarget()) return; m_listWidgets[RUN]->addProjectConfiguration(rc); if (update) updateRunListVisible(); } void MiniProjectTargetSelector::removedRunConfiguration(RunConfiguration *rc, bool update) { if (!m_project || rc->target() != m_project->activeTarget()) return; m_listWidgets[RUN]->removeProjectConfiguration(rc); if (update) updateRunListVisible(); } void MiniProjectTargetSelector::updateProjectListVisible() { int count = ProjectManager::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; for (Project *p : ProjectManager::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; for (Project *p : ProjectManager::projects()) { const QList targets = p->targets(); for (Target *t : 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; for (Project *p : ProjectManager::projects()) { const QList targets = p->targets(); for (Target *t : 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; for (Project *p : ProjectManager::projects()) { const QList targets = p->targets(); for (Target *t : 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(nullptr); } if (project) { QList list; const QList targets = project->targets(); for (Target *t : targets) list.append(t); m_listWidgets[TARGET]->setProjectConfigurations(list, project->activeTarget()); } else { m_listWidgets[TARGET]->setProjectConfigurations(QList(), nullptr); } updateActionAndSummary(); } void MiniProjectTargetSelector::activeTargetChanged(Target *target) { if (m_target) { disconnect(m_target, &Target::kitChanged, 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() : nullptr); 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; for (BuildConfiguration *bc : target->buildConfigurations()) bl.append(bc); m_listWidgets[BUILD]->setProjectConfigurations(bl, target->activeBuildConfiguration()); QList dl; for (DeployConfiguration *dc : target->deployConfigurations()) dl.append(dc); m_listWidgets[DEPLOY]->setProjectConfigurations(dl, target->activeDeployConfiguration()); QList rl; for (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, &Target::kitChanged, 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(), nullptr); m_listWidgets[DEPLOY]->setProjectConfigurations(QList(), nullptr); m_listWidgets[RUN]->setProjectConfigurations(QList(), nullptr); m_buildConfiguration = nullptr; m_deployConfiguration = nullptr; m_runConfiguration = nullptr; } 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(); 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 (auto *lw = qobject_cast(focusWidget())) { if (lw->currentIndex().row() < lw->model()->rowCount() -1) lw->setCurrentIndex(lw->model()->index(lw->currentIndex().row() + 1, 0)); else lw->setCurrentIndex(lw->model()->index(0, 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 = ProjectManager::startupProject(); if (project) { projectName = project->displayName(); for (Project *p : ProjectManager::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->expandedDisplayName(); targetToolTipText = target->overlayIconToolTip(); targetIcon = createCenteredIcon(target->icon(), target->overlayIcon()); } } m_projectAction->setProperty("heading", projectName); if (project && project->needsConfiguration()) m_projectAction->setProperty("subtitle", Tr::tr("Unconfigured")); else m_projectAction->setProperty("subtitle", buildConfig); m_projectAction->setIcon(targetIcon); QStringList lines; lines << Tr::tr("Project: %1").arg(projectName); if (!fileName.isEmpty()) lines << Tr::tr("Path: %1").arg(fileName); if (!targetName.isEmpty()) lines << Tr::tr("Kit: %1").arg(targetName); if (!buildConfig.isEmpty()) lines << Tr::tr("Build: %1").arg(buildConfig); if (!deployConfig.isEmpty()) lines << Tr::tr("Deploy: %1").arg(deployConfig); if (!runConfig.isEmpty()) lines << Tr::tr("Run: %1").arg(runConfig); if (!targetToolTipText.isEmpty()) lines << Tr::tr("%1").arg(targetToolTipText); QString toolTip = QString("%1") .arg(lines.join(QLatin1String("
"))); m_projectAction->setToolTip(toolTip); updateSummary(); } void MiniProjectTargetSelector::updateSummary() { QString summary; if (Project *startupProject = ProjectManager::startupProject()) { if (!m_projectListWidget->isVisibleTo(this)) summary.append(Tr::tr("Project: %1
").arg(startupProject->displayName())); if (Target *activeTarget = startupProject->activeTarget()) { if (!m_listWidgets[TARGET]->isVisibleTo(this)) summary.append(Tr::tr("Kit: %1
").arg( activeTarget->displayName())); if (!m_listWidgets[BUILD]->isVisibleTo(this) && activeTarget->activeBuildConfiguration()) summary.append(Tr::tr("Build: %1
").arg( activeTarget->activeBuildConfiguration()->displayName())); if (!m_listWidgets[DEPLOY]->isVisibleTo(this) && activeTarget->activeDeployConfiguration()) summary.append(Tr::tr("Deploy: %1
").arg( activeTarget->activeDeployConfiguration()->displayName())); if (!m_listWidgets[RUN]->isVisibleTo(this) && activeTarget->activeRunConfiguration()) summary.append(Tr::tr("Run: %1
").arg( activeTarget->activeRunConfiguration()->expandedDisplayName())); } else if (startupProject->needsConfiguration()) { summary = Tr::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("
")); } } if (summary != m_summaryLabel->text()) { m_summaryLabel->setText(summary); doLayout(); } } void MiniProjectTargetSelector::paintEvent(QPaintEvent *) { QPainter painter(this); painter.fillRect(rect(), StyleHelper::baseColor()); 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 const QImage image(":/projectexplorer/images/targetpanel_bottom.png"); StyleHelper::drawCornerImage(image, &painter, bottomRect, 1, 1, 1, 1); } } void MiniProjectTargetSelector::switchToProjectsMode() { Core::ModeManager::activateMode(Constants::MODE_SESSION); hide(); } } // ProjectExplorer::Internal #include