Make target selector popup size smart

Task-number: QTCREATORBUG-7705

Change-Id: Ica9a39f0557993a8bba6ec32f7373101203af4c0
Reviewed-by: Daniel Molkentin <daniel.molkentin@nokia.com>
This commit is contained in:
Daniel Teske
2012-08-24 17:20:19 +02:00
committed by Daniel Molkentin
parent 1faea8a0b3
commit 7ea3e2c2de
2 changed files with 330 additions and 133 deletions

View File

@@ -96,20 +96,21 @@ static bool projectLesserThan(Project *p1, Project *p2)
class TargetSelectorDelegate : public QItemDelegate class TargetSelectorDelegate : public QItemDelegate
{ {
public: public:
TargetSelectorDelegate(QObject *parent) : QItemDelegate(parent) { } TargetSelectorDelegate(ListWidget *parent) : QItemDelegate(parent), m_listWidget(parent) { }
private: private:
QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const; QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const;
void paint(QPainter *painter, void paint(QPainter *painter,
const QStyleOptionViewItem &option, const QStyleOptionViewItem &option,
const QModelIndex &index) const; const QModelIndex &index) const;
mutable QImage selectionGradient; mutable QImage selectionGradient;
ListWidget *m_listWidget;
}; };
QSize TargetSelectorDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const QSize TargetSelectorDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
{ {
Q_UNUSED(option) Q_UNUSED(option)
Q_UNUSED(index) Q_UNUSED(index)
return QSize(190, 30); return QSize(m_listWidget->size().width(), 30);
} }
void TargetSelectorDelegate::paint(QPainter *painter, void TargetSelectorDelegate::paint(QPainter *painter,
@@ -153,7 +154,7 @@ void TargetSelectorDelegate::paint(QPainter *painter,
// ListWidget // ListWidget
//////// ////////
ListWidget::ListWidget(QWidget *parent) ListWidget::ListWidget(QWidget *parent)
: QListWidget(parent), m_maxCount(0) : QListWidget(parent), m_maxCount(0), m_optimalWidth(0)
{ {
setFocusPolicy(Qt::NoFocus); setFocusPolicy(Qt::NoFocus);
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
@@ -165,25 +166,6 @@ ListWidget::ListWidget(QWidget *parent)
setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
} }
QSize ListWidget::sizeHint() const
{
int height = m_maxCount * 30;
int width = 190;
// We try to keep the height of the popup equal to the actionbar
QSize size(width, height);
static QStatusBar *statusBar = Core::ICore::statusBar();
static QWidget *actionBar = Core::ICore::mainWindow()->findChild<QWidget*>(QLatin1String("actionbar"));
Q_ASSERT(actionBar);
QMargins popupMargins = window()->contentsMargins();
int alignedWithActionHeight
= actionBar->height() - statusBar->height() - (popupMargins.top() + popupMargins.bottom());
size.setHeight(qBound(alignedWithActionHeight, height, 2 * alignedWithActionHeight));
return size;
}
void ListWidget::keyPressEvent(QKeyEvent *event) void ListWidget::keyPressEvent(QKeyEvent *event)
{ {
if (event->key() == Qt::Key_Left) if (event->key() == Qt::Key_Left)
@@ -202,13 +184,32 @@ void ListWidget::keyReleaseEvent(QKeyEvent *event)
void ListWidget::setMaxCount(int maxCount) void ListWidget::setMaxCount(int maxCount)
{ {
// Note: the current assumption is that, this is not called while the listwidget is visible
// Otherwise we would need to add code to MiniProjectTargetSelector reacting to the
// updateGeometry (which then would jump ugly)
m_maxCount = maxCount; m_maxCount = maxCount;
updateGeometry(); 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 30;
}
//////// ////////
// ProjectListWidget // ProjectListWidget
//////// ////////
@@ -274,6 +275,11 @@ void ProjectListWidget::addProject(Project *project)
setCurrentItem(item); setCurrentItem(item);
} }
QFontMetrics fn(font());
int width = fn.width(project->displayName()) + padding();
if (width > optimalWidth())
setOptimalWidth(width);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -300,8 +306,17 @@ void ProjectListWidget::removeProject(Project *project)
item(otherIndex)->setText(p->displayName()); item(otherIndex)->setText(p->displayName());
} }
m_ignoreIndexChange = false; QFontMetrics fn(font());
// recheck optimal width
int width = 0;
for (int i = 0; i < count(); ++i) {
Project *p = item(i)->data(Qt::UserRole).value<Project *>();
width = qMax(fn.width(p->displayName()) + padding(), width);
}
setOptimalWidth(width);
m_ignoreIndexChange = false;
} }
void ProjectListWidget::projectDisplayNameChanged(Project *project) void ProjectListWidget::projectDisplayNameChanged(Project *project)
@@ -338,6 +353,15 @@ void ProjectListWidget::projectDisplayNameChanged(Project *project)
if (isCurrentItem) if (isCurrentItem)
setCurrentRow(pos); setCurrentRow(pos);
// recheck optimal width
QFontMetrics fn(font());
int width = 0;
for (int i = 0; i < count(); ++i) {
Project *p = item(i)->data(Qt::UserRole).value<Project *>();
width = qMax(fn.width(p->displayName()) + padding(), width);
}
setOptimalWidth(width);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -376,9 +400,16 @@ void GenericListWidget::setProjectConfigurations(const QList<ProjectConfiguratio
disconnect(p, SIGNAL(displayNameChanged()), disconnect(p, SIGNAL(displayNameChanged()),
this, SLOT(displayNameChanged())); this, SLOT(displayNameChanged()));
} }
foreach (ProjectConfiguration *pc, list)
QFontMetrics fn(font());
int width = 0;
foreach (ProjectConfiguration *pc, list) {
addProjectConfiguration(pc); addProjectConfiguration(pc);
width = qMax(width, fn.width(pc->displayName()) + padding());
}
setOptimalWidth(width);
setActiveProjectConfiguration(active); setActiveProjectConfiguration(active);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -408,6 +439,11 @@ void GenericListWidget::addProjectConfiguration(ProjectExplorer::ProjectConfigur
connect(pc, SIGNAL(displayNameChanged()), connect(pc, SIGNAL(displayNameChanged()),
this, SLOT(displayNameChanged())); this, SLOT(displayNameChanged()));
QFontMetrics fn(font());
int width = fn.width(pc->displayName()) + padding();
if (width > optimalWidth())
setOptimalWidth(width);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -417,6 +453,15 @@ void GenericListWidget::removeProjectConfiguration(ProjectExplorer::ProjectConfi
disconnect(pc, SIGNAL(displayNameChanged()), disconnect(pc, SIGNAL(displayNameChanged()),
this, SLOT(displayNameChanged())); this, SLOT(displayNameChanged()));
delete itemForProjectConfiguration(pc); delete itemForProjectConfiguration(pc);
QFontMetrics fn(font());
int width = 0;
for (int i = 0; i < count(); ++i) {
ProjectConfiguration *p = item(i)->data(Qt::UserRole).value<ProjectConfiguration *>();
width = qMax(width, fn.width(p->displayName()) + padding());
}
setOptimalWidth(width);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -461,6 +506,15 @@ void GenericListWidget::displayNameChanged()
insertItem(pos, lwi); insertItem(pos, lwi);
if (activeProjectConfiguration) if (activeProjectConfiguration)
setCurrentItem(itemForProjectConfiguration(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<ProjectConfiguration *>();
width = qMax(width, fn.width(p->displayName()) + padding());
}
setOptimalWidth(width);
m_ignoreIndexChange = false; m_ignoreIndexChange = false;
} }
@@ -475,16 +529,15 @@ QListWidgetItem *GenericListWidget::itemForProjectConfiguration(ProjectConfigura
return 0; return 0;
} }
QWidget *createTitleLabel(const QString &text) QWidget *MiniProjectTargetSelector::createTitleLabel(const QString &text)
{ {
Utils::StyledBar *bar = new Utils::StyledBar; Utils::StyledBar *bar = new Utils::StyledBar(this);
bar->setSingleRow(true); bar->setSingleRow(true);
QVBoxLayout *toolLayout = new QVBoxLayout(bar); QVBoxLayout *toolLayout = new QVBoxLayout(bar);
toolLayout->setMargin(0); toolLayout->setContentsMargins(6, 0, 6, 0);
toolLayout->setSpacing(0); toolLayout->setSpacing(0);
QLabel *l = new QLabel(text); QLabel *l = new QLabel(text);
l->setIndent(6);
QFont f = l->font(); QFont f = l->font();
f.setBold(true); f.setBold(true);
l->setFont(f); l->setFont(f);
@@ -496,24 +549,6 @@ QWidget *createTitleLabel(const QString &text)
return bar; return bar;
} }
class OnePixelGreyLine : public QWidget
{
public:
OnePixelGreyLine(QWidget *parent)
: QWidget(parent)
{
setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding);
setMinimumWidth(1);
setMaximumWidth(1);
}
void paintEvent(QPaintEvent *e)
{
Q_UNUSED(e);
QPainter p(this);
p.fillRect(contentsRect(), QColor(160, 160, 160, 255));
}
};
MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorAction, SessionManager *sessionManager, QWidget *parent) : MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorAction, SessionManager *sessionManager, QWidget *parent) :
QWidget(parent), m_projectAction(targetSelectorAction), m_sessionManager(sessionManager), QWidget(parent), m_projectAction(targetSelectorAction), m_sessionManager(sessionManager),
m_project(0), m_project(0),
@@ -533,10 +568,6 @@ MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorActi
targetSelectorAction->setIcon(style()->standardIcon(QStyle::SP_ComputerIcon)); targetSelectorAction->setIcon(style()->standardIcon(QStyle::SP_ComputerIcon));
targetSelectorAction->setProperty("titledAction", true); targetSelectorAction->setProperty("titledAction", true);
QGridLayout *grid = new QGridLayout(this);
grid->setMargin(0);
grid->setSpacing(0);
m_summaryLabel = new QLabel(this); m_summaryLabel = new QLabel(this);
m_summaryLabel->setMargin(3); m_summaryLabel->setMargin(3);
m_summaryLabel->setAlignment(Qt::AlignLeft | Qt::AlignTop); m_summaryLabel->setAlignment(Qt::AlignLeft | Qt::AlignTop);
@@ -544,19 +575,12 @@ MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorActi
m_summaryLabel->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); m_summaryLabel->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
m_summaryLabel->setTextInteractionFlags(m_summaryLabel->textInteractionFlags() | Qt::LinksAccessibleByMouse); m_summaryLabel->setTextInteractionFlags(m_summaryLabel->textInteractionFlags() | Qt::LinksAccessibleByMouse);
grid->addWidget(m_summaryLabel, 0, 0, 1, 2 * LAST - 1);
m_listWidgets.resize(LAST); m_listWidgets.resize(LAST);
m_titleWidgets.resize(LAST); m_titleWidgets.resize(LAST);
m_separators.resize(LAST);
m_listWidgets[PROJECT] = 0; //project is not a generic list widget m_listWidgets[PROJECT] = 0; //project is not a generic list widget
m_titleWidgets[PROJECT] = createTitleLabel(tr("Project")); m_titleWidgets[PROJECT] = createTitleLabel(tr("Project"));
grid->addWidget(m_titleWidgets[PROJECT], 1, 0, 1, 2);
m_projectListWidget = new ProjectListWidget(m_sessionManager, this); m_projectListWidget = new ProjectListWidget(m_sessionManager, this);
grid->addWidget(m_projectListWidget, 2, 0);
m_separators[PROJECT] = new OnePixelGreyLine(this);
grid->addWidget(m_separators[PROJECT], 2, 1);
QStringList titles; QStringList titles;
titles << tr("Kit") << tr("Build") titles << tr("Kit") << tr("Build")
@@ -564,11 +588,7 @@ MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorActi
for (int i = TARGET; i < LAST; ++i) { for (int i = TARGET; i < LAST; ++i) {
m_titleWidgets[i] = createTitleLabel(titles.at(i -1)); m_titleWidgets[i] = createTitleLabel(titles.at(i -1));
grid->addWidget(m_titleWidgets[i], 1, 2 * i, 1, 2);
m_listWidgets[i] = new GenericListWidget(this); m_listWidgets[i] = new GenericListWidget(this);
grid->addWidget(m_listWidgets[i], 2, 2 *i);
m_separators[i] = new OnePixelGreyLine(this);
grid->addWidget(m_separators[i], 2, 1 + 2 * i);
} }
changeStartupProject(m_sessionManager->startupProject()); changeStartupProject(m_sessionManager->startupProject());
@@ -602,6 +622,236 @@ MiniProjectTargetSelector::MiniProjectTargetSelector(QAction *targetSelectorActi
this, SLOT(setActiveRunConfiguration(ProjectExplorer::ProjectConfiguration*))); this, SLOT(setActiveRunConfiguration(ProjectExplorer::ProjectConfiguration*)));
} }
bool MiniProjectTargetSelector::event(QEvent *event)
{
if (event->type() != QEvent::LayoutRequest)
return QWidget::event(event);
doLayout(true);
return true;
}
class IndexSorter
{
public:
enum SortOrder { Less = 0, Greater = 1};
IndexSorter(QVector<int> result, SortOrder order)
: m_result(result), m_order(order)
{ }
bool operator()(int i, int j)
{ return (m_result[i] < m_result[j]) ^ bool(m_order); }
private:
QVector<int> m_result;
SortOrder m_order;
};
// does some fancy calculations to ensure proper widths for the list widgets
QVector<int> MiniProjectTargetSelector::listWidgetWidths(int minSize, int maxSize)
{
QVector<int> 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<int> indexes;
indexes.reserve(LAST);
for (int i = PROJECT; i < LAST; ++i)
if (result[i] != -1)
indexes.append(i);
IndexSorter indexSorter(result, tooSmall ? IndexSorter::Less : IndexSorter::Greater);
qSort(indexes.begin(), indexes.end(), indexSorter);
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<QWidget*>(QLatin1String("actionbar"));
Q_ASSERT(actionBar);
// 1. Calculate the summary label height
int summaryLabelY = 1;
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) {
foreach (Project *p, m_sessionManager->projects()) {
if (p->needsConfiguration()) {
visibleLineCount = 3;
break;
}
}
}
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 = actionBar->height() - statusBar->height();
int bottomMargin = 9;
int totalHeight = 0;
if (!onlySummary) {
// list widget heigth
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) {
totalHeight = height();
} else {
// Clamp the size of the listwidgets to be
// at least as high as the the sidebar button
// and at most twice as high
totalHeight = summaryLabelHeight + qBound(alignedWithActionHeight,
maxItemCount * 30 + bottomMargin + titleWidgetsHeight,
alignedWithActionHeight * 2);
}
int titleY = summaryLabelY + summaryLabelHeight;
int listY = titleY + titleWidgetsHeight;
int listHeight = totalHeight - bottomMargin - listY + 1;
// list widget widths
int minWidth = qMax(m_summaryLabel->sizeHint().width(), 250);
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<int> 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);
setFixedSize(x, totalHeight);
} else {
if (keepSize)
totalHeight = height();
else
totalHeight = qMax(summaryLabelHeight + bottomMargin, alignedWithActionHeight);
m_summaryLabel->resize(m_summaryLabel->sizeHint().width(), totalHeight - bottomMargin);
setFixedSize(m_summaryLabel->width() + 1, totalHeight); //1 extra pixel for the border
}
if (isVisibleTo(parentWidget())) {
QPoint moveTo = statusBar->mapToGlobal(QPoint(0,0));
moveTo -= QPoint(0, totalHeight);
move(moveTo);
}
}
void MiniProjectTargetSelector::setActiveTarget(ProjectExplorer::ProjectConfiguration *pc) void MiniProjectTargetSelector::setActiveTarget(ProjectExplorer::ProjectConfiguration *pc)
{ {
m_project->setActiveTarget(static_cast<Target *>(pc)); m_project->setActiveTarget(static_cast<Target *>(pc));
@@ -821,7 +1071,6 @@ void MiniProjectTargetSelector::updateProjectListVisible()
m_titleWidgets[PROJECT]->setVisible(visible); m_titleWidgets[PROJECT]->setVisible(visible);
updateSummary(); updateSummary();
updateSeparatorVisible();
} }
void MiniProjectTargetSelector::updateTargetListVisible() void MiniProjectTargetSelector::updateTargetListVisible()
@@ -835,7 +1084,6 @@ void MiniProjectTargetSelector::updateTargetListVisible()
m_listWidgets[TARGET]->setMaxCount(maxCount); m_listWidgets[TARGET]->setMaxCount(maxCount);
m_titleWidgets[TARGET]->setVisible(visible); m_titleWidgets[TARGET]->setVisible(visible);
updateSummary(); updateSummary();
updateSeparatorVisible();
} }
void MiniProjectTargetSelector::updateBuildListVisible() void MiniProjectTargetSelector::updateBuildListVisible()
@@ -850,7 +1098,6 @@ void MiniProjectTargetSelector::updateBuildListVisible()
m_listWidgets[BUILD]->setMaxCount(maxCount); m_listWidgets[BUILD]->setMaxCount(maxCount);
m_titleWidgets[BUILD]->setVisible(visible); m_titleWidgets[BUILD]->setVisible(visible);
updateSummary(); updateSummary();
updateSeparatorVisible();
} }
void MiniProjectTargetSelector::updateDeployListVisible() void MiniProjectTargetSelector::updateDeployListVisible()
@@ -865,7 +1112,6 @@ void MiniProjectTargetSelector::updateDeployListVisible()
m_listWidgets[DEPLOY]->setMaxCount(maxCount); m_listWidgets[DEPLOY]->setMaxCount(maxCount);
m_titleWidgets[DEPLOY]->setVisible(visible); m_titleWidgets[DEPLOY]->setVisible(visible);
updateSummary(); updateSummary();
updateSeparatorVisible();
} }
void MiniProjectTargetSelector::updateRunListVisible() void MiniProjectTargetSelector::updateRunListVisible()
@@ -880,7 +1126,6 @@ void MiniProjectTargetSelector::updateRunListVisible()
m_listWidgets[RUN]->setMaxCount(maxCount); m_listWidgets[RUN]->setMaxCount(maxCount);
m_titleWidgets[RUN]->setVisible(visible); m_titleWidgets[RUN]->setVisible(visible);
updateSummary(); updateSummary();
updateSeparatorVisible();
} }
void MiniProjectTargetSelector::changeStartupProject(ProjectExplorer::Project *project) void MiniProjectTargetSelector::changeStartupProject(ProjectExplorer::Project *project)
@@ -1039,28 +1284,12 @@ void MiniProjectTargetSelector::activeRunConfigurationChanged(ProjectExplorer::R
updateActionAndSummary(); updateActionAndSummary();
} }
void MiniProjectTargetSelector::updateSeparatorVisible()
{
QVector<bool> visibility;
visibility.resize(LAST);
visibility[PROJECT] = m_projectListWidget->isVisibleTo(this);
for (int i = TARGET; i < LAST; ++i)
visibility[i] = m_listWidgets[i]->isVisibleTo(this);
int lastVisible = visibility.lastIndexOf(true);
if (lastVisible != -1)
visibility[lastVisible] = false;
for (int i = PROJECT; i < LAST; ++i)
m_separators[i]->setVisible(visibility[i]);
}
void MiniProjectTargetSelector::setVisible(bool visible) void MiniProjectTargetSelector::setVisible(bool visible)
{ {
QWidget::setVisible(visible);
m_projectAction->setChecked(visible);
if (visible) { if (visible) {
QStatusBar *statusBar = Core::ICore::statusBar(); doLayout(false);
QPoint moveTo = statusBar->mapToGlobal(QPoint(0,0));
moveTo -= QPoint(0, sizeHint().height());
move(moveTo);
if (!focusWidget() || !focusWidget()->isVisibleTo(this)) { // Does the second part actually work? if (!focusWidget() || !focusWidget()->isVisibleTo(this)) { // Does the second part actually work?
if (m_projectListWidget->isVisibleTo(this)) if (m_projectListWidget->isVisibleTo(this))
m_projectListWidget->setFocus(); m_projectListWidget->setFocus();
@@ -1072,9 +1301,6 @@ void MiniProjectTargetSelector::setVisible(bool visible)
} }
} }
} }
QWidget::setVisible(visible);
m_projectAction->setChecked(visible);
} }
void MiniProjectTargetSelector::toggleVisible() void MiniProjectTargetSelector::toggleVisible()
@@ -1125,21 +1351,6 @@ void MiniProjectTargetSelector::keyReleaseEvent(QKeyEvent *ke)
QWidget::keyReleaseEvent(ke); QWidget::keyReleaseEvent(ke);
} }
QSize MiniProjectTargetSelector::sizeHint() const
{
static QStatusBar *statusBar = Core::ICore::statusBar();
static QWidget *actionBar = Core::ICore::mainWindow()->findChild<QWidget*>(QLatin1String("actionbar"));
Q_ASSERT(actionBar);
// At least the size of the actionbar
int alignedWithActionHeight
= actionBar->height() - statusBar->height();
QSize s = QWidget::sizeHint();
if (s.height() < alignedWithActionHeight)
s.setHeight(alignedWithActionHeight);
return s;
}
void MiniProjectTargetSelector::delayedHide() void MiniProjectTargetSelector::delayedHide()
{ {
QDateTime current = QDateTime::currentDateTime(); QDateTime current = QDateTime::currentDateTime();
@@ -1221,29 +1432,6 @@ void MiniProjectTargetSelector::updateActionAndSummary()
void MiniProjectTargetSelector::updateSummary() void MiniProjectTargetSelector::updateSummary()
{ {
// 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) {
m_summaryLabel->setMinimumHeight(0);
m_summaryLabel->setMaximumHeight(800);
} else {
if (visibleLineCount < 3) {
foreach (Project *p, m_sessionManager->projects()) {
if (p->needsConfiguration()) {
visibleLineCount = 3;
break;
}
}
}
int height = visibleLineCount * QFontMetrics(m_summaryLabel->font()).height() + m_summaryLabel->margin() *2;
m_summaryLabel->setMinimumHeight(height);
m_summaryLabel->setMaximumHeight(height);
}
QString summary; QString summary;
if (Project *startupProject = m_sessionManager->startupProject()) { if (Project *startupProject = m_sessionManager->startupProject()) {
if (!m_projectListWidget->isVisibleTo(this)) if (!m_projectListWidget->isVisibleTo(this))
@@ -1283,6 +1471,8 @@ void MiniProjectTargetSelector::updateSummary()
void MiniProjectTargetSelector::paintEvent(QPaintEvent *) void MiniProjectTargetSelector::paintEvent(QPaintEvent *)
{ {
QPainter painter(this); QPainter painter(this);
painter.setBrush(QBrush(QColor(160, 160, 160, 255)));
painter.drawRect(rect());
painter.setPen(Utils::StyleHelper::borderColor()); painter.setPen(Utils::StyleHelper::borderColor());
painter.drawLine(rect().topLeft(), rect().topRight()); painter.drawLine(rect().topLeft(), rect().topRight());
painter.drawLine(rect().topRight(), rect().bottomRight()); painter.drawLine(rect().topRight(), rect().bottomRight());

View File

@@ -58,13 +58,18 @@ class ListWidget : public QListWidget
Q_OBJECT Q_OBJECT
public: public:
ListWidget(QWidget *parent); ListWidget(QWidget *parent);
QSize sizeHint() const;
void keyPressEvent(QKeyEvent *event); void keyPressEvent(QKeyEvent *event);
void keyReleaseEvent(QKeyEvent *event); void keyReleaseEvent(QKeyEvent *event);
void setMaxCount(int maxCount); void setMaxCount(int maxCount);
int maxCount();
int optimalWidth() const;
void setOptimalWidth(int width);
int padding();
private: private:
int m_maxCount; int m_maxCount;
int m_optimalWidth;
}; };
class ProjectListWidget : public ListWidget class ProjectListWidget : public ListWidget
@@ -115,7 +120,7 @@ public:
void keyPressEvent(QKeyEvent *ke); void keyPressEvent(QKeyEvent *ke);
void keyReleaseEvent(QKeyEvent *ke); void keyReleaseEvent(QKeyEvent *ke);
QSize sizeHint() const; bool event(QEvent *event);
public slots: public slots:
void toggleVisible(); void toggleVisible();
void nextOrShow(); void nextOrShow();
@@ -163,10 +168,13 @@ private:
void updateDeployListVisible(); void updateDeployListVisible();
void updateRunListVisible(); void updateRunListVisible();
void updateSummary(); void updateSummary();
void updateSeparatorVisible();
void paintEvent(QPaintEvent *); void paintEvent(QPaintEvent *);
void mousePressEvent(QMouseEvent *); void mousePressEvent(QMouseEvent *);
void doLayout(bool keepSize);
QVector<int> listWidgetWidths(int minSize, int maxSize);
QWidget *createTitleLabel(const QString &text);
QAction *m_projectAction; QAction *m_projectAction;
SessionManager *m_sessionManager; SessionManager *m_sessionManager;
@@ -174,7 +182,6 @@ private:
ProjectListWidget *m_projectListWidget; ProjectListWidget *m_projectListWidget;
QVector<GenericListWidget *> m_listWidgets; QVector<GenericListWidget *> m_listWidgets;
QVector<QWidget *> m_titleWidgets; QVector<QWidget *> m_titleWidgets;
QVector<QWidget *> m_separators;
QLabel *m_summaryLabel; QLabel *m_summaryLabel;
Project *m_project; Project *m_project;