xref: /glogg/src/mainwindow.cpp (revision 7847299cf77ef989e7bddad517b120ce847445ae)
1bb02e0acSNicolas Bonnefon /*
2bb02e0acSNicolas Bonnefon  * Copyright (C) 2009, 2010, 2011, 2013 Nicolas Bonnefon and other contributors
3bb02e0acSNicolas Bonnefon  *
4bb02e0acSNicolas Bonnefon  * This file is part of glogg.
5bb02e0acSNicolas Bonnefon  *
6bb02e0acSNicolas Bonnefon  * glogg is free software: you can redistribute it and/or modify
7bb02e0acSNicolas Bonnefon  * it under the terms of the GNU General Public License as published by
8bb02e0acSNicolas Bonnefon  * the Free Software Foundation, either version 3 of the License, or
9bb02e0acSNicolas Bonnefon  * (at your option) any later version.
10bb02e0acSNicolas Bonnefon  *
11bb02e0acSNicolas Bonnefon  * glogg is distributed in the hope that it will be useful,
12bb02e0acSNicolas Bonnefon  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13bb02e0acSNicolas Bonnefon  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14bb02e0acSNicolas Bonnefon  * GNU General Public License for more details.
15bb02e0acSNicolas Bonnefon  *
16bb02e0acSNicolas Bonnefon  * You should have received a copy of the GNU General Public License
17bb02e0acSNicolas Bonnefon  * along with glogg.  If not, see <http://www.gnu.org/licenses/>.
18bb02e0acSNicolas Bonnefon  */
19bb02e0acSNicolas Bonnefon 
20bb02e0acSNicolas Bonnefon // This file implements MainWindow. It is responsible for creating and
21bb02e0acSNicolas Bonnefon // managing the menus, the toolbar, and the CrawlerWidget. It also
22bb02e0acSNicolas Bonnefon // load/save the settings on opening/closing of the app
23bb02e0acSNicolas Bonnefon 
24bb02e0acSNicolas Bonnefon #include <iostream>
25eb6a8f99SNicolas Bonnefon 
26eb6a8f99SNicolas Bonnefon #include <QAction>
27eb6a8f99SNicolas Bonnefon #include <QDesktopWidget>
28eb6a8f99SNicolas Bonnefon #include <QMenuBar>
29eb6a8f99SNicolas Bonnefon #include <QToolBar>
30eb6a8f99SNicolas Bonnefon #include <QFileInfo>
31eb6a8f99SNicolas Bonnefon #include <QFileDialog>
32eb6a8f99SNicolas Bonnefon #include <QClipboard>
33eb6a8f99SNicolas Bonnefon #include <QMessageBox>
34eb6a8f99SNicolas Bonnefon #include <QCloseEvent>
35eb6a8f99SNicolas Bonnefon #include <QDragEnterEvent>
36eb6a8f99SNicolas Bonnefon #include <QMimeData>
37eb6a8f99SNicolas Bonnefon #include <QUrl>
38bb02e0acSNicolas Bonnefon 
39bb02e0acSNicolas Bonnefon #include "log.h"
40bb02e0acSNicolas Bonnefon 
41bb02e0acSNicolas Bonnefon #include "mainwindow.h"
42bb02e0acSNicolas Bonnefon 
43bb02e0acSNicolas Bonnefon #include "sessioninfo.h"
44bb02e0acSNicolas Bonnefon #include "recentfiles.h"
45bb02e0acSNicolas Bonnefon #include "crawlerwidget.h"
46bb02e0acSNicolas Bonnefon #include "filtersdialog.h"
47bb02e0acSNicolas Bonnefon #include "optionsdialog.h"
48bb02e0acSNicolas Bonnefon #include "persistentinfo.h"
49bb02e0acSNicolas Bonnefon #include "savedsearches.h"
50bb02e0acSNicolas Bonnefon #include "menuactiontooltipbehavior.h"
51bb02e0acSNicolas Bonnefon 
520a90ca6aSNicolas Bonnefon // Returns the size in human readable format
530a90ca6aSNicolas Bonnefon static QString readableSize( qint64 size );
540a90ca6aSNicolas Bonnefon 
55d96f3f21SNicolas Bonnefon MainWindow::MainWindow( std::unique_ptr<Session> session ) :
56d96f3f21SNicolas Bonnefon     session_( std::move( session )  ),
57d96f3f21SNicolas Bonnefon     recentFiles( Persistent<RecentFiles>( "recentFiles" ) ),
58313a820fSNicolas Bonnefon     mainIcon_(),
59313a820fSNicolas Bonnefon     signalMux_()
60bb02e0acSNicolas Bonnefon {
61bb02e0acSNicolas Bonnefon     createActions();
62bb02e0acSNicolas Bonnefon     createMenus();
63bb02e0acSNicolas Bonnefon     // createContextMenu();
64bb02e0acSNicolas Bonnefon     createToolBars();
65bb02e0acSNicolas Bonnefon     // createStatusBar();
66bb02e0acSNicolas Bonnefon 
67bb02e0acSNicolas Bonnefon     setAcceptDrops( true );
68bb02e0acSNicolas Bonnefon 
69bb02e0acSNicolas Bonnefon     // Default geometry
70bb02e0acSNicolas Bonnefon     const QRect geometry = QApplication::desktop()->availableGeometry( this );
71bb02e0acSNicolas Bonnefon     setGeometry( geometry.x() + 20, geometry.y() + 40,
72bb02e0acSNicolas Bonnefon             geometry.width() - 140, geometry.height() - 140 );
73bb02e0acSNicolas Bonnefon 
74bb02e0acSNicolas Bonnefon     mainIcon_.addFile( ":/images/hicolor/16x16/glogg.png" );
75bb02e0acSNicolas Bonnefon     mainIcon_.addFile( ":/images/hicolor/24x24/glogg.png" );
76bb02e0acSNicolas Bonnefon     mainIcon_.addFile( ":/images/hicolor/32x32/glogg.png" );
77bb02e0acSNicolas Bonnefon     mainIcon_.addFile( ":/images/hicolor/48x48/glogg.png" );
78bb02e0acSNicolas Bonnefon 
79bb02e0acSNicolas Bonnefon     setWindowIcon( mainIcon_ );
80f0708ca8SNicolas Bonnefon 
810a90ca6aSNicolas Bonnefon     readSettings();
820a90ca6aSNicolas Bonnefon 
83f0708ca8SNicolas Bonnefon     crawlerWidget = nullptr;
84bb02e0acSNicolas Bonnefon }
85bb02e0acSNicolas Bonnefon 
86bb02e0acSNicolas Bonnefon void MainWindow::loadInitialFile( QString fileName )
87bb02e0acSNicolas Bonnefon {
88bb02e0acSNicolas Bonnefon     LOG(logDEBUG) << "loadInitialFile";
89bb02e0acSNicolas Bonnefon 
90bb02e0acSNicolas Bonnefon     // Is there a file passed as argument?
91bb02e0acSNicolas Bonnefon     if ( !fileName.isEmpty() )
92bb02e0acSNicolas Bonnefon         loadFile( fileName );
93bb02e0acSNicolas Bonnefon     else if ( !previousFile.isEmpty() )
94bb02e0acSNicolas Bonnefon         loadFile( previousFile );
95bb02e0acSNicolas Bonnefon }
96bb02e0acSNicolas Bonnefon 
97bb02e0acSNicolas Bonnefon //
98bb02e0acSNicolas Bonnefon // Private functions
99bb02e0acSNicolas Bonnefon //
100bb02e0acSNicolas Bonnefon 
101bb02e0acSNicolas Bonnefon // Menu actions
102bb02e0acSNicolas Bonnefon void MainWindow::createActions()
103bb02e0acSNicolas Bonnefon {
104bb02e0acSNicolas Bonnefon     Configuration& config = Persistent<Configuration>( "settings" );
105bb02e0acSNicolas Bonnefon 
106bb02e0acSNicolas Bonnefon     openAction = new QAction(tr("&Open..."), this);
107bb02e0acSNicolas Bonnefon     openAction->setShortcut(QKeySequence::Open);
108bb02e0acSNicolas Bonnefon     openAction->setIcon( QIcon(":/images/open16.png") );
109bb02e0acSNicolas Bonnefon     openAction->setStatusTip(tr("Open a file"));
110bb02e0acSNicolas Bonnefon     connect(openAction, SIGNAL(triggered()), this, SLOT(open()));
111bb02e0acSNicolas Bonnefon 
112bb02e0acSNicolas Bonnefon     // Recent files
113bb02e0acSNicolas Bonnefon     for (int i = 0; i < MaxRecentFiles; ++i) {
114bb02e0acSNicolas Bonnefon         recentFileActions[i] = new QAction(this);
115bb02e0acSNicolas Bonnefon         recentFileActions[i]->setVisible(false);
116bb02e0acSNicolas Bonnefon         connect(recentFileActions[i], SIGNAL(triggered()),
117bb02e0acSNicolas Bonnefon                 this, SLOT(openRecentFile()));
118bb02e0acSNicolas Bonnefon     }
119bb02e0acSNicolas Bonnefon 
120bb02e0acSNicolas Bonnefon     exitAction = new QAction(tr("E&xit"), this);
121bb02e0acSNicolas Bonnefon     exitAction->setShortcut(tr("Ctrl+Q"));
122bb02e0acSNicolas Bonnefon     exitAction->setStatusTip(tr("Exit the application"));
123bb02e0acSNicolas Bonnefon     connect( exitAction, SIGNAL(triggered()), this, SLOT(close()) );
124bb02e0acSNicolas Bonnefon 
125bb02e0acSNicolas Bonnefon     copyAction = new QAction(tr("&Copy"), this);
126bb02e0acSNicolas Bonnefon     copyAction->setShortcut(QKeySequence::Copy);
127bb02e0acSNicolas Bonnefon     copyAction->setStatusTip(tr("Copy the selection"));
128bb02e0acSNicolas Bonnefon     connect( copyAction, SIGNAL(triggered()), this, SLOT(copy()) );
129bb02e0acSNicolas Bonnefon 
130bb02e0acSNicolas Bonnefon     selectAllAction = new QAction(tr("Select &All"), this);
131bb02e0acSNicolas Bonnefon     selectAllAction->setShortcut(tr("Ctrl+A"));
132bb02e0acSNicolas Bonnefon     selectAllAction->setStatusTip(tr("Select all the text"));
133bb02e0acSNicolas Bonnefon     connect( selectAllAction, SIGNAL(triggered()),
134bb02e0acSNicolas Bonnefon              this, SLOT( selectAll() ) );
135bb02e0acSNicolas Bonnefon 
136bb02e0acSNicolas Bonnefon     findAction = new QAction(tr("&Find..."), this);
137bb02e0acSNicolas Bonnefon     findAction->setShortcut(QKeySequence::Find);
138bb02e0acSNicolas Bonnefon     findAction->setStatusTip(tr("Find the text"));
139bb02e0acSNicolas Bonnefon     connect( findAction, SIGNAL(triggered()),
140bb02e0acSNicolas Bonnefon             this, SLOT( find() ) );
141bb02e0acSNicolas Bonnefon 
142bb02e0acSNicolas Bonnefon     overviewVisibleAction = new QAction( tr("Matches &overview"), this );
143bb02e0acSNicolas Bonnefon     overviewVisibleAction->setCheckable( true );
144bb02e0acSNicolas Bonnefon     overviewVisibleAction->setChecked( config.isOverviewVisible() );
145bb02e0acSNicolas Bonnefon     connect( overviewVisibleAction, SIGNAL( toggled( bool ) ),
146bb02e0acSNicolas Bonnefon             this, SLOT( toggleOverviewVisibility( bool )) );
147bb02e0acSNicolas Bonnefon 
148bb02e0acSNicolas Bonnefon     lineNumbersVisibleInMainAction =
149bb02e0acSNicolas Bonnefon         new QAction( tr("Line &numbers in main view"), this );
150bb02e0acSNicolas Bonnefon     lineNumbersVisibleInMainAction->setCheckable( true );
151bb02e0acSNicolas Bonnefon     lineNumbersVisibleInMainAction->setChecked( config.mainLineNumbersVisible() );
152bb02e0acSNicolas Bonnefon     connect( lineNumbersVisibleInMainAction, SIGNAL( toggled( bool ) ),
153bb02e0acSNicolas Bonnefon             this, SLOT( toggleMainLineNumbersVisibility( bool )) );
154bb02e0acSNicolas Bonnefon 
155bb02e0acSNicolas Bonnefon     lineNumbersVisibleInFilteredAction =
156bb02e0acSNicolas Bonnefon         new QAction( tr("Line &numbers in filtered view"), this );
157bb02e0acSNicolas Bonnefon     lineNumbersVisibleInFilteredAction->setCheckable( true );
158bb02e0acSNicolas Bonnefon     lineNumbersVisibleInFilteredAction->setChecked( config.filteredLineNumbersVisible() );
159bb02e0acSNicolas Bonnefon     connect( lineNumbersVisibleInFilteredAction, SIGNAL( toggled( bool ) ),
160bb02e0acSNicolas Bonnefon             this, SLOT( toggleFilteredLineNumbersVisibility( bool )) );
161bb02e0acSNicolas Bonnefon 
162bb02e0acSNicolas Bonnefon     followAction = new QAction( tr("&Follow File"), this );
163bb02e0acSNicolas Bonnefon     followAction->setShortcut(Qt::Key_F);
164bb02e0acSNicolas Bonnefon     followAction->setCheckable(true);
165bb02e0acSNicolas Bonnefon     connect( followAction, SIGNAL(toggled( bool )),
166bb02e0acSNicolas Bonnefon             this, SIGNAL(followSet( bool )) );
167bb02e0acSNicolas Bonnefon 
168bb02e0acSNicolas Bonnefon     reloadAction = new QAction( tr("&Reload"), this );
169bb02e0acSNicolas Bonnefon     reloadAction->setShortcut(QKeySequence::Refresh);
170bb02e0acSNicolas Bonnefon     reloadAction->setIcon( QIcon(":/images/reload16.png") );
17132e44cfdSNicolas Bonnefon     signalMux_.connect( reloadAction, SIGNAL(triggered()), SLOT(reload()) );
172bb02e0acSNicolas Bonnefon 
173bb02e0acSNicolas Bonnefon     stopAction = new QAction( tr("&Stop"), this );
174bb02e0acSNicolas Bonnefon     stopAction->setIcon( QIcon(":/images/stop16.png") );
175bb02e0acSNicolas Bonnefon     stopAction->setEnabled( false );
176*7847299cSNicolas Bonnefon     signalMux_.connect( stopAction, SIGNAL(triggered()), SLOT(stopLoading()) );
177bb02e0acSNicolas Bonnefon 
178bb02e0acSNicolas Bonnefon     filtersAction = new QAction(tr("&Filters..."), this);
179bb02e0acSNicolas Bonnefon     filtersAction->setStatusTip(tr("Show the Filters box"));
180bb02e0acSNicolas Bonnefon     connect( filtersAction, SIGNAL(triggered()), this, SLOT(filters()) );
181bb02e0acSNicolas Bonnefon 
182bb02e0acSNicolas Bonnefon     optionsAction = new QAction(tr("&Options..."), this);
183bb02e0acSNicolas Bonnefon     optionsAction->setStatusTip(tr("Show the Options box"));
184bb02e0acSNicolas Bonnefon     connect( optionsAction, SIGNAL(triggered()), this, SLOT(options()) );
185bb02e0acSNicolas Bonnefon 
186bb02e0acSNicolas Bonnefon     aboutAction = new QAction(tr("&About"), this);
187bb02e0acSNicolas Bonnefon     aboutAction->setStatusTip(tr("Show the About box"));
188bb02e0acSNicolas Bonnefon     connect( aboutAction, SIGNAL(triggered()), this, SLOT(about()) );
189bb02e0acSNicolas Bonnefon 
190bb02e0acSNicolas Bonnefon     aboutQtAction = new QAction(tr("About &Qt"), this);
191bb02e0acSNicolas Bonnefon     aboutAction->setStatusTip(tr("Show the Qt library's About box"));
192bb02e0acSNicolas Bonnefon     connect( aboutQtAction, SIGNAL(triggered()), this, SLOT(aboutQt()) );
193bb02e0acSNicolas Bonnefon }
194bb02e0acSNicolas Bonnefon 
195bb02e0acSNicolas Bonnefon void MainWindow::createMenus()
196bb02e0acSNicolas Bonnefon {
197bb02e0acSNicolas Bonnefon     fileMenu = menuBar()->addMenu( tr("&File") );
198bb02e0acSNicolas Bonnefon     fileMenu->addAction( openAction );
199bb02e0acSNicolas Bonnefon     fileMenu->addSeparator();
200bb02e0acSNicolas Bonnefon     for (int i = 0; i < MaxRecentFiles; ++i) {
201bb02e0acSNicolas Bonnefon         fileMenu->addAction( recentFileActions[i] );
202bb02e0acSNicolas Bonnefon         recentFileActionBehaviors[i] =
203bb02e0acSNicolas Bonnefon             new MenuActionToolTipBehavior(recentFileActions[i], fileMenu, this);
204bb02e0acSNicolas Bonnefon     }
205bb02e0acSNicolas Bonnefon     fileMenu->addSeparator();
206bb02e0acSNicolas Bonnefon     fileMenu->addAction( exitAction );
207bb02e0acSNicolas Bonnefon 
208bb02e0acSNicolas Bonnefon     editMenu = menuBar()->addMenu( tr("&Edit") );
209bb02e0acSNicolas Bonnefon     editMenu->addAction( copyAction );
210bb02e0acSNicolas Bonnefon     editMenu->addAction( selectAllAction );
211bb02e0acSNicolas Bonnefon     editMenu->addSeparator();
212bb02e0acSNicolas Bonnefon     editMenu->addAction( findAction );
213bb02e0acSNicolas Bonnefon 
214bb02e0acSNicolas Bonnefon     viewMenu = menuBar()->addMenu( tr("&View") );
215bb02e0acSNicolas Bonnefon     viewMenu->addAction( overviewVisibleAction );
216bb02e0acSNicolas Bonnefon     viewMenu->addSeparator();
217bb02e0acSNicolas Bonnefon     viewMenu->addAction( lineNumbersVisibleInMainAction );
218bb02e0acSNicolas Bonnefon     viewMenu->addAction( lineNumbersVisibleInFilteredAction );
219bb02e0acSNicolas Bonnefon     viewMenu->addSeparator();
220bb02e0acSNicolas Bonnefon     viewMenu->addAction( followAction );
221bb02e0acSNicolas Bonnefon     viewMenu->addSeparator();
222bb02e0acSNicolas Bonnefon     viewMenu->addAction( reloadAction );
223bb02e0acSNicolas Bonnefon 
224bb02e0acSNicolas Bonnefon     toolsMenu = menuBar()->addMenu( tr("&Tools") );
225bb02e0acSNicolas Bonnefon     toolsMenu->addAction( filtersAction );
226bb02e0acSNicolas Bonnefon     toolsMenu->addSeparator();
227bb02e0acSNicolas Bonnefon     toolsMenu->addAction( optionsAction );
228bb02e0acSNicolas Bonnefon 
229bb02e0acSNicolas Bonnefon     menuBar()->addSeparator();
230bb02e0acSNicolas Bonnefon 
231bb02e0acSNicolas Bonnefon     helpMenu = menuBar()->addMenu( tr("&Help") );
232bb02e0acSNicolas Bonnefon     helpMenu->addAction( aboutAction );
233bb02e0acSNicolas Bonnefon }
234bb02e0acSNicolas Bonnefon 
235bb02e0acSNicolas Bonnefon void MainWindow::createToolBars()
236bb02e0acSNicolas Bonnefon {
237bb02e0acSNicolas Bonnefon     infoLine = new InfoLine();
238bb02e0acSNicolas Bonnefon     infoLine->setFrameStyle( QFrame::WinPanel | QFrame::Sunken );
239bb02e0acSNicolas Bonnefon     infoLine->setLineWidth( 0 );
240bb02e0acSNicolas Bonnefon 
241bb02e0acSNicolas Bonnefon     lineNbField = new QLabel( );
242bb02e0acSNicolas Bonnefon     lineNbField->setText( "Line 0" );
243bb02e0acSNicolas Bonnefon     lineNbField->setAlignment( Qt::AlignLeft | Qt::AlignVCenter );
244bb02e0acSNicolas Bonnefon     lineNbField->setMinimumSize(
245bb02e0acSNicolas Bonnefon             lineNbField->fontMetrics().size( 0, "Line 0000000") );
246bb02e0acSNicolas Bonnefon 
247bb02e0acSNicolas Bonnefon     toolBar = addToolBar( tr("&Toolbar") );
248bb02e0acSNicolas Bonnefon     toolBar->setIconSize( QSize( 16, 16 ) );
249bb02e0acSNicolas Bonnefon     toolBar->setMovable( false );
250bb02e0acSNicolas Bonnefon     toolBar->addAction( openAction );
251bb02e0acSNicolas Bonnefon     toolBar->addAction( reloadAction );
252bb02e0acSNicolas Bonnefon     toolBar->addWidget( infoLine );
253bb02e0acSNicolas Bonnefon     toolBar->addAction( stopAction );
254bb02e0acSNicolas Bonnefon     toolBar->addWidget( lineNbField );
255bb02e0acSNicolas Bonnefon }
256bb02e0acSNicolas Bonnefon 
257bb02e0acSNicolas Bonnefon //
258bb02e0acSNicolas Bonnefon // Slots
259bb02e0acSNicolas Bonnefon //
260bb02e0acSNicolas Bonnefon 
261bb02e0acSNicolas Bonnefon // Opens the file selection dialog to select a new log file
262bb02e0acSNicolas Bonnefon void MainWindow::open()
263bb02e0acSNicolas Bonnefon {
264bb02e0acSNicolas Bonnefon     QString defaultDir = ".";
265bb02e0acSNicolas Bonnefon 
266bb02e0acSNicolas Bonnefon     // Default to the path of the current file if there is one
267bb02e0acSNicolas Bonnefon     if ( !currentFile.isEmpty() ) {
268bb02e0acSNicolas Bonnefon         QFileInfo fileInfo = QFileInfo( currentFile );
269bb02e0acSNicolas Bonnefon         defaultDir = fileInfo.path();
270bb02e0acSNicolas Bonnefon     }
271bb02e0acSNicolas Bonnefon 
272bb02e0acSNicolas Bonnefon     QString fileName = QFileDialog::getOpenFileName(this,
273bb02e0acSNicolas Bonnefon             tr("Open file"), defaultDir, tr("All files (*)"));
274bb02e0acSNicolas Bonnefon     if (!fileName.isEmpty())
275bb02e0acSNicolas Bonnefon         loadFile(fileName);
276bb02e0acSNicolas Bonnefon }
277bb02e0acSNicolas Bonnefon 
278bb02e0acSNicolas Bonnefon // Opens a log file from the recent files list
279bb02e0acSNicolas Bonnefon void MainWindow::openRecentFile()
280bb02e0acSNicolas Bonnefon {
281bb02e0acSNicolas Bonnefon     QAction* action = qobject_cast<QAction*>(sender());
282bb02e0acSNicolas Bonnefon     if (action)
283bb02e0acSNicolas Bonnefon         loadFile(action->data().toString());
284bb02e0acSNicolas Bonnefon }
285bb02e0acSNicolas Bonnefon 
286bb02e0acSNicolas Bonnefon // Select all the text in the currently selected view
287bb02e0acSNicolas Bonnefon void MainWindow::selectAll()
288bb02e0acSNicolas Bonnefon {
289bb02e0acSNicolas Bonnefon     crawlerWidget->selectAll();
290bb02e0acSNicolas Bonnefon }
291bb02e0acSNicolas Bonnefon 
292bb02e0acSNicolas Bonnefon // Copy the currently selected line into the clipboard
293bb02e0acSNicolas Bonnefon void MainWindow::copy()
294bb02e0acSNicolas Bonnefon {
295bb02e0acSNicolas Bonnefon     static QClipboard* clipboard = QApplication::clipboard();
296bb02e0acSNicolas Bonnefon 
297bb02e0acSNicolas Bonnefon     clipboard->setText( crawlerWidget->getSelectedText() );
298bb02e0acSNicolas Bonnefon 
299bb02e0acSNicolas Bonnefon     // Put it in the global selection as well (X11 only)
300bb02e0acSNicolas Bonnefon     clipboard->setText( crawlerWidget->getSelectedText(),
301bb02e0acSNicolas Bonnefon             QClipboard::Selection );
302bb02e0acSNicolas Bonnefon }
303bb02e0acSNicolas Bonnefon 
304bb02e0acSNicolas Bonnefon // Display the QuickFind bar
305bb02e0acSNicolas Bonnefon void MainWindow::find()
306bb02e0acSNicolas Bonnefon {
307bb02e0acSNicolas Bonnefon     crawlerWidget->displayQuickFindBar( QuickFindMux::Forward );
308bb02e0acSNicolas Bonnefon }
309bb02e0acSNicolas Bonnefon 
310bb02e0acSNicolas Bonnefon // Opens the 'Filters' dialog box
311bb02e0acSNicolas Bonnefon void MainWindow::filters()
312bb02e0acSNicolas Bonnefon {
313bb02e0acSNicolas Bonnefon     FiltersDialog dialog(this);
314bb02e0acSNicolas Bonnefon     connect(&dialog, SIGNAL( optionsChanged() ), crawlerWidget, SLOT( applyConfiguration() ));
315bb02e0acSNicolas Bonnefon     dialog.exec();
316bb02e0acSNicolas Bonnefon }
317bb02e0acSNicolas Bonnefon 
318bb02e0acSNicolas Bonnefon // Opens the 'Options' modal dialog box
319bb02e0acSNicolas Bonnefon void MainWindow::options()
320bb02e0acSNicolas Bonnefon {
321bb02e0acSNicolas Bonnefon     OptionsDialog dialog(this);
322bb02e0acSNicolas Bonnefon     connect(&dialog, SIGNAL( optionsChanged() ), crawlerWidget, SLOT( applyConfiguration() ));
323bb02e0acSNicolas Bonnefon     dialog.exec();
324bb02e0acSNicolas Bonnefon }
325bb02e0acSNicolas Bonnefon 
326bb02e0acSNicolas Bonnefon // Opens the 'About' dialog box.
327bb02e0acSNicolas Bonnefon void MainWindow::about()
328bb02e0acSNicolas Bonnefon {
329bb02e0acSNicolas Bonnefon     QMessageBox::about(this, tr("About glogg"),
330bb02e0acSNicolas Bonnefon             tr("<h2>glogg " GLOGG_VERSION "</h2>"
331bb02e0acSNicolas Bonnefon                 "<p>A fast, advanced log explorer."
332bb02e0acSNicolas Bonnefon #ifdef GLOGG_COMMIT
333bb02e0acSNicolas Bonnefon                 "<p>Built " GLOGG_DATE " from " GLOGG_COMMIT
334bb02e0acSNicolas Bonnefon #endif
335bb02e0acSNicolas Bonnefon                 "<p>Copyright &copy; 2009, 2010, 2011, 2012, 2013 Nicolas Bonnefon and other contributors"
336bb02e0acSNicolas Bonnefon                 "<p>You may modify and redistribute the program under the terms of the GPL (version 3 or later)." ) );
337bb02e0acSNicolas Bonnefon }
338bb02e0acSNicolas Bonnefon 
339bb02e0acSNicolas Bonnefon // Opens the 'About Qt' dialog box.
340bb02e0acSNicolas Bonnefon void MainWindow::aboutQt()
341bb02e0acSNicolas Bonnefon {
342bb02e0acSNicolas Bonnefon }
343bb02e0acSNicolas Bonnefon 
344bb02e0acSNicolas Bonnefon void MainWindow::toggleOverviewVisibility( bool isVisible )
345bb02e0acSNicolas Bonnefon {
346bb02e0acSNicolas Bonnefon     Configuration& config = Persistent<Configuration>( "settings" );
347bb02e0acSNicolas Bonnefon     config.setOverviewVisible( isVisible );
348bb02e0acSNicolas Bonnefon     emit optionsChanged();
349bb02e0acSNicolas Bonnefon }
350bb02e0acSNicolas Bonnefon 
351bb02e0acSNicolas Bonnefon void MainWindow::toggleMainLineNumbersVisibility( bool isVisible )
352bb02e0acSNicolas Bonnefon {
353bb02e0acSNicolas Bonnefon     Configuration& config = Persistent<Configuration>( "settings" );
354bb02e0acSNicolas Bonnefon     config.setMainLineNumbersVisible( isVisible );
355bb02e0acSNicolas Bonnefon     emit optionsChanged();
356bb02e0acSNicolas Bonnefon }
357bb02e0acSNicolas Bonnefon 
358bb02e0acSNicolas Bonnefon void MainWindow::toggleFilteredLineNumbersVisibility( bool isVisible )
359bb02e0acSNicolas Bonnefon {
360bb02e0acSNicolas Bonnefon     Configuration& config = Persistent<Configuration>( "settings" );
361bb02e0acSNicolas Bonnefon     config.setFilteredLineNumbersVisible( isVisible );
362bb02e0acSNicolas Bonnefon     emit optionsChanged();
363bb02e0acSNicolas Bonnefon }
364bb02e0acSNicolas Bonnefon 
365bb02e0acSNicolas Bonnefon void MainWindow::disableFollow()
366bb02e0acSNicolas Bonnefon {
367bb02e0acSNicolas Bonnefon     followAction->setChecked( false );
368bb02e0acSNicolas Bonnefon }
369bb02e0acSNicolas Bonnefon 
370bb02e0acSNicolas Bonnefon void MainWindow::lineNumberHandler( int line )
371bb02e0acSNicolas Bonnefon {
372bb02e0acSNicolas Bonnefon     // The line number received is the internal (starts at 0)
373bb02e0acSNicolas Bonnefon     lineNbField->setText( tr( "Line %1" ).arg( line + 1 ) );
374bb02e0acSNicolas Bonnefon }
375bb02e0acSNicolas Bonnefon 
376bb02e0acSNicolas Bonnefon void MainWindow::updateLoadingProgress( int progress )
377bb02e0acSNicolas Bonnefon {
378bb02e0acSNicolas Bonnefon     LOG(logDEBUG) << "Loading progress: " << progress;
379bb02e0acSNicolas Bonnefon 
380bb02e0acSNicolas Bonnefon     // We ignore 0% and 100% to avoid a flash when the file (or update)
381bb02e0acSNicolas Bonnefon     // is very short.
382bb02e0acSNicolas Bonnefon     if ( progress > 0 && progress < 100 ) {
383bb02e0acSNicolas Bonnefon         infoLine->setText( loadingFileName + tr( " - Indexing lines... (%1 %)" ).arg( progress ) );
384bb02e0acSNicolas Bonnefon         infoLine->displayGauge( progress );
385bb02e0acSNicolas Bonnefon 
386bb02e0acSNicolas Bonnefon         stopAction->setEnabled( true );
387bb02e0acSNicolas Bonnefon     }
388bb02e0acSNicolas Bonnefon }
389bb02e0acSNicolas Bonnefon 
390bb02e0acSNicolas Bonnefon void MainWindow::displayNormalStatus( bool success )
391bb02e0acSNicolas Bonnefon {
392bb02e0acSNicolas Bonnefon     QLocale defaultLocale;
393bb02e0acSNicolas Bonnefon 
394bb02e0acSNicolas Bonnefon     LOG(logDEBUG) << "displayNormalStatus";
395bb02e0acSNicolas Bonnefon 
396bb02e0acSNicolas Bonnefon     if ( success )
397bb02e0acSNicolas Bonnefon         setCurrentFile( loadingFileName );
398bb02e0acSNicolas Bonnefon 
3990a90ca6aSNicolas Bonnefon     uint64_t fileSize;
4000a90ca6aSNicolas Bonnefon     uint32_t fileNbLine;
401bb02e0acSNicolas Bonnefon     QDateTime lastModified;
402bb02e0acSNicolas Bonnefon 
4030a90ca6aSNicolas Bonnefon     session_->getFileInfo( crawlerWidget,
4040a90ca6aSNicolas Bonnefon             &fileSize, &fileNbLine, &lastModified );
405bb02e0acSNicolas Bonnefon     if ( lastModified.isValid() ) {
406bb02e0acSNicolas Bonnefon         const QString date =
407bb02e0acSNicolas Bonnefon             defaultLocale.toString( lastModified, QLocale::NarrowFormat );
408bb02e0acSNicolas Bonnefon         infoLine->setText( tr( "%1 (%2 - %3 lines - modified on %4)" )
409bb02e0acSNicolas Bonnefon                 .arg(currentFile).arg(readableSize(fileSize))
410bb02e0acSNicolas Bonnefon                 .arg(fileNbLine).arg( date ) );
411bb02e0acSNicolas Bonnefon     }
412bb02e0acSNicolas Bonnefon     else {
413bb02e0acSNicolas Bonnefon         infoLine->setText( tr( "%1 (%2 - %3 lines)" )
414bb02e0acSNicolas Bonnefon                 .arg(currentFile).arg(readableSize(fileSize))
415bb02e0acSNicolas Bonnefon                 .arg(fileNbLine) );
416bb02e0acSNicolas Bonnefon     }
417bb02e0acSNicolas Bonnefon 
418bb02e0acSNicolas Bonnefon     infoLine->hideGauge();
419bb02e0acSNicolas Bonnefon     stopAction->setEnabled( false );
4200a90ca6aSNicolas Bonnefon 
4210a90ca6aSNicolas Bonnefon     // Now everything is ready, we can finally show the file!
4220a90ca6aSNicolas Bonnefon     crawlerWidget->show();
423bb02e0acSNicolas Bonnefon }
424bb02e0acSNicolas Bonnefon 
425bb02e0acSNicolas Bonnefon //
426bb02e0acSNicolas Bonnefon // Events
427bb02e0acSNicolas Bonnefon //
428bb02e0acSNicolas Bonnefon 
429bb02e0acSNicolas Bonnefon // Closes the application
430bb02e0acSNicolas Bonnefon void MainWindow::closeEvent( QCloseEvent *event )
431bb02e0acSNicolas Bonnefon {
432bb02e0acSNicolas Bonnefon     writeSettings();
433bb02e0acSNicolas Bonnefon     event->accept();
434bb02e0acSNicolas Bonnefon }
435bb02e0acSNicolas Bonnefon 
436bb02e0acSNicolas Bonnefon // Accepts the drag event if it looks like a filename
437bb02e0acSNicolas Bonnefon void MainWindow::dragEnterEvent( QDragEnterEvent* event )
438bb02e0acSNicolas Bonnefon {
439bb02e0acSNicolas Bonnefon     if ( event->mimeData()->hasFormat( "text/uri-list" ) )
440bb02e0acSNicolas Bonnefon         event->acceptProposedAction();
441bb02e0acSNicolas Bonnefon }
442bb02e0acSNicolas Bonnefon 
443bb02e0acSNicolas Bonnefon // Tries and loads the file if the URL dropped is local
444bb02e0acSNicolas Bonnefon void MainWindow::dropEvent( QDropEvent* event )
445bb02e0acSNicolas Bonnefon {
446bb02e0acSNicolas Bonnefon     QList<QUrl> urls = event->mimeData()->urls();
447bb02e0acSNicolas Bonnefon     if ( urls.isEmpty() )
448bb02e0acSNicolas Bonnefon         return;
449bb02e0acSNicolas Bonnefon 
450bb02e0acSNicolas Bonnefon     QString fileName = urls.first().toLocalFile();
451bb02e0acSNicolas Bonnefon     if ( fileName.isEmpty() )
452bb02e0acSNicolas Bonnefon         return;
453bb02e0acSNicolas Bonnefon 
454bb02e0acSNicolas Bonnefon     loadFile( fileName );
455bb02e0acSNicolas Bonnefon }
456bb02e0acSNicolas Bonnefon 
457bb02e0acSNicolas Bonnefon //
458bb02e0acSNicolas Bonnefon // Private functions
459bb02e0acSNicolas Bonnefon //
460bb02e0acSNicolas Bonnefon 
4610a90ca6aSNicolas Bonnefon // Create a CrawlerWidget for the passed file, start its loading
4620a90ca6aSNicolas Bonnefon // and update the title bar.
463bb02e0acSNicolas Bonnefon // The loading is done asynchronously.
464bb02e0acSNicolas Bonnefon bool MainWindow::loadFile( const QString& fileName )
465bb02e0acSNicolas Bonnefon {
466bb02e0acSNicolas Bonnefon     LOG(logDEBUG) << "loadFile ( " << fileName.toStdString() << " )";
467bb02e0acSNicolas Bonnefon 
468f0708ca8SNicolas Bonnefon     // First get the global search history
469f0708ca8SNicolas Bonnefon     savedSearches = &(Persistent<SavedSearches>( "savedSearches" ));
470f0708ca8SNicolas Bonnefon 
471bb02e0acSNicolas Bonnefon     // Load the file
472bb02e0acSNicolas Bonnefon     loadingFileName = fileName;
473039481acSNicolas Bonnefon 
474f0708ca8SNicolas Bonnefon     crawlerWidget = dynamic_cast<CrawlerWidget*>( session_->open( fileName.toStdString(),
475f0708ca8SNicolas Bonnefon             [this]() { return new CrawlerWidget( savedSearches, this ); } ) );
476f0708ca8SNicolas Bonnefon 
4770a90ca6aSNicolas Bonnefon     // We won't show the widget until the file is fully loaded
4780a90ca6aSNicolas Bonnefon     crawlerWidget->hide();
479f0708ca8SNicolas Bonnefon 
480313a820fSNicolas Bonnefon     signalMux_.setCurrentDocument( crawlerWidget );
481313a820fSNicolas Bonnefon 
482f0708ca8SNicolas Bonnefon     // Send actions to the crawlerwidget
483313a820fSNicolas Bonnefon     signalMux_.connect( this, SIGNAL( followSet( bool ) ),
484313a820fSNicolas Bonnefon             SIGNAL( followSet( bool ) ) );
485313a820fSNicolas Bonnefon     signalMux_.connect( this, SIGNAL( optionsChanged() ),
486313a820fSNicolas Bonnefon             SLOT( applyConfiguration() ) );
487f0708ca8SNicolas Bonnefon 
488f0708ca8SNicolas Bonnefon     // Actions from the CrawlerWidget
489313a820fSNicolas Bonnefon     signalMux_.connect( SIGNAL( followDisabled() ),
490f0708ca8SNicolas Bonnefon             this, SLOT( disableFollow() ) );
491313a820fSNicolas Bonnefon     signalMux_.connect( SIGNAL( updateLineNumber( int ) ),
492f0708ca8SNicolas Bonnefon             this, SLOT( lineNumberHandler( int ) ) );
493f0708ca8SNicolas Bonnefon 
4940a90ca6aSNicolas Bonnefon     // FIXME: is it necessary?
495f0708ca8SNicolas Bonnefon     emit optionsChanged();
496f0708ca8SNicolas Bonnefon 
497f0708ca8SNicolas Bonnefon     // We start with the empty file
498f0708ca8SNicolas Bonnefon     setCurrentFile( "" );
499f0708ca8SNicolas Bonnefon 
500f0708ca8SNicolas Bonnefon     // Register for progress status bar
501313a820fSNicolas Bonnefon     signalMux_.connect( SIGNAL( loadingProgressed( int ) ),
502f0708ca8SNicolas Bonnefon             this, SLOT( updateLoadingProgress( int ) ) );
503313a820fSNicolas Bonnefon     signalMux_.connect( SIGNAL( loadingFinished( bool ) ),
504f0708ca8SNicolas Bonnefon             this, SLOT( displayNormalStatus( bool ) ) );
505f0708ca8SNicolas Bonnefon 
506f0708ca8SNicolas Bonnefon     setCentralWidget(crawlerWidget);
507f0708ca8SNicolas Bonnefon 
508bb02e0acSNicolas Bonnefon     LOG(logDEBUG) << "Success loading file " << fileName.toStdString();
509bb02e0acSNicolas Bonnefon     return true;
510bb02e0acSNicolas Bonnefon }
511bb02e0acSNicolas Bonnefon 
512bb02e0acSNicolas Bonnefon // Strips the passed filename from its directory part.
513bb02e0acSNicolas Bonnefon QString MainWindow::strippedName( const QString& fullFileName ) const
514bb02e0acSNicolas Bonnefon {
515bb02e0acSNicolas Bonnefon     return QFileInfo( fullFileName ).fileName();
516bb02e0acSNicolas Bonnefon }
517bb02e0acSNicolas Bonnefon 
518bb02e0acSNicolas Bonnefon // Add the filename to the recent files list and update the title bar.
519bb02e0acSNicolas Bonnefon void MainWindow::setCurrentFile( const QString& fileName )
520bb02e0acSNicolas Bonnefon {
52132e44cfdSNicolas Bonnefon     if ( fileName != currentFile )
52232e44cfdSNicolas Bonnefon     {
523bb02e0acSNicolas Bonnefon         // Change the current file
524bb02e0acSNicolas Bonnefon         currentFile = fileName;
525bb02e0acSNicolas Bonnefon         QString shownName = tr( "Untitled" );
526bb02e0acSNicolas Bonnefon         if ( !currentFile.isEmpty() ) {
527bb02e0acSNicolas Bonnefon             // (reload the list first in case another glogg changed it)
528bb02e0acSNicolas Bonnefon             GetPersistentInfo().retrieve( "recentFiles" );
529bb02e0acSNicolas Bonnefon             recentFiles.addRecent( currentFile );
530bb02e0acSNicolas Bonnefon             GetPersistentInfo().save( "recentFiles" );
531bb02e0acSNicolas Bonnefon             updateRecentFileActions();
532bb02e0acSNicolas Bonnefon             shownName = strippedName( currentFile );
533bb02e0acSNicolas Bonnefon         }
534bb02e0acSNicolas Bonnefon 
535bb02e0acSNicolas Bonnefon         setWindowTitle(
536bb02e0acSNicolas Bonnefon                 tr("%1 - %2").arg(shownName).arg(tr("glogg"))
537bb02e0acSNicolas Bonnefon #ifdef GLOGG_COMMIT
538bb02e0acSNicolas Bonnefon                 + " (dev build " GLOGG_VERSION ")"
539bb02e0acSNicolas Bonnefon #endif
540bb02e0acSNicolas Bonnefon                 );
541bb02e0acSNicolas Bonnefon     }
54232e44cfdSNicolas Bonnefon     else
54332e44cfdSNicolas Bonnefon     {
54432e44cfdSNicolas Bonnefon         // Nothing, happens when e.g., the file is reloaded
54532e44cfdSNicolas Bonnefon     }
54632e44cfdSNicolas Bonnefon }
547bb02e0acSNicolas Bonnefon 
548bb02e0acSNicolas Bonnefon // Updates the actions for the recent files.
549bb02e0acSNicolas Bonnefon // Must be called after having added a new name to the list.
550bb02e0acSNicolas Bonnefon void MainWindow::updateRecentFileActions()
551bb02e0acSNicolas Bonnefon {
552bb02e0acSNicolas Bonnefon     QStringList recent_files = recentFiles.recentFiles();
553bb02e0acSNicolas Bonnefon 
554bb02e0acSNicolas Bonnefon     for ( int j = 0; j < MaxRecentFiles; ++j ) {
555bb02e0acSNicolas Bonnefon         if ( j < recent_files.count() ) {
556bb02e0acSNicolas Bonnefon             QString text = tr("&%1 %2").arg(j + 1).arg(strippedName(recent_files[j]));
557bb02e0acSNicolas Bonnefon             recentFileActions[j]->setText( text );
558bb02e0acSNicolas Bonnefon             recentFileActions[j]->setToolTip( recent_files[j] );
559bb02e0acSNicolas Bonnefon             recentFileActions[j]->setData( recent_files[j] );
560bb02e0acSNicolas Bonnefon             recentFileActions[j]->setVisible( true );
561bb02e0acSNicolas Bonnefon         }
562bb02e0acSNicolas Bonnefon         else {
563bb02e0acSNicolas Bonnefon             recentFileActions[j]->setVisible( false );
564bb02e0acSNicolas Bonnefon         }
565bb02e0acSNicolas Bonnefon     }
566bb02e0acSNicolas Bonnefon 
567bb02e0acSNicolas Bonnefon     // separatorAction->setVisible(!recentFiles.isEmpty());
568bb02e0acSNicolas Bonnefon }
569bb02e0acSNicolas Bonnefon 
570bb02e0acSNicolas Bonnefon // Write settings to permanent storage
571bb02e0acSNicolas Bonnefon void MainWindow::writeSettings()
572bb02e0acSNicolas Bonnefon {
573bb02e0acSNicolas Bonnefon     // Save the session
574bb02e0acSNicolas Bonnefon     SessionInfo& session = Persistent<SessionInfo>( "session" );
575bb02e0acSNicolas Bonnefon     session.setGeometry( saveGeometry() );
576bb02e0acSNicolas Bonnefon     session.setCrawlerState( crawlerWidget->saveState() );
577bb02e0acSNicolas Bonnefon     session.setCurrentFile( currentFile );
578bb02e0acSNicolas Bonnefon     GetPersistentInfo().save( QString( "session" ) );
579bb02e0acSNicolas Bonnefon 
580bb02e0acSNicolas Bonnefon     // User settings
581bb02e0acSNicolas Bonnefon     GetPersistentInfo().save( QString( "settings" ) );
582bb02e0acSNicolas Bonnefon }
583bb02e0acSNicolas Bonnefon 
584bb02e0acSNicolas Bonnefon // Read settings from permanent storage
585bb02e0acSNicolas Bonnefon void MainWindow::readSettings()
586bb02e0acSNicolas Bonnefon {
587bb02e0acSNicolas Bonnefon     // Get and restore the session
588bb02e0acSNicolas Bonnefon     GetPersistentInfo().retrieve( QString( "session" ) );
589bb02e0acSNicolas Bonnefon     SessionInfo session = Persistent<SessionInfo>( "session" );
590bb02e0acSNicolas Bonnefon     restoreGeometry( session.geometry() );
591bb02e0acSNicolas Bonnefon     previousFile = session.currentFile();
5920a90ca6aSNicolas Bonnefon     /*
5930a90ca6aSNicolas Bonnefon      * FIXME: should be in the session
5940a90ca6aSNicolas Bonnefon     crawlerWidget->restoreState( session.crawlerState() );
5950a90ca6aSNicolas Bonnefon     */
596bb02e0acSNicolas Bonnefon 
597bb02e0acSNicolas Bonnefon     // History of recent files
598bb02e0acSNicolas Bonnefon     GetPersistentInfo().retrieve( QString( "recentFiles" ) );
599bb02e0acSNicolas Bonnefon     updateRecentFileActions();
600bb02e0acSNicolas Bonnefon 
601bb02e0acSNicolas Bonnefon     GetPersistentInfo().retrieve( QString( "savedSearches" ) );
602bb02e0acSNicolas Bonnefon     GetPersistentInfo().retrieve( QString( "settings" ) );
603bb02e0acSNicolas Bonnefon     GetPersistentInfo().retrieve( QString( "filterSet" ) );
604bb02e0acSNicolas Bonnefon }
605bb02e0acSNicolas Bonnefon 
606bb02e0acSNicolas Bonnefon // Returns the size in human readable format
6070a90ca6aSNicolas Bonnefon static QString readableSize( qint64 size )
608bb02e0acSNicolas Bonnefon {
609bb02e0acSNicolas Bonnefon     static const QString sizeStrs[] = {
6100a90ca6aSNicolas Bonnefon         QObject::tr("B"), QObject::tr("KiB"), QObject::tr("MiB"),
6110a90ca6aSNicolas Bonnefon         QObject::tr("GiB"), QObject::tr("TiB") };
612bb02e0acSNicolas Bonnefon 
613bb02e0acSNicolas Bonnefon     QLocale defaultLocale;
614bb02e0acSNicolas Bonnefon     unsigned int i;
615bb02e0acSNicolas Bonnefon     double humanSize = size;
616bb02e0acSNicolas Bonnefon 
617bb02e0acSNicolas Bonnefon     for ( i=0; i+1 < (sizeof(sizeStrs)/sizeof(QString)) && (humanSize/1024.0) >= 1024.0; i++ )
618bb02e0acSNicolas Bonnefon         humanSize /= 1024.0;
619bb02e0acSNicolas Bonnefon 
620bb02e0acSNicolas Bonnefon     if ( humanSize >= 1024.0 ) {
621bb02e0acSNicolas Bonnefon         humanSize /= 1024.0;
622bb02e0acSNicolas Bonnefon         i++;
623bb02e0acSNicolas Bonnefon     }
624bb02e0acSNicolas Bonnefon 
625bb02e0acSNicolas Bonnefon     QString output;
626bb02e0acSNicolas Bonnefon     if ( i == 0 )
627bb02e0acSNicolas Bonnefon         // No decimal part if we display straight bytes.
628bb02e0acSNicolas Bonnefon         output = defaultLocale.toString( (int) humanSize );
629bb02e0acSNicolas Bonnefon     else
630bb02e0acSNicolas Bonnefon         output = defaultLocale.toString( humanSize, 'f', 1 );
631bb02e0acSNicolas Bonnefon 
632bb02e0acSNicolas Bonnefon     output += QString(" ") + sizeStrs[i];
633bb02e0acSNicolas Bonnefon 
634bb02e0acSNicolas Bonnefon     return output;
635bb02e0acSNicolas Bonnefon }
636