| /* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil -*- */ |
| /* |
| * Copyright (c) 2012-2013 University of California, Los Angeles |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 as |
| * published by the Free Software Foundation; |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| * |
| * Author: Jared Lindblom <lindblom@cs.ucla.edu> |
| * Alexander Afanasyev <alexander.afanasyev@ucla.edu> |
| * Zhenkai Zhu <zhenkai@cs.ucla.edu> |
| * Ilya Moiseenko <iliamo@ucla.edu> |
| */ |
| |
| #include "chronosharegui.h" |
| #include "config.h" |
| |
| #include "logging.h" |
| #include "ccnx-wrapper.h" |
| #include <QValidator> |
| #include <QDir> |
| #include <QFileInfo> |
| #include <QDesktopServices> |
| |
| #include <boost/make_shared.hpp> |
| |
| using namespace boost; |
| using namespace Ccnx; |
| |
| static const string HTTP_SERVER_ADDRESS = "localhost"; |
| static const string HTTP_SERVER_PORT = "9001"; |
| #ifdef _DEBUG |
| static const string DOC_ROOT = "gui/html"; |
| #else |
| static const string DOC_ROOT = ":/html"; |
| #endif |
| static const QString ICON_BIG_FILE(":/images/chronoshare-big.png"); |
| static const QString ICON_TRAY_FILE(":/images/" TRAY_ICON); |
| |
| INIT_LOGGER ("Gui"); |
| |
| ChronoShareGui::ChronoShareGui(QWidget *parent) |
| : QDialog(parent) |
| , m_watcher(0) |
| , m_dispatcher(0) |
| , m_httpServer(0) |
| #ifdef ADHOC_SUPPORTED |
| , m_executor (1) |
| #endif |
| #ifdef SPARKLE_SUPPORTED |
| , m_autoUpdate(new SparkleAutoUpdate(tr("http://irl.cs.ucla.edu/~zhenkai/chronoshare_dist/chronoshare.xml"))) |
| #endif |
| { |
| setWindowTitle("Settings"); |
| setMinimumWidth (600); |
| |
| labelUsername = new QLabel("Username (hint: /<username>)"); |
| labelSharedFolder = new QLabel("Shared Folder Name"); |
| labelSharedFolderPath = new QLabel("Shared Folder Path"); |
| |
| QRegExp regex("(/[^/]+)+$"); |
| QValidator *prefixValidator = new QRegExpValidator(regex, this); |
| |
| editUsername = new QLineEdit(); |
| editUsername->setValidator(prefixValidator); |
| |
| QRegExp noPureWhiteSpace("^\\S+.*$"); |
| QValidator *wsValidator = new QRegExpValidator(noPureWhiteSpace, this); |
| editSharedFolder = new QLineEdit(); |
| editSharedFolder->setValidator(wsValidator); |
| |
| editSharedFolderPath = new QLineEdit(); |
| editSharedFolderPath->setReadOnly(true); |
| QPalette pal = editSharedFolderPath->palette(); |
| pal.setColor(QPalette::Active, QPalette::Base, pal.color(QPalette::Disabled, QPalette::Base)); |
| editSharedFolderPath->setPalette(pal); |
| button = new QPushButton("Save and apply settings"); |
| |
| QString versionString = QString("Version: ChronoShare v%1").arg(CHRONOSHARE_VERSION); |
| label = new QLabel(versionString, this); |
| |
| connect(button, SIGNAL(clicked()), this, SLOT(changeSettings())); |
| |
| mainLayout = new QVBoxLayout; //vertically |
| mainLayout->addWidget(labelUsername); |
| mainLayout->addWidget(editUsername); |
| mainLayout->addWidget(labelSharedFolder); |
| mainLayout->addWidget(editSharedFolder); |
| mainLayout->addWidget(labelSharedFolderPath); |
| mainLayout->addWidget(editSharedFolderPath); |
| mainLayout->addWidget(button); |
| mainLayout->addWidget(label); |
| setLayout(mainLayout); |
| |
| // create actions that result from clicking a menu option |
| createActionsAndMenu(); |
| |
| // create tray icon |
| createTrayIcon(); |
| |
| // set icon image |
| setIcon(); |
| |
| // show tray icon |
| m_trayIcon->show(); |
| |
| // load settings |
| if(!loadSettings()) |
| { |
| // prompt user to choose folder |
| openMessageBox("First Time Setup", "Please enter a username, shared folder name and choose the shared folder path on your local filesystem."); |
| viewSettings(); |
| openFileDialog(); |
| viewSettings(); |
| } |
| else |
| { |
| if (m_username.isNull () || m_username == "" || |
| m_sharedFolderName.isNull () || m_sharedFolderName == "") |
| { |
| openMessageBox("First Time Setup", "To activate ChronoShare, please configure your username and shared folder name."); |
| viewSettings(); |
| } |
| else |
| { |
| startBackend(); |
| } |
| } |
| |
| #ifdef ADHOC_SUPPORTED |
| m_executor.start (); |
| #endif |
| } |
| |
| void |
| ChronoShareGui::startBackend (bool restart/*=false*/) |
| { |
| if (m_username.isNull () || m_username=="" || |
| m_sharedFolderName.isNull () || m_sharedFolderName=="" || |
| m_dirPath.isNull () || m_dirPath=="") |
| { |
| _LOG_DEBUG ("Don't start backend, because settings are in progress or incomplete"); |
| return; |
| } |
| |
| if (m_watcher != 0 && m_dispatcher != 0) |
| { |
| if (!restart) |
| { |
| return; |
| } |
| |
| _LOG_DEBUG ("Restarting Dispatcher and FileWatcher for the new location or new username"); |
| delete m_watcher; // stop filewatching ASAP |
| delete m_dispatcher; // stop dispatcher ASAP, but after watcher (to prevent triggering callbacks on deleted object) |
| } |
| |
| filesystem::path realPathToFolder (m_dirPath.toStdString ()); |
| realPathToFolder /= m_sharedFolderName.toStdString (); |
| |
| m_dispatcher = new Dispatcher (m_username.toStdString (), m_sharedFolderName.toStdString (), |
| realPathToFolder, make_shared<CcnxWrapper> ()); |
| |
| // Alex: this **must** be here, otherwise m_dirPath will be uninitialized |
| m_watcher = new FsWatcher (realPathToFolder.string ().c_str (), |
| bind (&Dispatcher::Did_LocalFile_AddOrModify, m_dispatcher, _1), |
| bind (&Dispatcher::Did_LocalFile_Delete, m_dispatcher, _1)); |
| |
| if (m_httpServer != 0) |
| { |
| // no need to restart webserver if it already exists |
| return; |
| } |
| |
| QFileInfo indexHtmlInfo(":/html/index.html"); |
| if (indexHtmlInfo.exists()) |
| { |
| try |
| { |
| m_httpServer = new http::server::server(HTTP_SERVER_ADDRESS, HTTP_SERVER_PORT, DOC_ROOT); |
| m_httpServerThread = boost::thread(&http::server::server::run, m_httpServer); |
| } |
| catch (std::exception &e) |
| { |
| _LOG_ERROR ("Start http server failed"); |
| m_httpServer = 0; // just to make sure |
| QMessageBox msgBox; |
| msgBox.setText ("WARNING: Cannot start http server!"); |
| msgBox.setIcon (QMessageBox::Warning); |
| msgBox.setInformativeText(QString("Starting http server failed. You will not be able to check history from web brower. Exception caused: %1").arg(e.what())); |
| msgBox.setStandardButtons(QMessageBox::Ok); |
| msgBox.exec(); |
| } |
| } |
| else |
| { |
| _LOG_ERROR ("Http server doc root dir does not exist!"); |
| } |
| } |
| |
| ChronoShareGui::~ChronoShareGui() |
| { |
| #ifdef ADHOC_SUPPORTED |
| m_executor.shutdown (); |
| #endif |
| |
| delete m_watcher; // stop filewatching ASAP |
| delete m_dispatcher; // stop dispatcher ASAP, but after watcher (to prevent triggering callbacks on deleted object) |
| if (m_httpServer != 0) |
| { |
| m_httpServer->handle_stop(); |
| m_httpServerThread.join(); |
| delete m_httpServer; |
| } |
| |
| // cleanup |
| delete m_trayIcon; |
| delete m_trayIconMenu; |
| #ifdef ADHOC_SUPPORTED |
| delete m_wifiAction; |
| #endif |
| #ifdef SPARKLE_SUPPORTED |
| delete m_autoUpdate; |
| delete m_checkForUpdates; |
| #endif |
| delete m_openFolder; |
| delete m_viewSettings; |
| delete m_changeFolder; |
| delete m_quitProgram; |
| |
| delete labelUsername; |
| delete labelSharedFolder; |
| delete editUsername; |
| delete editSharedFolder; |
| delete button; |
| delete label; |
| delete mainLayout; |
| } |
| |
| void ChronoShareGui::openMessageBox(QString title, QString text) |
| { |
| QMessageBox messageBox(this); |
| messageBox.setWindowTitle(title); |
| messageBox.setText(text); |
| |
| messageBox.setIconPixmap(QPixmap(ICON_BIG_FILE)); |
| |
| messageBox.exec(); |
| } |
| |
| void ChronoShareGui::openMessageBox(QString title, QString text, QString infotext) |
| { |
| QMessageBox messageBox(this); |
| messageBox.setWindowTitle(title); |
| messageBox.setText(text); |
| messageBox.setInformativeText(infotext); |
| |
| messageBox.setIconPixmap(QPixmap(ICON_BIG_FILE)); |
| |
| messageBox.exec(); |
| } |
| |
| void ChronoShareGui::createActionsAndMenu() |
| { |
| _LOG_DEBUG ("Create actions"); |
| |
| // create the "open folder" action |
| m_openFolder = new QAction(tr("&Open Folder"), this); |
| connect(m_openFolder, SIGNAL(triggered()), this, SLOT(openSharedFolder())); |
| |
| m_openWeb = new QAction(tr("Open in &Browser"), this); |
| connect(m_openWeb, SIGNAL(triggered()), this, SLOT(openInWebBrowser())); |
| |
| m_recentFilesMenu = new QMenu(tr("Recently Changed Files"), this); |
| for (int i = 0; i < 5; i++) |
| { |
| m_fileActions[i] = new QAction(this); |
| m_fileActions[i]->setVisible(false); |
| connect(m_fileActions[i], SIGNAL(triggered()), this, SLOT(openFile())); |
| m_recentFilesMenu->addAction(m_fileActions[i]); |
| } |
| connect(m_recentFilesMenu, SIGNAL(aboutToShow()), this, SLOT(updateRecentFilesMenu())); |
| |
| // create the "view settings" action |
| m_viewSettings = new QAction(tr("&View Settings"), this); |
| connect(m_viewSettings, SIGNAL(triggered()), this, SLOT(viewSettings())); |
| |
| // create the "change folder" action |
| m_changeFolder = new QAction(tr("&Change Folder"), this); |
| connect(m_changeFolder, SIGNAL(triggered()), this, SLOT(openFileDialog())); |
| |
| #ifdef ADHOC_SUPPORTED |
| // create "AdHoc Wifi" action |
| m_wifiAction = new QAction (tr("Enable AdHoc &WiFI"), m_trayIconMenu); |
| m_wifiAction->setChecked (false); |
| m_wifiAction->setCheckable (true); |
| connect (m_wifiAction, SIGNAL (toggled(bool)), this, SLOT(onAdHocChange(bool))); |
| #endif |
| |
| #ifdef SPARKLE_SUPPORTED |
| m_checkForUpdates = new QAction (tr("Check For Updates"), this); |
| connect (m_checkForUpdates, SIGNAL(triggered()), this, SLOT(onCheckForUpdates())); |
| #endif |
| |
| // create the "quit program" action |
| m_quitProgram = new QAction(tr("&Quit"), this); |
| connect(m_quitProgram, SIGNAL(triggered()), qApp, SLOT(quit())); |
| } |
| |
| void ChronoShareGui::createTrayIcon() |
| { |
| // create a new icon menu |
| m_trayIconMenu = new QMenu(this); |
| |
| // add actions to the menu |
| m_trayIconMenu->addAction(m_openFolder); |
| m_trayIconMenu->addAction(m_openWeb); |
| m_trayIconMenu->addMenu(m_recentFilesMenu); |
| m_trayIconMenu->addSeparator(); |
| m_trayIconMenu->addAction(m_viewSettings); |
| m_trayIconMenu->addAction(m_changeFolder); |
| |
| #ifdef SPARKLE_SUPPORTED |
| m_trayIconMenu->addSeparator(); |
| m_trayIconMenu->addAction(m_checkForUpdates); |
| #endif |
| |
| #ifdef ADHOC_SUPPORTED |
| m_trayIconMenu->addSeparator(); |
| m_trayIconMenu->addAction(m_wifiAction); |
| #endif |
| |
| m_trayIconMenu->addSeparator(); |
| m_trayIconMenu->addAction(m_quitProgram); |
| |
| // create new tray icon |
| m_trayIcon = new QSystemTrayIcon(this); |
| |
| // associate the menu with the tray icon |
| m_trayIcon->setContextMenu(m_trayIconMenu); |
| |
| // handle left click of icon |
| connect(m_trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)), this, SLOT(trayIconClicked(QSystemTrayIcon::ActivationReason))); |
| } |
| |
| void |
| ChronoShareGui::onAdHocChange (bool state) |
| { |
| #ifdef ADHOC_SUPPORTED |
| if (m_wifiAction->isChecked ()) |
| { |
| QMessageBox msgBox; |
| msgBox.setText ("WARNING: your WiFi will be disconnected"); |
| msgBox.setIcon (QMessageBox::Warning); |
| msgBox.setInformativeText("AdHoc WiFi will disconnect your current WiFi.\n Are you sure that you are OK with that?"); |
| msgBox.setStandardButtons(QMessageBox::Ok | QMessageBox::Cancel); |
| msgBox.setDefaultButton(QMessageBox::Cancel); |
| int ret = msgBox.exec(); |
| |
| if (ret == QMessageBox::Cancel) |
| { |
| m_wifiAction->setChecked (false); |
| } |
| else |
| { |
| |
| m_wifiAction->setText (tr("Disable AdHoc WiFI")); |
| |
| // create adhoc |
| m_executor.execute (Adhoc::CreateAdhoc); |
| } |
| } |
| else |
| { |
| m_wifiAction->setText (tr("Enable AdHoc WiFI")); |
| |
| // disable adhoc |
| m_executor.execute (Adhoc::DestroyAdhoc); |
| |
| // a trick in DestroyAdhoc ensures that WiFi will be reconnected to a default WiFi |
| } |
| #endif |
| } |
| |
| void |
| ChronoShareGui::onCheckForUpdates() |
| { |
| #ifdef SPARKLE_SUPPORTED |
| cout << "+++++++++++ trying to update +++++++ " << endl; |
| m_autoUpdate->checkForUpdates(); |
| cout << "+++++++++++ end trying to update +++++++ " << endl; |
| #endif |
| } |
| |
| void ChronoShareGui::setIcon() |
| { |
| // set the icon image |
| m_trayIcon->setIcon(QIcon(ICON_TRAY_FILE)); |
| } |
| |
| void ChronoShareGui::openSharedFolder() |
| { |
| filesystem::path realPathToFolder (m_dirPath.toStdString ()); |
| realPathToFolder /= m_sharedFolderName.toStdString (); |
| QString path = QDir::toNativeSeparators(realPathToFolder.string().c_str()); |
| QDesktopServices::openUrl(QUrl("file:///" + path)); |
| } |
| |
| void ChronoShareGui::openInWebBrowser() |
| { |
| QUrl url ("http://localhost:9001/"); |
| url.setFragment ("folderHistory&" |
| "user=" + QUrl::toPercentEncoding (m_username) + "&" |
| "folder=" + QUrl::toPercentEncoding (m_sharedFolderName)); |
| |
| // i give up. there is double encoding and I have no idea how to fight it... |
| QDesktopServices::openUrl (url); |
| } |
| |
| void ChronoShareGui::openFile() |
| { |
| // figure out who sent the signal |
| QAction *pAction = qobject_cast<QAction*>(sender()); |
| if (pAction->isEnabled()) |
| { |
| // we stored full path of the file in this toolTip field |
| #ifdef Q_WS_MAC |
| // we do some hack so we could show the file in Finder highlighted |
| QStringList args; |
| args << "-e"; |
| args << "tell application \"Finder\""; |
| args << "-e"; |
| args << "activate"; |
| args << "-e"; |
| args << "select POSIX file \"" + pAction->toolTip() + "/" + pAction->text() + "\""; |
| args << "-e"; |
| args << "end tell"; |
| QProcess::startDetached("osascript", args); |
| #else |
| // too bad qt couldn't do highlighting for Linux (or Mac) |
| QDesktopServices::openUrl(QUrl("file:///" + pAction->toolTip())); |
| #endif |
| } |
| } |
| |
| void ChronoShareGui::updateRecentFilesMenu() |
| { |
| for (int i = 0; i < 5; i++) |
| { |
| m_fileActions[i]->setVisible(false); |
| } |
| m_dispatcher->LookupRecentFileActions(boost::bind(&ChronoShareGui::checkFileAction, this, _1, _2, _3), 5); |
| } |
| |
| void ChronoShareGui::checkFileAction (const std::string &filename, int action, int index) |
| { |
| filesystem::path realPathToFolder (m_dirPath.toStdString ()); |
| realPathToFolder /= m_sharedFolderName.toStdString (); |
| realPathToFolder /= filename; |
| QString fullPath = realPathToFolder.string().c_str(); |
| QFileInfo fileInfo(fullPath); |
| if (fileInfo.exists()) |
| { |
| // This is a hack, we just use some field to store the path |
| m_fileActions[index]->setToolTip(fileInfo.absolutePath()); |
| m_fileActions[index]->setEnabled(true); |
| } |
| else |
| { |
| // after half an hour frustrating test and search around, |
| // I think it's the problem of Qt. |
| // According to the Qt doc, the action cannot be clicked |
| // and the area would be grey, but it didn't happen |
| // User can still trigger the action, and not greyed |
| // added check in SLOT to see if the action is "enalbed" |
| // as a remedy |
| // Give up at least for now |
| m_fileActions[index]->setEnabled(false); |
| // UPDATE, file not fetched yet |
| if (action == 0) |
| { |
| QFont font; |
| // supposed by change the font, didn't happen |
| font.setWeight(QFont::Light); |
| m_fileActions[index]->setFont(font); |
| m_fileActions[index]->setToolTip(tr("Fetching...")); |
| } |
| // DELETE |
| else |
| { |
| QFont font; |
| // supposed by change the font, didn't happen |
| font.setStrikeOut(true); |
| m_fileActions[index]->setFont(font); |
| m_fileActions[index]->setToolTip(tr("Deleted...")); |
| } |
| } |
| m_fileActions[index]->setText(fileInfo.fileName()); |
| m_fileActions[index]->setVisible(true); |
| } |
| |
| void ChronoShareGui::changeSettings() |
| { |
| QString oldUsername = m_username; |
| QString oldSharedFolderName = m_sharedFolderName; |
| |
| if(!editUsername->text().isEmpty()) |
| m_username = editUsername->text().trimmed(); |
| else |
| editUsername->setText(m_username); |
| |
| if(!editSharedFolder->text().isEmpty()) |
| m_sharedFolderName = editSharedFolder->text().trimmed(); |
| else |
| editSharedFolder->setText(m_sharedFolderName); |
| |
| if (m_username.isNull () || m_username=="" || |
| m_sharedFolderName.isNull () || m_sharedFolderName=="") |
| { |
| openMessageBox ("Error", |
| "Username and shared folder name cannot be empty"); |
| } |
| else |
| { |
| saveSettings(); |
| this->hide(); |
| |
| if (m_username != oldUsername || |
| oldSharedFolderName != m_sharedFolderName) |
| { |
| startBackend (true); // restart dispatcher/fswatcher |
| } |
| } |
| } |
| |
| void ChronoShareGui::openFileDialog() |
| { |
| while (true) |
| { |
| // prompt user for new directory |
| QString tempPath = QFileDialog::getExistingDirectory(this, tr("Choose ChronoShare folder"), |
| m_dirPath, QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); |
| if (tempPath.isNull ()) |
| { |
| if (m_dirPath.isNull ()) |
| { |
| openMessageBox ("Notice", "ChronoShare will use [" + QDir::homePath () + "/ChronoShare]. \n\nYou can change it later selecting \"Change Folder\" menu."); |
| |
| m_dirPath = QDir::homePath () + "/ChronoShare"; |
| editSharedFolderPath->setText (m_dirPath); |
| break; |
| } |
| else |
| { |
| // user just cancelled, no need to do anything else |
| return; |
| } |
| } |
| |
| QFileInfo qFileInfo (tempPath); |
| |
| if (!qFileInfo.isDir()) |
| { |
| openMessageBox ("Error", "Please select an existing folder or create a new one."); |
| continue; |
| } |
| |
| if (m_dirPath == tempPath) |
| { |
| // user selected the same directory, no need to do anything |
| return; |
| } |
| |
| m_dirPath = tempPath; |
| editSharedFolderPath->setText(m_dirPath); |
| break; |
| } |
| |
| _LOG_DEBUG ("Selected path: " << m_dirPath.toStdString ()); |
| // save settings |
| saveSettings (); |
| |
| startBackend (true); // restart dispatcher/fswatcher |
| } |
| |
| void ChronoShareGui::trayIconClicked (QSystemTrayIcon::ActivationReason reason) |
| { |
| // if double clicked, open shared folder |
| if(reason == QSystemTrayIcon::DoubleClick) |
| { |
| openSharedFolder(); |
| } |
| } |
| |
| void ChronoShareGui::viewSettings() |
| { |
| //simple for now |
| this->show(); |
| this->raise(); |
| this->activateWindow(); |
| } |
| |
| bool ChronoShareGui::loadSettings() |
| { |
| bool successful = true; |
| |
| // Load Settings |
| // QSettings settings(m_settingsFilePath, QSettings::NativeFormat); |
| QSettings settings (QSettings::NativeFormat, QSettings::UserScope, "irl.cs.ucla.edu", "ChronoShare"); |
| |
| // _LOG_DEBUG (lexical_cast<string> (settings.allKeys())); |
| |
| if(settings.contains("username")) |
| { |
| m_username = settings.value("username", "admin").toString(); |
| } |
| else |
| { |
| successful = false; |
| } |
| |
| editUsername->setText(m_username); |
| |
| if(settings.contains("sharedfoldername")) |
| { |
| m_sharedFolderName = settings.value("sharedfoldername", "shared").toString(); |
| } |
| else |
| { |
| successful = false; |
| } |
| |
| editSharedFolder->setText(m_sharedFolderName); |
| |
| if(settings.contains("dirPath")) |
| { |
| m_dirPath = settings.value("dirPath", QDir::homePath()).toString(); |
| } |
| else |
| { |
| successful = false; |
| } |
| |
| editSharedFolderPath->setText(m_dirPath); |
| |
| _LOG_DEBUG ("Found configured path: " << (successful ? m_dirPath.toStdString () : std::string("no"))); |
| |
| return successful; |
| } |
| |
| void ChronoShareGui::saveSettings() |
| { |
| // Save Settings |
| // QSettings settings(m_settingsFilePath, QSettings::NativeFormat); |
| QSettings settings (QSettings::NativeFormat, QSettings::UserScope, "irl.cs.ucla.edu", "ChronoShare"); |
| |
| settings.setValue("dirPath", m_dirPath); |
| settings.setValue("username", m_username); |
| settings.setValue("sharedfoldername", m_sharedFolderName); |
| } |
| |
| void ChronoShareGui::closeEvent(QCloseEvent* event) |
| { |
| _LOG_DEBUG ("Close Event"); |
| |
| if (m_username.isNull () || m_username == "" || |
| m_sharedFolderName.isNull () || m_sharedFolderName == "") |
| { |
| openMessageBox ("ChronoShare is not active", "Username and/or shared folder name are not set.\n\n To activate ChronoShare, open Settings menu and configure your username and shared folder name"); |
| } |
| |
| this->hide(); |
| event->ignore(); // don't let the event propagate to the base class |
| } |
| |
| #if WAF |
| #include "chronosharegui.moc" |
| #include "chronosharegui.cpp.moc" |
| #endif |