VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/ui/VBoxVMLogViewer.ui.h@ 3829

Last change on this file since 3829 was 3818, checked in by vboxsync, 18 years ago

2124: VirtualBox LogViewer search mechanism:

LogViewer search mechanism improvements/bugfixes done:

  1. The [Save] button was the default one but was not focused by default (fixed).
  2. The search shortcuts are now (F3/Shift-F3) work even if the search-panel is closed.
  3. Own "ensure-found-string-is-visible" mechanism implemented.
  4. Fixed bug when the clean (without any log files) logger become enabled after "Refresh".
  • Property svn:eol-style set to native
  • Property svn:keywords set to Date Revision Author Id
File size: 22.5 KB
Line 
1/**
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * "Virtual Log Viewer" dialog UI include (Qt Designer)
5 */
6
7/*
8 * Copyright (C) 2006 innotek GmbH
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License as published by the Free Software Foundation,
14 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
15 * distribution. VirtualBox OSE is distributed in the hope that it will
16 * be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * If you received this file as part of a commercial VirtualBox
19 * distribution, then only the terms of your commercial VirtualBox
20 * license agreement apply instead of the previous paragraph.
21 */
22
23/****************************************************************************
24** ui.h extension file, included from the uic-generated form implementation.
25**
26** If you wish to add, delete or rename functions or slots use
27** Qt Designer which will update this file, preserving your code. Create an
28** init() function in place of a constructor, and a destroy() function in
29** place of a destructor.
30*****************************************************************************/
31
32
33class VBoxLogSearchPanel : public QWidget
34{
35 Q_OBJECT
36
37public:
38
39 VBoxLogSearchPanel (QWidget *aParent,
40 VBoxVMLogViewer *aViewer,
41 const char *aName)
42 : QWidget (aParent, aName)
43 , mViewer (aViewer)
44 , mButtonClose (0)
45 , mSearchName (0), mSearchString (0)
46 , mButtonPrev (0), mButtonNext (0)
47 , mCaseSensitive (0)
48 , mWarningSpacer (0), mWarningIcon (0), mWarningString (0)
49 {
50 mButtonClose = new QToolButton (this);
51 mButtonClose->setAutoRaise (true);
52 mButtonClose->setFocusPolicy (QWidget::TabFocus);
53 mButtonClose->setAccel (QKeySequence (Qt::Key_Escape));
54 connect (mButtonClose, SIGNAL (clicked()), this, SLOT (hide()));
55 mButtonClose->setIconSet (VBoxGlobal::iconSet ("delete_16px.png",
56 "delete_dis_16px.png"));
57
58 mSearchName = new QLabel (this);
59 mSearchString = new QLineEdit (this);
60 mSearchString->setSizePolicy (QSizePolicy::Preferred,
61 QSizePolicy::Fixed);
62 connect (mSearchString, SIGNAL (textChanged (const QString &)),
63 this, SLOT (findCurrent (const QString &)));
64
65 mButtonNext = new QToolButton (this);
66 mButtonNext->setEnabled (false);
67 mButtonNext->setAutoRaise (true);
68 mButtonNext->setFocusPolicy (QWidget::TabFocus);
69 mButtonNext->setUsesTextLabel (true);
70 mButtonNext->setTextPosition (QToolButton::BesideIcon);
71 connect (mButtonNext, SIGNAL (clicked()), this, SLOT (findNext()));
72 mButtonNext->setIconSet (VBoxGlobal::iconSet ("list_movedown_16px.png",
73 "list_movedown_disabled_16px.png"));
74
75 mButtonPrev = new QToolButton (this);
76 mButtonPrev->setEnabled (false);
77 mButtonPrev->setAutoRaise (true);
78 mButtonPrev->setFocusPolicy (QWidget::TabFocus);
79 mButtonPrev->setUsesTextLabel (true);
80 mButtonPrev->setTextPosition (QToolButton::BesideIcon);
81 connect (mButtonPrev, SIGNAL (clicked()), this, SLOT (findBack()));
82 mButtonPrev->setIconSet (VBoxGlobal::iconSet ("list_moveup_16px.png",
83 "list_moveup_disabled_16px.png"));
84
85 mCaseSensitive = new QCheckBox (this);
86
87 mWarningSpacer = new QSpacerItem (0, 0, QSizePolicy::Fixed,
88 QSizePolicy::Minimum);
89 mWarningIcon = new QLabel (this);
90 mWarningIcon->hide();
91 QImage img = QMessageBox::standardIcon (QMessageBox::Warning).
92 convertToImage();
93 if (!img.isNull())
94 {
95 img = img.smoothScale (16, 16);
96 QPixmap pixmap;
97 pixmap.convertFromImage (img);
98 mWarningIcon->setPixmap (pixmap);
99 }
100 mWarningString = new QLabel (this);
101 mWarningString->hide();
102
103 QSpacerItem *spacer = new QSpacerItem (0, 0, QSizePolicy::Expanding,
104 QSizePolicy::Minimum);
105
106 QHBoxLayout *mainLayout = new QHBoxLayout (this, 5, 5);
107 mainLayout->addWidget (mButtonClose);
108 mainLayout->addWidget (mSearchName);
109 mainLayout->addWidget (mSearchString);
110 mainLayout->addWidget (mButtonNext);
111 mainLayout->addWidget (mButtonPrev);
112 mainLayout->addWidget (mCaseSensitive);
113 mainLayout->addItem (mWarningSpacer);
114 mainLayout->addWidget (mWarningIcon);
115 mainLayout->addWidget (mWarningString);
116 mainLayout->addItem (spacer);
117
118 setFocusProxy (mCaseSensitive);
119 qApp->installEventFilter (this);
120
121 languageChange();
122 }
123
124 void languageChange()
125 {
126 QToolTip::add (mButtonClose, tr ("Close the search panel"));
127 mSearchName->setText (tr ("Find "));
128 QToolTip::add (mSearchString, tr ("Enter a search string here"));
129 mButtonPrev->setTextLabel (tr ("&Previous"));
130 mButtonPrev->setAccel (QKeySequence (tr ("Alt+P")));
131 QToolTip::add (mButtonPrev,
132 tr ("Search for the previous occurrence of the string"));
133 mButtonNext->setTextLabel (tr ("&Next"));
134 mButtonNext->setAccel (QKeySequence (tr ("Alt+N")));
135 QToolTip::add (mButtonNext,
136 tr ("Search for the next occurrence of the string"));
137 mCaseSensitive->setText (tr ("C&ase Sensitive"));
138 QToolTip::add (mCaseSensitive,
139 tr ("Perform case sensitive search (when checked)"));
140 mWarningString->setText (tr ("String not found"));
141 }
142
143private slots:
144
145 void findNext()
146 {
147 search (true);
148 }
149
150 void findBack()
151 {
152 search (false);
153 }
154
155 void findCurrent (const QString &aSearchString)
156 {
157 mButtonNext->setEnabled (aSearchString.length());
158 mButtonPrev->setEnabled (aSearchString.length());
159 toggleWarning (!aSearchString.length());
160 if (aSearchString.length())
161 search (true, true);
162 else
163 mViewer->currentLogPage()->removeSelection();
164 }
165
166private:
167
168 void search (bool aForward, bool aStartCurrent = false)
169 {
170 QTextBrowser *browser = mViewer->currentLogPage();
171 if (!browser) return;
172
173 int startPrg = 0, endPrg = 0;
174 int startInd = 0, endInd = 0;
175 if (browser->hasSelectedText())
176 browser->getSelection (&startPrg, &startInd, &endPrg, &endInd);
177
178 bool found = false;
179 int increment = aForward ? 1 : -1;
180 int border = aForward ? browser->paragraphs() : -1;
181 int startFrom = aStartCurrent ? startInd : startInd + increment;
182 int paragraph = startFrom < 0 ? startPrg + increment : startPrg;
183 for (; paragraph != border; paragraph += increment)
184 {
185 QString text = browser->text (paragraph);
186 int res = aForward ?
187 text.find (mSearchString->text(), startFrom,
188 mCaseSensitive->isChecked()) :
189 text.findRev (mSearchString->text(), startFrom,
190 mCaseSensitive->isChecked());
191 if (res != -1)
192 {
193 found = true;
194 browser->setSelection (paragraph, res, paragraph,
195 res + mSearchString->text().length());
196 /* ensures the selected word visible */
197 int curPrg = 0, curInd = 0;
198 browser->getCursorPosition (&curPrg, &curInd);
199 QRect rect = browser->paragraphRect (curPrg);
200 QString string = browser->text (curPrg);
201 string.truncate (curInd);
202 int x = rect.x() + browser->fontMetrics().width (string);
203 int y = rect.y() + browser->pointSize() / 2;
204 browser->ensureVisible (x, y, 40, 40);
205 break;
206 }
207 startFrom = aForward ? 0 : -1;
208 }
209
210 toggleWarning (found);
211 if (!found)
212 browser->setSelection (startPrg, startInd, endPrg, endInd);
213 }
214
215 bool eventFilter (QObject *aObject, QEvent *aEvent)
216 {
217 switch (aEvent->type())
218 {
219 case QEvent::KeyPress:
220 {
221 QKeyEvent *e = static_cast<QKeyEvent*> (aEvent);
222
223 /* processing the return keypress for the mSearchString
224 * widget as the search next string action */
225 if (aObject == mSearchString &&
226 (e->state() == 0 || e->state() & Keypad) &&
227 (e->key() == Key_Enter || e->key() == Key_Return))
228 {
229 findNext();
230 return true;
231 }
232 /* processing other search next/previous shortcuts */
233 else if (e->key() == Key_F3)
234 {
235 if (e->state() == 0)
236 findNext();
237 else if (e->state() == ShiftButton)
238 findBack();
239 return true;
240 }
241 /* processing ctrl-f key combination as the shortcut to
242 * move to the search field */
243 else if (e->state() == ControlButton && e->key() == Key_F)
244 {
245 if (mViewer->currentLogPage())
246 {
247 if (isHidden()) show();
248 mSearchString->setFocus();
249 return true;
250 }
251 }
252 /* processing the alpha-numeric keys as the shortcuts to the
253 * search panel displaying and search activation */
254 else if (e->state() == 0 &&
255 e->key() >= Qt::Key_Exclam &&
256 e->key() <= Qt::Key_AsciiTilde)
257 {
258 if (mViewer->currentLogPage())
259 {
260 if (isHidden()) show();
261 mSearchString->setFocus();
262 mSearchString->insert (e->text());
263 return true;
264 }
265 }
266
267 break;
268 }
269 default:
270 break;
271 }
272 return false;
273 }
274
275 void showEvent (QShowEvent *aEvent)
276 {
277 QWidget::showEvent (aEvent);
278 mSearchString->setFocus();
279 mSearchString->selectAll();
280 }
281
282 void hideEvent (QHideEvent *aEvent)
283 {
284 if (focusData()->focusWidget()->parent() == this)
285 focusNextPrevChild (true);
286 QWidget::hideEvent (aEvent);
287 }
288
289 void toggleWarning (bool aHide)
290 {
291 mWarningSpacer->changeSize (aHide ? 0 : 16, 0, QSizePolicy::Fixed,
292 QSizePolicy::Minimum);
293 mWarningIcon->setHidden (aHide);
294 mWarningString->setHidden (aHide);
295 }
296
297 VBoxVMLogViewer *mViewer;
298 QToolButton *mButtonClose;
299 QLabel *mSearchName;
300 QLineEdit *mSearchString;
301 QToolButton *mButtonPrev;
302 QToolButton *mButtonNext;
303 QCheckBox *mCaseSensitive;
304 QSpacerItem *mWarningSpacer;
305 QLabel *mWarningIcon;
306 QLabel *mWarningString;
307};
308
309
310VBoxVMLogViewer::LogViewersMap VBoxVMLogViewer::mSelfArray = LogViewersMap();
311
312void VBoxVMLogViewer::createLogViewer (CMachine &aMachine)
313{
314 if (mSelfArray.find (aMachine.GetName()) == mSelfArray.end())
315 {
316 /* creating new log viewer if there is no one existing */
317 mSelfArray [aMachine.GetName()] = new VBoxVMLogViewer (0,
318 "VBoxVMLogViewer", WType_TopLevel | WDestructiveClose);
319 /* read new machine data for this log viewer */
320 mSelfArray [aMachine.GetName()]->setup (aMachine);
321 }
322
323 VBoxVMLogViewer *viewer = mSelfArray [aMachine.GetName()];
324 viewer->show();
325 viewer->setWindowState (viewer->windowState() & ~WindowMinimized);
326 viewer->setActiveWindow();
327}
328
329
330void VBoxVMLogViewer::init()
331{
332 /* prepare dialog to first run */
333 mFirstRun = true;
334
335 /* dialog initially is not polished */
336 mIsPolished = false;
337
338 /* search the default button */
339 mDefaultButton = searchDefaultButton();
340 qApp->installEventFilter (this);
341
342 /* setup a dialog icon */
343 setIcon (QPixmap::fromMimeSource ("show_logs_16px.png"));
344
345 /* statusbar initially disabled */
346 statusBar()->setHidden (true);
347
348 /* setup size grip */
349 mSizeGrip = new QSizeGrip (centralWidget(), "mSizeGrip");
350 mSizeGrip->resize (mSizeGrip->sizeHint());
351 mSizeGrip->stackUnder (mCloseButton);
352
353 /* logs list creation */
354 mLogList = new QTabWidget (mLogsFrame, "mLogList");
355 QVBoxLayout *logsFrameLayout = new QVBoxLayout (mLogsFrame);
356 logsFrameLayout->addWidget (mLogList);
357
358 /* search panel creation */
359 mSearchPanel = new VBoxLogSearchPanel (mLogsFrame, this,
360 "VBoxLogSearchPanel");
361 logsFrameLayout->addWidget (mSearchPanel);
362 mSearchPanel->hide();
363
364 /* fix the tab order to ensure the dialog keys are always the last */
365 setTabOrder (mSearchPanel->focusProxy(), mHelpButton);
366 setTabOrder (mHelpButton, mFindButton);
367 setTabOrder (mFindButton, mSaveButton);
368 setTabOrder (mSaveButton, mRefreshButton);
369 setTabOrder (mRefreshButton, mCloseButton);
370 setTabOrder (mCloseButton, mLogList);
371
372 /* make the [Save] button focused by default */
373 mSaveButton->setFocus();
374
375 /* applying language settings */
376 languageChangeImp();
377}
378
379
380void VBoxVMLogViewer::destroy()
381{
382 mSelfArray.erase (mMachine.GetName());
383}
384
385
386void VBoxVMLogViewer::setup (CMachine &aMachine)
387{
388 /* saving related machine */
389 mMachine = aMachine;
390
391 /* reading log files */
392 refresh();
393
394 /* loading language constants */
395 languageChangeImp();
396}
397
398
399const CMachine& VBoxVMLogViewer::machine()
400{
401 return mMachine;
402}
403
404
405void VBoxVMLogViewer::languageChangeImp()
406{
407 /* setup a dialog caption */
408 if (!mMachine.isNull())
409 setCaption (tr ("%1 - VirtualBox Log Viewer").arg (mMachine.GetName()));
410 /* translate a search panel */
411 if (mSearchPanel)
412 mSearchPanel->languageChange();
413}
414
415
416QPushButton* VBoxVMLogViewer::searchDefaultButton()
417{
418 /* this mechanism is used for searching the default dialog button
419 * and similar the same mechanism in Qt::QDialog inner source */
420 QPushButton *button = 0;
421 QObjectList *list = queryList ("QPushButton");
422 QObjectListIt it (*list);
423 while ((button = (QPushButton*)it.current()) && !button->isDefault())
424 ++ it;
425 return button;
426}
427
428
429bool VBoxVMLogViewer::eventFilter (QObject *aObject, QEvent *aEvent)
430{
431 switch (aEvent->type())
432 {
433 /* auto-default button focus-in processor used to move the "default"
434 * button property into the currently focused button */
435 case QEvent::FocusIn:
436 {
437 if (aObject->inherits ("QPushButton") &&
438 aObject->parent() == centralWidget())
439 {
440 ((QPushButton*)aObject)->setDefault (aObject != mDefaultButton);
441 if (mDefaultButton)
442 mDefaultButton->setDefault (aObject == mDefaultButton);
443 }
444 break;
445 }
446 /* auto-default button focus-out processor used to remove the "default"
447 * button property from the previously focused button */
448 case QEvent::FocusOut:
449 {
450 if (aObject->inherits ("QPushButton") &&
451 aObject->parent() == centralWidget())
452 {
453 if (mDefaultButton)
454 mDefaultButton->setDefault (aObject != mDefaultButton);
455 ((QPushButton*)aObject)->setDefault (aObject == mDefaultButton);
456 }
457 break;
458 }
459 default:
460 break;
461 }
462 return QMainWindow::eventFilter (aObject, aEvent);
463}
464
465
466bool VBoxVMLogViewer::event (QEvent *aEvent)
467{
468 bool result = QMainWindow::event (aEvent);
469 switch (aEvent->type())
470 {
471 case QEvent::LanguageChange:
472 {
473 languageChangeImp();
474 break;
475 }
476 default:
477 break;
478 }
479 return result;
480}
481
482
483void VBoxVMLogViewer::keyPressEvent (QKeyEvent *aEvent)
484{
485 if (aEvent->state() == 0 ||
486 (aEvent->state() & Keypad && aEvent->key() == Key_Enter))
487 {
488 switch (aEvent->key())
489 {
490 /* processing the return keypress for the auto-default button */
491 case Key_Enter:
492 case Key_Return:
493 {
494 QPushButton *currentDefault = searchDefaultButton();
495 if (currentDefault)
496 currentDefault->animateClick();
497 break;
498 }
499 /* processing the escape keypress as the close dialog action */
500 case Key_Escape:
501 {
502 mCloseButton->animateClick();
503 break;
504 }
505 }
506 }
507 else
508 aEvent->ignore();
509}
510
511
512void VBoxVMLogViewer::showEvent (QShowEvent *aEvent)
513{
514 QMainWindow::showEvent (aEvent);
515
516 /* one may think that QWidget::polish() is the right place to do things
517 * below, but apparently, by the time when QWidget::polish() is called,
518 * the widget style & layout are not fully done, at least the minimum
519 * size hint is not properly calculated. Since this is sometimes necessary,
520 * we provide our own "polish" implementation. */
521
522 if (mIsPolished)
523 return;
524
525 mIsPolished = true;
526
527 VBoxGlobal::centerWidget (this, parentWidget());
528}
529
530
531void VBoxVMLogViewer::resizeEvent (QResizeEvent*)
532{
533 /* adjust the size-grip location for the current resize event */
534 mSizeGrip->move (centralWidget()->rect().bottomRight() -
535 QPoint (mSizeGrip->rect().width() - 1,
536 mSizeGrip->rect().height() - 1));
537}
538
539
540void VBoxVMLogViewer::refresh()
541{
542 /* clearing old data if any */
543 mLogFilesList.clear();
544 mLogList->setEnabled (true);
545 while (mLogList->count())
546 {
547 QWidget *logPage = mLogList->page (0);
548 mLogList->removePage (logPage);
549 delete logPage;
550 }
551
552 bool isAnyLogPresent = false;
553
554 /* entering log files folder */
555 QString logFilesPath = mMachine.GetLogFolder();
556 QDir logFilesDir (logFilesPath);
557 if (logFilesDir.exists())
558 {
559 /* reading log files folder */
560 logFilesDir.setNameFilter ("*.log *.log.*");
561 QStringList logList = logFilesDir.entryList (QDir::Files);
562 if (!logList.empty()) isAnyLogPresent = true;
563 for (QStringList::Iterator it = logList.begin(); it != logList.end(); ++it)
564 loadLogFile (logFilesDir.filePath (*it));
565 }
566
567 /* create an empty log page if there are no logs at all */
568 if (!isAnyLogPresent)
569 {
570 QTextBrowser *dummyLog = createLogPage ("VBox.log");
571 dummyLog->setTextFormat (Qt::RichText);
572 dummyLog->setWordWrap (QTextEdit::WidgetWidth);
573 dummyLog->setText (tr ("<p>No log files found. Press the <b>Refresh</b> "
574 "button to rescan the log folder <nobr><b>%1</b></nobr>.</p>")
575 .arg (logFilesPath));
576 /* we don't want it to remain white */
577 dummyLog->setPaper (backgroundBrush());
578 }
579
580 /* restore previous tab-widget margin which was reseted when
581 * the tab widget's children was removed */
582 mLogList->setMargin (10);
583
584 /* show the first tab widget's page after the refresh */
585 mLogList->showPage (mLogList->page(0));
586
587 /* enable/disable save button & tab widget according log presence */
588 mFindButton->setEnabled (isAnyLogPresent);
589 mSaveButton->setEnabled (isAnyLogPresent);
590 mLogList->setEnabled (isAnyLogPresent);
591
592 if (mFirstRun)
593 {
594 /* resize the whole log-viewer to fit 80 symbols in text-browser for
595 * the first time started */
596 QTextBrowser *firstPage = static_cast <QTextBrowser *> (mLogList->page(0));
597 int fullWidth = firstPage->fontMetrics().width (QChar ('x')) * 80 +
598 firstPage->verticalScrollBar()->width() +
599 firstPage->frameWidth() * 2 +
600 5 + 4 /* left text margin + QTabWidget frame width */ +
601 mLogList->margin() * 2 +
602 centralWidget()->layout()->margin() * 2;
603 resize (fullWidth, height());
604 mFirstRun = false;
605 }
606}
607
608
609void VBoxVMLogViewer::loadLogFile (const QString &aFileName)
610{
611 /* prepare log file */
612 QFile logFile (aFileName);
613 if (!logFile.exists() || !logFile.open (IO_ReadOnly))
614 return;
615
616 /* read log file and write it into the log page */
617 QTextBrowser *logViewer = createLogPage (QFileInfo (aFileName).fileName());
618 logViewer->setText (logFile.readAll());
619
620 mLogFilesList << aFileName;
621}
622
623
624QTextBrowser* VBoxVMLogViewer::createLogPage (const QString &aName)
625{
626 QTextBrowser *logViewer = new QTextBrowser();
627 logViewer->setTextFormat (Qt::PlainText);
628 QFont font = logViewer->currentFont();
629 font.setFamily ("Courier New,courier");
630 logViewer->setFont (font);
631 logViewer->setWordWrap (QTextEdit::NoWrap);
632 logViewer->setVScrollBarMode (QScrollView::AlwaysOn);
633 mLogList->addTab (logViewer, aName);
634 return logViewer;
635}
636
637
638QTextBrowser* VBoxVMLogViewer::currentLogPage()
639{
640 return mLogList->isEnabled() ?
641 static_cast<QTextBrowser*> (mLogList->currentPage()) : 0;
642}
643
644
645void VBoxVMLogViewer::save()
646{
647 /* prepare "save as" dialog */
648 QFileInfo fileInfo (mLogFilesList [mLogList->currentPageIndex()]);
649 QDateTime dtInfo = fileInfo.lastModified();
650 QString dtString = dtInfo.toString ("yyyy-MM-dd-hh-mm-ss");
651 QString defaultFileName = QString ("%1-%2.log")
652 .arg (mMachine.GetName()).arg (dtString);
653 QString defaultFullName = QDir::convertSeparators (QDir::home().absPath() +
654 "/" + defaultFileName);
655
656 QString newFileName = QFileDialog::getSaveFileName (defaultFullName,
657 QString::null, this, "SaveLogAsDialog", tr ("Save VirtualBox Log As"));
658
659 /* save new log into the file */
660 if (!newFileName.isEmpty())
661 {
662 /* reread log data */
663 QFile oldFile (mLogFilesList [mLogList->currentPageIndex()]);
664 QFile newFile (newFileName);
665 if (!oldFile.open (IO_ReadOnly) || !newFile.open (IO_WriteOnly))
666 return;
667
668 /* save log data into the new file */
669 newFile.writeBlock (oldFile.readAll());
670 }
671}
672
673void VBoxVMLogViewer::search()
674{
675 mSearchPanel->isHidden() ? mSearchPanel->show() : mSearchPanel->hide();
676}
677
678#include "VBoxVMLogViewer.ui.moc"
679
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette