diff --git a/CMakeLists.txt b/CMakeLists.txt index 41870512b..ef054eb4d 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -165,6 +165,8 @@ set(APP_SOURCES src/gui/formwelcome.cpp src/gui/formabout.cpp src/gui/shortcutcatcher.cpp + src/gui/shortcutbutton.cpp + src/gui/dynamicshortcutswidget.cpp # CORE sources. src/core/debugging.cpp @@ -172,7 +174,7 @@ set(APP_SOURCES src/core/systemfactory.cpp src/core/datetime.cpp src/core/localization.cpp - src/gui/shortcutbutton.cpp + src/core/dynamicshortcuts.cpp # Basic application sources. src/main.cpp @@ -195,6 +197,7 @@ set(APP_HEADERS src/gui/formabout.h src/gui/shortcutcatcher.h src/gui/shortcutbutton.h + src/gui/dynamicshortcutswidget.h # CORE headers. ) diff --git a/src/core/datetime.cpp b/src/core/datetime.cpp index a7af1018a..7bf7c523b 100644 --- a/src/core/datetime.cpp +++ b/src/core/datetime.cpp @@ -5,6 +5,9 @@ #include "core/datetime.h" +DateTime::DateTime() { +} + QDateTime DateTime::fromString(const QString &date_time) { QString date = date_time.simplified(); QDateTime dt; diff --git a/src/core/datetime.h b/src/core/datetime.h index d844c5269..58d595e8f 100644 --- a/src/core/datetime.h +++ b/src/core/datetime.h @@ -5,6 +5,9 @@ class DateTime { + private: + DateTime(); + public: // Returns QDatetime instance from input QString. // If parsing fails, then invalid QDateTime is returned. diff --git a/src/core/debugging.cpp b/src/core/debugging.cpp index f18d05301..3f8149adc 100644 --- a/src/core/debugging.cpp +++ b/src/core/debugging.cpp @@ -17,6 +17,9 @@ #endif +Debugging::Debugging() { +} + void Debugging::debugHandler(QtMsgType type, const QMessageLogContext &placement, const QString &message) { diff --git a/src/core/debugging.h b/src/core/debugging.h index e434bc2e8..5af9c14df 100644 --- a/src/core/debugging.h +++ b/src/core/debugging.h @@ -5,6 +5,9 @@ class Debugging { + private: + Debugging(); + public: // Specifies format of output console messages. // Macros: diff --git a/src/core/defs.h.in b/src/core/defs.h.in index 292926b51..20cb5dcea 100644 --- a/src/core/defs.h.in +++ b/src/core/defs.h.in @@ -17,6 +17,7 @@ #define APP_CFG_PATH "data/config/config.ini" #define APP_CFG_GUI "gui" #define APP_CFG_GEN "main" +#define APP_CFG_CUTS "keyboard" #define APP_DB_PATH "data/storage/database.db" #define APP_PREFIX "@CMAKE_INSTALL_PREFIX@" diff --git a/src/core/dynamicshortcuts.cpp b/src/core/dynamicshortcuts.cpp new file mode 100644 index 000000000..eac3c8b25 --- /dev/null +++ b/src/core/dynamicshortcuts.cpp @@ -0,0 +1,27 @@ +#include + +#include "core/dynamicshortcuts.h" +#include "core/settings.h" +#include "core/defs.h" + + +DynamicShortcuts::DynamicShortcuts() { +} + +void DynamicShortcuts::save(const QList actions) { + foreach (QAction *action, actions) { + Settings::getInstance()->setValue(APP_CFG_CUTS, + action->objectName(), + action->shortcut().toString(QKeySequence::NativeText)); + } +} + +void DynamicShortcuts::load(const QList actions) { + foreach (QAction *action, actions) { + QString shortcut_for_action = Settings::getInstance()->value(APP_CFG_CUTS, + action->objectName(), + action->shortcut().toString(QKeySequence::NativeText)).toString(); + action->setShortcut(QKeySequence::fromString(shortcut_for_action, + QKeySequence::NativeText)); + } +} diff --git a/src/core/dynamicshortcuts.h b/src/core/dynamicshortcuts.h new file mode 100644 index 000000000..ce7f85a39 --- /dev/null +++ b/src/core/dynamicshortcuts.h @@ -0,0 +1,22 @@ +#ifndef DYNAMICSHORTCUTS_H +#define DYNAMICSHORTCUTS_H + +#include + +class QAction; + +class DynamicShortcuts { + private: + DynamicShortcuts(); + + public: + // Checks the application settings and then initializes shortcut of + // each action from actions from the settings. + static void load(const QList actions); + + // Stores shortcut of each action from actions into the application + // settings. + static void save(const QList actions); +}; + +#endif // DYNAMICSHORTCUTS_H diff --git a/src/core/localization.cpp b/src/core/localization.cpp index 04bf761ef..d5879e963 100644 --- a/src/core/localization.cpp +++ b/src/core/localization.cpp @@ -41,12 +41,11 @@ void Localization::load() { "en").toString(); QTranslator qt_translator, app_translator; - // Load localizations and setup locales. + // Load localizations. if (app_translator.load(QString("rssguard_%1.qm").arg(locale_name), APP_LANG_PATH)) { qDebug("Application localization %s loaded successfully. Setting up locale.", qPrintable(locale_name)); - QLocale::setDefault(QLocale(locale_name)); } else { qDebug("Application localization %s was not loaded.", qPrintable(locale_name)); @@ -56,9 +55,11 @@ void Localization::load() { APP_LANG_PATH)) { qDebug("Qt localization %s loaded successfully. Setting up locale.", qPrintable(locale_name)); - QLocale::setDefault(QLocale(locale_name)); } else { qDebug("Qt localization %s was not loaded.", qPrintable(locale_name)); } + + // Setup locale. + QLocale::setDefault(QLocale(locale_name)); } diff --git a/src/core/localization.h b/src/core/localization.h index 9bf3348ed..5c674183e 100644 --- a/src/core/localization.h +++ b/src/core/localization.h @@ -13,7 +13,6 @@ struct Language { }; class Localization { - // TODO: Finish implementation of this class. private: Localization(); diff --git a/src/core/systemfactory.cpp b/src/core/systemfactory.cpp index 6afffba3e..eb3c7c414 100644 --- a/src/core/systemfactory.cpp +++ b/src/core/systemfactory.cpp @@ -16,9 +16,9 @@ SystemFactory::SystemFactory() { SystemFactory::AutoStartStatus SystemFactory::getAutoStartStatus() { // User registry way to auto-start the application on Windows. #if defined(Q_OS_WIN) - QSettings registr_key("HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Run", + QSettings registry_key("HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Run", QSettings::NativeFormat); - bool autostart_enabled = registr_key.value(APP_LOW_NAME, + bool autostart_enabled = registry_key.value(APP_LOW_NAME, "").toString().replace('\\', '/') == QtSingleApplication::applicationFilePath(); diff --git a/src/core/systemfactory.h b/src/core/systemfactory.h index 13329d71c..7cd2a6b23 100644 --- a/src/core/systemfactory.h +++ b/src/core/systemfactory.h @@ -7,6 +7,7 @@ class SystemFactory { SystemFactory(); public: + // Specifies possible states of auto-start functionality. enum AutoStartStatus { Enabled, Disabled, diff --git a/src/gui/dynamicshortcutswidget.cpp b/src/gui/dynamicshortcutswidget.cpp new file mode 100644 index 000000000..b98efc787 --- /dev/null +++ b/src/gui/dynamicshortcutswidget.cpp @@ -0,0 +1,45 @@ +#include +#include + +#include "gui/dynamicshortcutswidget.h" +#include "gui/shortcutcatcher.h" + + +DynamicShortcutsWidget::DynamicShortcutsWidget(QWidget *parent) : QWidget(parent) { + // Create layout for this control and set is as active. + m_layout = new QFormLayout(this); + m_layout->setMargin(0); + + setLayout(m_layout); +} + +DynamicShortcutsWidget::~DynamicShortcutsWidget() { + delete m_layout; +} + +void DynamicShortcutsWidget::updateShortcuts() { + foreach (ActionBinding binding, m_actionBindings) { + binding.first->setShortcut(binding.second->keySequence()); + } +} + +void DynamicShortcutsWidget::populate(const QList actions) { + m_actionBindings.clear(); + + foreach (QAction *action, actions) { + // Create shortcut catcher for this action and set default shortcut. + ShortcutCatcher *catcher = new ShortcutCatcher(this); + catcher->setKeySequence(action->shortcut()); + + // Store information for re-initialization of shortcuts + // of actions when widget gets "confirmed". + QPair new_binding; + new_binding.first = action; + new_binding.second = catcher; + + m_actionBindings << new_binding; + + // Add new catcher to our control. + m_layout->addRow(action->text(), catcher); + } +} diff --git a/src/gui/dynamicshortcutswidget.h b/src/gui/dynamicshortcutswidget.h new file mode 100644 index 000000000..431c0f9e7 --- /dev/null +++ b/src/gui/dynamicshortcutswidget.h @@ -0,0 +1,35 @@ +#ifndef DYNAMICSHORTCUTSOVERVIEW_H +#define DYNAMICSHORTCUTSOVERVIEW_H + +#include + + +class QFormLayout; +class ShortcutCatcher; + +typedef QPair ActionBinding; + +class DynamicShortcutsWidget : public QWidget { + Q_OBJECT + + public: + explicit DynamicShortcutsWidget(QWidget *parent = 0); + virtual ~DynamicShortcutsWidget(); + + // Updates shortcuts of all actions according to changes. + // NOTE: No access to settings is done here. + // Shortcuts are fetched from settings when applications starts + // and stored back to settings when application quits. + void updateShortcuts(); + + // Populates this widget with shortcut widgets for given actions. + // NOTE: This gets initial shortcut for each action from its properties, NOT from + // the application settings. + void populate(const QList actions); + + private: + QFormLayout *m_layout; + QList m_actionBindings; +}; + +#endif // DYNAMICSHORTCUTSOVERVIEW_H diff --git a/src/gui/formmain.cpp b/src/gui/formmain.cpp index 1df3e58f9..29c9435b7 100644 --- a/src/gui/formmain.cpp +++ b/src/gui/formmain.cpp @@ -42,6 +42,13 @@ QMenu *FormMain::getTrayMenu() { return m_trayMenu; } +QList FormMain::getActions() { + QList actions; + actions << m_ui->m_actionImport << m_ui->m_actionExport << + m_ui->m_actionSettings << m_ui->m_actionQuit; + return actions; +} + void FormMain::prepareMenus() { // Setup menu for tray icon. if (SystemTrayIcon::isSystemTrayAvailable()) { @@ -101,6 +108,7 @@ bool FormMain::event(QEvent *event) { if (event->type() == ThemeFactoryEvent::type()) { // Handle the change of icon theme. setupIcons(); + event->accept(); return true; } diff --git a/src/gui/formmain.h b/src/gui/formmain.h index e792ac8df..20fd8ab55 100644 --- a/src/gui/formmain.h +++ b/src/gui/formmain.h @@ -17,6 +17,10 @@ class FormMain : public QMainWindow { // Returns menu for the tray icon. QMenu *getTrayMenu(); + // Returns list of all globally available actions. + // NOTE: This is used for setting dynamic shortcuts for given actions. + QList getActions(); + static FormMain *getInstance(); protected: diff --git a/src/gui/formmain.ui b/src/gui/formmain.ui index 7724dd9b8..13d9cf3a2 100644 --- a/src/gui/formmain.ui +++ b/src/gui/formmain.ui @@ -206,16 +206,6 @@ - - - - 320 - 90 - 211 - 80 - - - @@ -241,7 +231,7 @@ - + &View @@ -253,7 +243,7 @@ - + @@ -262,21 +252,33 @@ &Import + + Ctrl+Shift+I + E&xport + + Ctrl+Shift+E + &Quit + + Ctrl+Shift+Q + &Settings + + Ctrl+Shift+S + @@ -284,14 +286,6 @@ - - - ShortcutCatcher - QWidget -
shortcutcatcher.h
- 1 -
-
diff --git a/src/gui/formsettings.cpp b/src/gui/formsettings.cpp index cb622eff7..fff323c04 100644 --- a/src/gui/formsettings.cpp +++ b/src/gui/formsettings.cpp @@ -9,6 +9,7 @@ #include "core/defs.h" #include "core/localization.h" #include "core/systemfactory.h" +#include "core/dynamicshortcuts.h" FormSettings::FormSettings(QWidget *parent) : QDialog(parent), m_ui(new Ui::FormSettings) { @@ -38,6 +39,7 @@ FormSettings::FormSettings(QWidget *parent) : QDialog(parent), m_ui(new Ui::Form // Load all settings. loadGeneral(); + loadShortcuts(); loadInterface(); loadLanguage(); } @@ -49,6 +51,7 @@ FormSettings::~FormSettings() { void FormSettings::saveSettings() { // Save all settings. saveGeneral(); + saveShortcuts(); saveInterface(); saveLanguage(); } @@ -111,6 +114,18 @@ void FormSettings::saveLanguage() { } } +void FormSettings::loadShortcuts() { + m_ui->m_shortcuts->populate(FormMain::getInstance()->getActions()); +} + +void FormSettings::saveShortcuts() { + // Update the actual shortcuts of some actions. + m_ui->m_shortcuts->updateShortcuts(); + + // Save new shortcuts to the settings. + DynamicShortcuts::save(FormMain::getInstance()->getActions()); +} + void FormSettings::loadGeneral() { // Load auto-start status. SystemFactory::AutoStartStatus autostart_status = SystemFactory::getAutoStartStatus(); diff --git a/src/gui/formsettings.h b/src/gui/formsettings.h index 9573c7487..af114c05c 100644 --- a/src/gui/formsettings.h +++ b/src/gui/formsettings.h @@ -30,6 +30,9 @@ class FormSettings : public QDialog { void loadLanguage(); void saveLanguage(); + + void loadShortcuts(); + void saveShortcuts(); private: Ui::FormSettings *m_ui; diff --git a/src/gui/formsettings.ui b/src/gui/formsettings.ui index 68a2fa08c..e5a793e2c 100644 --- a/src/gui/formsettings.ui +++ b/src/gui/formsettings.ui @@ -30,6 +30,11 @@ General + + + Keyboard shortcuts + + User interface @@ -45,7 +50,7 @@ - 2 + 1 @@ -70,6 +75,59 @@ + + + + 0 + + + 0 + + + 0 + + + 0 + + + + + QFrame::NoFrame + + + true + + + + + 0 + 0 + 506 + 368 + + + + + 0 + + + 0 + + + 0 + + + 0 + + + + + + + + + + @@ -90,7 +148,7 @@ QTabWidget::North - 1 + 0 @@ -254,6 +312,14 @@ + + + DynamicShortcutsWidget + QWidget +
dynamicshortcutswidget.h
+ 1 +
+
diff --git a/src/gui/shortcutbutton.h b/src/gui/shortcutbutton.h index 8d674c469..229261c32 100644 --- a/src/gui/shortcutbutton.h +++ b/src/gui/shortcutbutton.h @@ -1,3 +1,31 @@ +/****************************************************************************** +Copyright (c) 2010, Artem Galichkin +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of the nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL BE LIABLE FOR ANY +DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND +ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*******************************************************************************/ + #ifndef SHORTCUTBUTTON_H #define SHORTCUTBUTTON_H diff --git a/src/gui/shortcutcatcher.cpp b/src/gui/shortcutcatcher.cpp index 5af15cb4e..3dac75fda 100644 --- a/src/gui/shortcutcatcher.cpp +++ b/src/gui/shortcutcatcher.cpp @@ -34,9 +34,11 @@ SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #include "gui/themefactory.h" -ShortcutCatcher::ShortcutCatcher(QWidget *parent) : QWidget(parent) { +ShortcutCatcher::ShortcutCatcher(QWidget *parent) + : QWidget(parent) { // Setup layout of the control m_layout = new QHBoxLayout(this); + m_layout->setMargin(0); m_layout->setSpacing(1); // Create clear button. @@ -83,6 +85,8 @@ void ShortcutCatcher::doneRecording() { m_sequenceButton->setDown(false); updateDisplayShortcut(); + + emit keySequenceChanged(m_currentSequence); } void ShortcutCatcher::controlModifierlessTimout() { diff --git a/src/gui/shortcutcatcher.h b/src/gui/shortcutcatcher.h index 1a0a677bd..ca37bb5d3 100644 --- a/src/gui/shortcutcatcher.h +++ b/src/gui/shortcutcatcher.h @@ -1,3 +1,31 @@ +/****************************************************************************** +Copyright (c) 2010, Artem Galichkin +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of the nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL BE LIABLE FOR ANY +DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND +ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*******************************************************************************/ + #ifndef KEYSEQUENCECATCHER_H #define KEYSEQUENCECATCHER_H @@ -29,6 +57,9 @@ class ShortcutCatcher : public QWidget { void setKeySequence(const QKeySequence& key); void clearKeySequence(); + signals: + void keySequenceChanged(QKeySequence seguence); + private: QToolButton *m_clearButton; ShortcutButton *m_sequenceButton; diff --git a/src/gui/themefactory.cpp b/src/gui/themefactory.cpp index d408d20b4..116460fd3 100644 --- a/src/gui/themefactory.cpp +++ b/src/gui/themefactory.cpp @@ -43,9 +43,6 @@ void ThemeFactory::setupSearchPaths() { qPrintable(QIcon::themeSearchPaths().join(", "))); } -// TODO: Load currently selected "real" icon theme name instead of -// Qt default "", which stands for currently active system icon theme name on -// linux. On Windows, tiny "oxygen" version will be used as default icon theme. QString ThemeFactory::getSystemIconTheme() { #if defined(Q_OS_LINUX) // Empty string forces Qt to use icon theme from operating system. diff --git a/src/main.cpp b/src/main.cpp index 0a7f13d88..469e20840 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -7,6 +7,7 @@ #include "core/debugging.h" #include "core/localization.h" #include "core/settings.h" +#include "core/dynamicshortcuts.h" #include "gui/themefactory.h" #include "gui/formmain.h" #include "gui/formwelcome.h" @@ -14,6 +15,13 @@ #include "qtsingleapplication/qtsingleapplication.h" + +// TODO: Check if extra UNIX signalling is needed. +// Use header for it - signal function and catch SIGHUP +// void my_terminate (int param) { +// qApp->quit(); +// } + int main(int argc, char *argv[]) { //: Name of language, e.g. English. QObject::tr("LANG_NAME"); @@ -58,7 +66,7 @@ int main(int argc, char *argv[]) { // Add an extra path for non-system icon themes. ThemeFactory::setupSearchPaths(); - // Load localization and setup locale. + // Load localization and setup locale before any widget is constructed. Localization::load(); // These settings needs to be set before any QSettings object. @@ -74,6 +82,9 @@ int main(int argc, char *argv[]) { // Set correct information for main window. window.setWindowTitle(APP_LONG_NAME); + // Now is a good time to initialize dynamic keyboard shortcuts. + DynamicShortcuts::load(FormMain::getInstance()->getActions()); + // Display welcome dialog if application is launched for the first time. if (Settings::getInstance()->value(APP_CFG_GEN, "first_start", true).toBool()) { Settings::getInstance()->setValue(APP_CFG_GEN, "first_start", false);