annotate filestatuswidget.cpp @ 202:3d4291d4226c

* Treat command failure on hg heads as a success, so that following actions happen properly (hg heads fails on empty repo) * Hide no-modifications-yet label when widget is first created, show it only if it would actually be true * Set directory name to window title * Various textual fixes
author Chris Cannam
date Tue, 04 Jan 2011 14:31:22 +0000
parents f16fe0db11f3
children c77c4d00a4fe
rev   line source
Chris@88 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@88 2
Chris@88 3 /*
Chris@88 4 EasyMercurial
Chris@88 5
Chris@88 6 Based on HgExplorer by Jari Korhonen
Chris@88 7 Copyright (c) 2010 Jari Korhonen
Chris@88 8 Copyright (c) 2010 Chris Cannam
Chris@88 9 Copyright (c) 2010 Queen Mary, University of London
Chris@88 10
Chris@88 11 This program is free software; you can redistribute it and/or
Chris@88 12 modify it under the terms of the GNU General Public License as
Chris@88 13 published by the Free Software Foundation; either version 2 of the
Chris@88 14 License, or (at your option) any later version. See the file
Chris@88 15 COPYING included with this distribution for more information.
Chris@88 16 */
Chris@88 17
Chris@88 18 #include "filestatuswidget.h"
Chris@115 19 #include "debug.h"
Chris@115 20 #include "multichoicedialog.h"
Chris@186 21 #include "clickablelabel.h"
Chris@88 22
Chris@88 23 #include <QLabel>
Chris@88 24 #include <QListWidget>
Chris@88 25 #include <QGridLayout>
Chris@93 26 #include <QFileInfo>
Chris@93 27 #include <QApplication>
Chris@93 28 #include <QDateTime>
Chris@115 29 #include <QPushButton>
Chris@186 30 #include <QToolButton>
Chris@186 31 #include <QDir>
Chris@186 32 #include <QProcess>
Chris@199 33 #include <QCheckBox>
Chris@88 34
Chris@88 35 FileStatusWidget::FileStatusWidget(QWidget *parent) :
Chris@93 36 QWidget(parent),
Chris@93 37 m_dateReference(0)
Chris@88 38 {
Chris@88 39 QGridLayout *layout = new QGridLayout;
Chris@88 40 setLayout(layout);
Chris@88 41
Chris@88 42 int row = 0;
Chris@88 43
Chris@115 44 layout->addItem(new QSpacerItem(1, 1), row, 0);
Chris@115 45
Chris@115 46 ++row;
Chris@88 47 layout->addWidget(new QLabel(tr("Local:")), row, 0);
Chris@186 48
Chris@186 49 m_openButton = new ClickableLabel;
Chris@186 50 QFont f(m_openButton->font());
Chris@89 51 f.setBold(true);
Chris@186 52 m_openButton->setFont(f);
Chris@186 53 m_openButton->setMouseUnderline(true);
Chris@186 54 connect(m_openButton, SIGNAL(clicked()), this, SLOT(openButtonClicked()));
Chris@187 55 layout->addWidget(m_openButton, row, 1, 1, 2, Qt::AlignLeft);
Chris@88 56
Chris@88 57 ++row;
Chris@88 58 layout->addWidget(new QLabel(tr("Remote:")), row, 0);
Chris@88 59 m_remoteURLLabel = new QLabel;
Chris@186 60 layout->addWidget(m_remoteURLLabel, row, 1, 1, 2);
Chris@88 61
Chris@106 62 ++row;
Chris@115 63 layout->addWidget(new QLabel(tr("State:")), row, 0);
Chris@115 64 m_stateLabel = new QLabel;
Chris@115 65 layout->addWidget(m_stateLabel, row, 1, 1, 2);
Chris@106 66
Chris@89 67 layout->setColumnStretch(1, 20);
Chris@88 68
Chris@115 69 layout->addWidget(new QLabel("<qt><hr></qt>"), ++row, 0, 1, 3);
Chris@115 70
Chris@115 71 ++row;
Chris@115 72 m_noModificationsLabel = new QLabel
Chris@202 73 (tr("<qt>This area will list files in your working folder that you have changed.<br>At the moment you have no uncommitted changes.<br><br>To see changes made to the repository previously,<br>switch to the History tab.<br><br>%1</qt>")
Chris@202 74 #if defined Q_OS_MAC
Chris@202 75 .arg(tr("To open the working folder in Finder,<br>click on the &ldquo;Local&rdquo; folder path shown above."))
Chris@202 76 #elif defined Q_OS_WIN32
Chris@202 77 .arg(tr("To open the working folder in Windows Explorer,<br>click on the &ldquo;Local&rdquo; folder path shown above."))
Chris@202 78 #else
Chris@202 79 .arg(tr("To open the working folder in your system file manager,<br>click the &ldquo;Local&rdquo; folder path shown above."))
Chris@202 80 #endif
Chris@202 81 );
Chris@115 82 layout->addWidget(m_noModificationsLabel, row, 1, 1, 2);
Chris@202 83 m_noModificationsLabel->hide();
Chris@106 84
Chris@100 85 m_simpleLabels[FileStates::Clean] = tr("Unmodified:");
Chris@100 86 m_simpleLabels[FileStates::Modified] = tr("Modified:");
Chris@100 87 m_simpleLabels[FileStates::Added] = tr("Added:");
Chris@100 88 m_simpleLabels[FileStates::Removed] = tr("Removed:");
Chris@100 89 m_simpleLabels[FileStates::Missing] = tr("Missing:");
Chris@163 90 m_simpleLabels[FileStates::InConflict] = tr("In Conflict:");
Chris@100 91 m_simpleLabels[FileStates::Unknown] = tr("Untracked:");
Chris@199 92 m_simpleLabels[FileStates::Ignored] = tr("Ignored:");
Chris@99 93
Chris@100 94 m_descriptions[FileStates::Clean] = tr("You have not changed these files.");
Chris@100 95 m_descriptions[FileStates::Modified] = tr("You have changed these files since you last committed them.");
Chris@100 96 m_descriptions[FileStates::Added] = tr("These files will be added to version control next time you commit.");
Chris@100 97 m_descriptions[FileStates::Removed] = tr("These files will be removed from version control next time you commit.<br>"
Chris@100 98 "They will not be deleted from the local folder.");
Chris@109 99 m_descriptions[FileStates::Missing] = tr("These files are recorded in the version control, but absent from your working folder.<br>"
Chris@153 100 "If you intended to delete them, select them and use Remove to tell the version control system about it.<br>"
Chris@153 101 "If you deleted them by accident, select them and use Revert to restore their previous contents.");
Chris@163 102 m_descriptions[FileStates::InConflict] = tr("These files are unresolved following an incomplete merge.<br>Select a file and use Merge to try to resolve the merge again.");
Chris@100 103 m_descriptions[FileStates::Unknown] = tr("These files are in your working folder but are not under version control.<br>"
Chris@186 104 // "Select a file and use Add to place it under version control or Ignore to remove it from this list.");
Chris@186 105 "Select a file and use Add to place it under version control.");
Chris@199 106 m_descriptions[FileStates::Ignored] = tr("These files have names that match entries in the working folder's .hgignore file,<br>"
Chris@199 107 "and so will be ignored by the version control system.");
Chris@100 108
Chris@118 109 m_highlightExplanation = tr("Files highlighted <font color=#d40000>in red</font> "
Chris@100 110 "have appeared since your most recent commit or update.");
Chris@89 111
Chris@94 112 for (int i = int(FileStates::FirstState);
Chris@163 113 i <= int(FileStates::LastState); ++i) {
Chris@89 114
Chris@94 115 FileStates::State s = FileStates::State(i);
Chris@89 116
Chris@89 117 QWidget *box = new QWidget;
Chris@89 118 QGridLayout *boxlayout = new QGridLayout;
Chris@99 119 boxlayout->setMargin(0);
Chris@89 120 box->setLayout(boxlayout);
Chris@89 121
Chris@106 122 boxlayout->addItem(new QSpacerItem(5, 5), 0, 0);
Chris@101 123
Chris@101 124 boxlayout->addWidget(new QLabel(labelFor(s)), 1, 0);
Chris@89 125
Chris@94 126 QListWidget *w = new QListWidget;
Chris@94 127 m_stateListMap[s] = w;
Chris@94 128 w->setSelectionMode(QListWidget::ExtendedSelection);
Chris@101 129 boxlayout->addWidget(w, 2, 0);
Chris@89 130
Chris@95 131 connect(w, SIGNAL(itemSelectionChanged()),
Chris@95 132 this, SLOT(itemSelectionChanged()));
Chris@95 133
Chris@115 134 layout->addWidget(box, ++row, 0, 1, 3);
Chris@89 135 box->hide();
Chris@89 136 }
Chris@89 137
Chris@89 138 layout->setRowStretch(++row, 20);
Chris@186 139
Chris@199 140 layout->addItem(new QSpacerItem(1, 1), ++row, 0);
Chris@199 141
Chris@199 142 m_showAllFiles = new QCheckBox(tr("Show all files"), this);
Chris@199 143 layout->addWidget(m_showAllFiles, ++row, 0, 1, 3, Qt::AlignLeft);
Chris@199 144 connect(m_showAllFiles, SIGNAL(toggled(bool)),
Chris@199 145 this, SIGNAL(showAllChanged(bool)));
Chris@88 146 }
Chris@88 147
Chris@93 148 FileStatusWidget::~FileStatusWidget()
Chris@93 149 {
Chris@93 150 delete m_dateReference;
Chris@93 151 }
Chris@93 152
Chris@186 153 void FileStatusWidget::openButtonClicked()
Chris@186 154 {
Chris@186 155 QDir d(m_localPath);
Chris@186 156 if (d.exists()) {
Chris@186 157 QStringList args;
Chris@186 158 args << d.canonicalPath();
Chris@186 159 QProcess::execute(
Chris@186 160 #if defined Q_OS_WIN32
Chris@186 161 "c:/windows/explorer.exe",
Chris@186 162 #elif defined Q_OS_MAC
Chris@186 163 "/usr/bin/open",
Chris@186 164 #else
Chris@186 165 "/usr/bin/xdg-open",
Chris@186 166 #endif
Chris@186 167 args);
Chris@186 168 }
Chris@186 169 }
Chris@186 170
Chris@100 171 QString FileStatusWidget::labelFor(FileStates::State s, bool addHighlightExplanation)
Chris@100 172 {
Chris@100 173 if (addHighlightExplanation) {
Chris@100 174 return QString("<qt><b>%1</b><br>%2<br>%3</qt>")
Chris@100 175 .arg(m_simpleLabels[s])
Chris@100 176 .arg(m_descriptions[s])
Chris@100 177 .arg(m_highlightExplanation);
Chris@100 178 } else {
Chris@100 179 return QString("<qt><b>%1</b><br>%2</qt>")
Chris@100 180 .arg(m_simpleLabels[s])
Chris@100 181 .arg(m_descriptions[s]);
Chris@100 182 }
Chris@100 183 }
Chris@100 184
Chris@95 185 void FileStatusWidget::itemSelectionChanged()
Chris@95 186 {
Chris@135 187 DEBUG << "FileStatusWidget::itemSelectionChanged" << endl;
Chris@135 188
Chris@135 189 QListWidget *list = qobject_cast<QListWidget *>(sender());
Chris@135 190
Chris@135 191 if (list) {
Chris@135 192 foreach (QListWidget *w, m_stateListMap) {
Chris@135 193 if (w != list) {
Chris@135 194 w->blockSignals(true);
Chris@135 195 w->clearSelection();
Chris@135 196 w->blockSignals(false);
Chris@135 197 }
Chris@135 198 }
Chris@135 199 }
Chris@135 200
Chris@95 201 m_selectedFiles.clear();
Chris@95 202
Chris@95 203 foreach (QListWidget *w, m_stateListMap) {
Chris@95 204 QList<QListWidgetItem *> sel = w->selectedItems();
Chris@95 205 foreach (QListWidgetItem *i, sel) {
Chris@95 206 m_selectedFiles.push_back(i->text());
Chris@95 207 DEBUG << "file " << i->text() << " is selected" << endl;
Chris@95 208 }
Chris@95 209 }
Chris@95 210
Chris@95 211 emit selectionChanged();
Chris@95 212 }
Chris@95 213
Chris@94 214 void FileStatusWidget::clearSelections()
Chris@94 215 {
Chris@95 216 m_selectedFiles.clear();
Chris@94 217 foreach (QListWidget *w, m_stateListMap) {
Chris@94 218 w->clearSelection();
Chris@94 219 }
Chris@94 220 }
Chris@94 221
Chris@95 222 bool FileStatusWidget::haveChangesToCommit() const
Chris@95 223 {
Chris@95 224 return !m_fileStates.added().empty() ||
Chris@95 225 !m_fileStates.removed().empty() ||
Chris@95 226 !m_fileStates.modified().empty();
Chris@95 227 }
Chris@95 228
Chris@95 229 bool FileStatusWidget::haveSelection() const
Chris@95 230 {
Chris@95 231 return !m_selectedFiles.empty();
Chris@95 232 }
Chris@95 233
Chris@95 234 QStringList FileStatusWidget::getAllSelectedFiles() const
Chris@95 235 {
Chris@95 236 return m_selectedFiles;
Chris@95 237 }
Chris@95 238
Chris@95 239 QStringList FileStatusWidget::getSelectedCommittableFiles() const
Chris@95 240 {
Chris@95 241 QStringList files;
Chris@95 242 foreach (QString f, m_selectedFiles) {
Chris@95 243 switch (m_fileStates.getStateOfFile(f)) {
Chris@95 244 case FileStates::Added:
Chris@95 245 case FileStates::Modified:
Chris@95 246 case FileStates::Removed:
Chris@95 247 files.push_back(f);
Chris@95 248 break;
Chris@95 249 default: break;
Chris@95 250 }
Chris@95 251 }
Chris@95 252 return files;
Chris@95 253 }
Chris@95 254
Chris@103 255 QStringList FileStatusWidget::getAllCommittableFiles() const
Chris@103 256 {
Chris@103 257 QStringList files;
Chris@103 258 files << m_fileStates.getFilesInState(FileStates::Modified);
Chris@103 259 files << m_fileStates.getFilesInState(FileStates::Added);
Chris@103 260 files << m_fileStates.getFilesInState(FileStates::Removed);
Chris@103 261 return files;
Chris@103 262 }
Chris@103 263
Chris@109 264 QStringList FileStatusWidget::getSelectedRevertableFiles() const
Chris@109 265 {
Chris@109 266 QStringList files;
Chris@109 267 foreach (QString f, m_selectedFiles) {
Chris@109 268 switch (m_fileStates.getStateOfFile(f)) {
Chris@109 269 case FileStates::Added:
Chris@109 270 case FileStates::Modified:
Chris@109 271 case FileStates::Removed:
Chris@109 272 case FileStates::Missing:
Chris@163 273 case FileStates::InConflict:
Chris@109 274 files.push_back(f);
Chris@109 275 break;
Chris@109 276 default: break;
Chris@109 277 }
Chris@109 278 }
Chris@109 279 return files;
Chris@109 280 }
Chris@109 281
Chris@109 282 QStringList FileStatusWidget::getAllRevertableFiles() const
Chris@109 283 {
Chris@109 284 QStringList files;
Chris@109 285 files << m_fileStates.getFilesInState(FileStates::Modified);
Chris@109 286 files << m_fileStates.getFilesInState(FileStates::Added);
Chris@109 287 files << m_fileStates.getFilesInState(FileStates::Removed);
Chris@109 288 files << m_fileStates.getFilesInState(FileStates::Missing);
Chris@163 289 files << m_fileStates.getFilesInState(FileStates::InConflict);
Chris@163 290 return files;
Chris@163 291 }
Chris@163 292
Chris@163 293 QStringList FileStatusWidget::getSelectedUnresolvedFiles() const
Chris@163 294 {
Chris@163 295 QStringList files;
Chris@163 296 foreach (QString f, m_selectedFiles) {
Chris@163 297 switch (m_fileStates.getStateOfFile(f)) {
Chris@163 298 case FileStates::InConflict:
Chris@163 299 files.push_back(f);
Chris@163 300 break;
Chris@163 301 default: break;
Chris@163 302 }
Chris@163 303 }
Chris@163 304 return files;
Chris@163 305 }
Chris@163 306
Chris@163 307 QStringList FileStatusWidget::getAllUnresolvedFiles() const
Chris@163 308 {
Chris@163 309 QStringList files;
Chris@163 310 files << m_fileStates.getFilesInState(FileStates::InConflict);
Chris@109 311 return files;
Chris@109 312 }
Chris@109 313
Chris@95 314 QStringList FileStatusWidget::getSelectedAddableFiles() const
Chris@95 315 {
Chris@95 316 QStringList files;
Chris@95 317 foreach (QString f, m_selectedFiles) {
Chris@95 318 switch (m_fileStates.getStateOfFile(f)) {
Chris@95 319 case FileStates::Unknown:
Chris@95 320 case FileStates::Removed:
Chris@95 321 files.push_back(f);
Chris@95 322 break;
Chris@95 323 default: break;
Chris@95 324 }
Chris@95 325 }
Chris@95 326 return files;
Chris@95 327 }
Chris@95 328
Chris@103 329 QStringList FileStatusWidget::getAllAddableFiles() const
Chris@103 330 {
Chris@103 331 QStringList files;
Chris@103 332 files << m_fileStates.getFilesInState(FileStates::Removed);
Chris@103 333 files << m_fileStates.getFilesInState(FileStates::Unknown);
Chris@103 334 return files;
Chris@103 335 }
Chris@103 336
Chris@95 337 QStringList FileStatusWidget::getSelectedRemovableFiles() const
Chris@95 338 {
Chris@95 339 QStringList files;
Chris@95 340 foreach (QString f, m_selectedFiles) {
Chris@95 341 switch (m_fileStates.getStateOfFile(f)) {
Chris@95 342 case FileStates::Clean:
Chris@95 343 case FileStates::Added:
Chris@95 344 case FileStates::Modified:
Chris@95 345 case FileStates::Missing:
Chris@163 346 case FileStates::InConflict:
Chris@95 347 files.push_back(f);
Chris@95 348 break;
Chris@95 349 default: break;
Chris@95 350 }
Chris@95 351 }
Chris@95 352 return files;
Chris@95 353 }
Chris@95 354
Chris@103 355 QStringList FileStatusWidget::getAllRemovableFiles() const
Chris@103 356 {
Chris@103 357 QStringList files;
Chris@103 358 files << m_fileStates.getFilesInState(FileStates::Clean);
Chris@103 359 files << m_fileStates.getFilesInState(FileStates::Added);
Chris@103 360 files << m_fileStates.getFilesInState(FileStates::Modified);
Chris@103 361 files << m_fileStates.getFilesInState(FileStates::Missing);
Chris@163 362 files << m_fileStates.getFilesInState(FileStates::InConflict);
Chris@103 363 return files;
Chris@103 364 }
Chris@103 365
Chris@88 366 void
Chris@88 367 FileStatusWidget::setLocalPath(QString p)
Chris@88 368 {
Chris@88 369 m_localPath = p;
Chris@186 370 m_openButton->setText(p);
Chris@93 371 delete m_dateReference;
Chris@93 372 m_dateReference = new QFileInfo(p + "/.hg/dirstate");
Chris@93 373 if (!m_dateReference->exists() ||
Chris@93 374 !m_dateReference->isFile() ||
Chris@93 375 !m_dateReference->isReadable()) {
Chris@93 376 DEBUG << "FileStatusWidget::setLocalPath: date reference file "
Chris@93 377 << m_dateReference->absoluteFilePath()
Chris@93 378 << " does not exist, is not a file, or cannot be read"
Chris@93 379 << endl;
Chris@93 380 delete m_dateReference;
Chris@93 381 m_dateReference = 0;
Chris@93 382 }
Chris@186 383 m_openButton->setEnabled(QDir(m_localPath).exists());
Chris@88 384 }
Chris@88 385
Chris@88 386 void
Chris@88 387 FileStatusWidget::setRemoteURL(QString r)
Chris@88 388 {
Chris@88 389 m_remoteURL = r;
Chris@88 390 m_remoteURLLabel->setText(r);
Chris@88 391 }
Chris@88 392
Chris@88 393 void
Chris@92 394 FileStatusWidget::setFileStates(FileStates p)
Chris@88 395 {
Chris@92 396 m_fileStates = p;
Chris@88 397 updateWidgets();
Chris@88 398 }
Chris@88 399
Chris@88 400 void
Chris@115 401 FileStatusWidget::setState(QString b)
Chris@106 402 {
Chris@115 403 m_state = b;
Chris@115 404 updateStateLabel();
Chris@106 405 }
Chris@106 406
Chris@106 407 void
Chris@88 408 FileStatusWidget::updateWidgets()
Chris@88 409 {
Chris@95 410 QDateTime lastInteractionTime;
Chris@95 411 if (m_dateReference) {
Chris@95 412 lastInteractionTime = m_dateReference->lastModified();
Chris@95 413 DEBUG << "reference time: " << lastInteractionTime << endl;
Chris@95 414 }
Chris@95 415
Chris@95 416 QSet<QString> selectedFiles;
Chris@95 417 foreach (QString f, m_selectedFiles) selectedFiles.insert(f);
Chris@95 418
Chris@115 419 bool haveAnything = false;
Chris@115 420
Chris@94 421 foreach (FileStates::State s, m_stateListMap.keys()) {
Chris@95 422
Chris@94 423 QListWidget *w = m_stateListMap[s];
Chris@94 424 w->clear();
Chris@95 425 QStringList files = m_fileStates.getFilesInState(s);
Chris@93 426
Chris@95 427 QStringList highPriority, lowPriority;
Chris@95 428
Chris@95 429 foreach (QString file, files) {
Chris@95 430
Chris@95 431 bool highlighted = false;
Chris@95 432
Chris@95 433 if (s == FileStates::Unknown) {
Chris@95 434 // We want to highlight untracked files that have appeared
Chris@95 435 // since the last interaction with the repo
Chris@95 436 QString fn(m_localPath + "/" + file);
Chris@95 437 DEBUG << "comparing with " << fn << endl;
Chris@95 438 QFileInfo fi(fn);
Chris@100 439 if (fi.exists() && fi.created() > lastInteractionTime) {
Chris@95 440 DEBUG << "file " << fn << " is newer (" << fi.lastModified()
Chris@95 441 << ") than reference" << endl;
Chris@95 442 highlighted = true;
Chris@95 443 }
Chris@95 444 }
Chris@95 445
Chris@95 446 if (highlighted) {
Chris@95 447 highPriority.push_back(file);
Chris@95 448 } else {
Chris@95 449 lowPriority.push_back(file);
Chris@93 450 }
Chris@93 451 }
Chris@95 452
Chris@95 453 foreach (QString file, highPriority) {
Chris@95 454 QListWidgetItem *item = new QListWidgetItem(file);
Chris@95 455 w->addItem(item);
Chris@118 456 item->setForeground(QColor("#d40000")); //!!! and a nice gold star
Chris@95 457 item->setSelected(selectedFiles.contains(file));
Chris@95 458 }
Chris@95 459
Chris@95 460 foreach (QString file, lowPriority) {
Chris@95 461 QListWidgetItem *item = new QListWidgetItem(file);
Chris@95 462 w->addItem(item);
Chris@95 463 item->setSelected(selectedFiles.contains(file));
Chris@95 464 }
Chris@95 465
Chris@100 466 setLabelFor(w, s, !highPriority.empty());
Chris@100 467
Chris@115 468 if (files.empty()) {
Chris@115 469 w->parentWidget()->hide();
Chris@115 470 } else {
Chris@115 471 w->parentWidget()->show();
Chris@115 472 haveAnything = true;
Chris@115 473 }
Chris@93 474 }
Chris@115 475
Chris@115 476 m_noModificationsLabel->setVisible(!haveAnything);
Chris@115 477
Chris@115 478 updateStateLabel();
Chris@88 479 }
Chris@88 480
Chris@100 481 void FileStatusWidget::setLabelFor(QWidget *w, FileStates::State s, bool addHighlight)
Chris@100 482 {
Chris@100 483 QString text = labelFor(s, addHighlight);
Chris@100 484 QWidget *p = w->parentWidget();
Chris@100 485 QList<QLabel *> ql = p->findChildren<QLabel *>();
Chris@100 486 if (!ql.empty()) ql[0]->setText(text);
Chris@100 487 }
Chris@115 488
Chris@115 489 void FileStatusWidget::updateStateLabel()
Chris@115 490 {
Chris@115 491 m_stateLabel->setText(m_state);
Chris@115 492 }