/* This file is (c) 2008-2011 Konstantin Isakov * Part of GoldenDict. Licensed under GPLv3 or later, see the LICENSE file */ #ifndef __CONFIG_HH_INCLUDED__ #define __CONFIG_HH_INCLUDED__ #include #include #include #include #include #include #include "ex.hh" /// GoldenDict's configuration namespace Config { using std::vector; /// A path where to search for the dictionaries struct Path { QString path; bool recursive; Path(): recursive( false ) {} Path( QString const & path_, bool recursive_ ): path( path_ ), recursive( recursive_ ) {} bool operator == ( Path const & other ) const { return path == other.path && recursive == other.recursive; } }; /// A list of paths where to search for the dictionaries typedef vector< Path > Paths; /// A directory holding bunches of audiofiles, which is indexed into a separate /// dictionary. struct SoundDir { QString path, name; SoundDir() {} SoundDir( QString const & path_, QString const & name_ ): path( path_ ), name( name_ ) {} bool operator == ( SoundDir const & other ) const { return path == other.path && name == other.name; } }; /// A list of SoundDirs typedef vector< SoundDir > SoundDirs; struct DictionaryRef { QString id; // Dictionrary id, which is usually an md5 hash QString name; // Dictionary name, used to recover when its id changes DictionaryRef() {} DictionaryRef( QString const & id_, QString const & name_ ): id( id_ ), name( name_ ) {} bool operator == ( DictionaryRef const & other ) const { return id == other.id && name == other.name; } }; /// A dictionary group struct Group { unsigned id; QString name, icon; QByteArray iconData; QKeySequence shortcut; vector< DictionaryRef > dictionaries; Group(): id( 0 ) {} bool operator == ( Group const & other ) const { return id == other.id && name == other.name && icon == other.icon && dictionaries == other.dictionaries && shortcut == other.shortcut && iconData == iconData; } bool operator != ( Group const & other ) const { return ! operator == ( other ); } }; /// All the groups struct Groups: public vector< Group > { unsigned nextId; // Id to use to create the group next time Groups(): nextId( 1 ) {} }; /// Proxy server configuration struct ProxyServer { bool enabled; enum Type { Socks5 = 0, HttpConnect, HttpGet } type; QString host; unsigned port; QString user, password; ProxyServer(); }; // A hotkey -- currently qt modifiers plus one or two keys struct HotKey { Qt::KeyboardModifiers modifiers; int key1, key2; HotKey(); /// We use the first two keys of QKeySequence, with modifiers being stored /// in the first one. HotKey( QKeySequence const & ); QKeySequence toKeySequence() const; }; /// Various user preferences struct Preferences { QString interfaceLanguage; // Empty value corresponds to system default QString displayStyle; // Empty value corresponds to the default one bool newTabsOpenAfterCurrentOne; bool newTabsOpenInBackground; bool hideSingleTab; bool hideMenubar; bool enableTrayIcon; bool startToTray; bool closeToTray; bool autoStart; bool doubleClickTranslates; bool escKeyHidesMainWindow; bool enableMainWindowHotkey; HotKey mainWindowHotkey; bool enableClipboardHotkey; HotKey clipboardHotkey; bool enableScanPopup; bool startWithScanPopupOn; bool enableScanPopupModifiers; unsigned long scanPopupModifiers; // Combination of KeyboardState::Modifier bool scanPopupAltMode; // When you press modifier shortly after the selection unsigned scanPopupAltModeSecs; // Whether the word should be pronounced on page load, in main window/popup bool pronounceOnLoadMain, pronounceOnLoadPopup; QString audioPlaybackProgram; bool useExternalPlayer; bool useWindowsPlaySound; ProxyServer proxyServer; bool checkForNewReleases; bool disallowContentFromOtherSites; bool enableWebPlugins; qreal zoomFactor; int wordsZoomLevel; Preferences(); }; /// A MediaWiki network dictionary definition struct MediaWiki { QString id, name, url; bool enabled; MediaWiki(): enabled( false ) {} MediaWiki( QString const & id_, QString const & name_, QString const & url_, bool enabled_ ): id( id_ ), name( name_ ), url( url_ ), enabled( enabled_ ) {} bool operator == ( MediaWiki const & other ) const { return id == other.id && name == other.name && url == other.url && enabled == other.enabled; } }; /// Any website which can be queried though a simple template substitution struct WebSite { QString id, name, url; bool enabled; WebSite(): enabled( false ) {} WebSite( QString const & id_, QString const & name_, QString const & url_, bool enabled_ ): id( id_ ), name( name_ ), url( url_ ), enabled( enabled_ ) {} bool operator == ( WebSite const & other ) const { return id == other.id && name == other.name && url == other.url && enabled == other.enabled; } }; /// All the WebSites typedef vector< WebSite > WebSites; /// Hunspell configuration struct Hunspell { QString dictionariesPath; typedef vector< QString > Dictionaries; Dictionaries enabledDictionaries; bool operator == ( Hunspell const & other ) const { return dictionariesPath == other.dictionariesPath && enabledDictionaries == other.enabledDictionaries; } bool operator != ( Hunspell const & other ) const { return ! operator == ( other ); } }; /// All the MediaWikis typedef vector< MediaWiki > MediaWikis; /// Romaji transliteration configuration struct Romaji { bool enable; bool enableHepburn; bool enableNihonShiki; bool enableKunreiShiki; bool enableHiragana; bool enableKatakana; Romaji(); bool operator == ( Romaji const & other ) const { return enable == other.enable && enableHepburn == other.enableHepburn && enableNihonShiki == other.enableNihonShiki && enableKunreiShiki == other.enableKunreiShiki && enableHiragana == other.enableHiragana && enableKatakana == other.enableKatakana; } bool operator != ( Romaji const & other ) const { return ! operator == ( other ); } }; struct Transliteration { bool enableRussianTransliteration; bool enableGermanTransliteration; bool enableGreekTransliteration; Romaji romaji; bool operator == ( Transliteration const & other ) const { return enableRussianTransliteration == other.enableRussianTransliteration && romaji == other.romaji && enableGermanTransliteration == other.enableGermanTransliteration && enableGreekTransliteration == other.enableGreekTransliteration; } bool operator != ( Transliteration const & other ) const { return ! operator == ( other ); } Transliteration(): enableRussianTransliteration( false ), enableGermanTransliteration( false ), enableGreekTransliteration( false ) {} }; struct Forvo { bool enable; QString apiKey; QString languageCodes; Forvo(): enable( false ) {} bool operator == ( Forvo const & other ) const { return enable == other.enable && apiKey == other.apiKey && languageCodes == other.languageCodes; } bool operator != ( Forvo const & other ) const { return ! operator == ( other ); } }; struct Program { bool enabled; enum Type { Audio, PlainText, Html, PrefixMatch, MaxTypeValue } type; QString id, name, commandLine; Program(): enabled( false ) {} Program( bool enabled_, Type type_, QString const & id_, QString const & name_, QString const & commandLine_ ): enabled( enabled_ ), type( type_ ), id( id_ ), name( name_ ), commandLine( commandLine_ ) {} bool operator == ( Program const & other ) const { return enabled == other.enabled && type == other.type && name == other.name && commandLine == other.commandLine; } bool operator != ( Program const & other ) const { return ! operator == ( other ); } }; typedef vector< Program > Programs; /// Dictionaries which are temporarily disabled via the dictionary bar. typedef QSet< QString > MutedDictionaries; struct Class { Paths paths; SoundDirs soundDirs; Group dictionaryOrder; Group inactiveDictionaries; Groups groups; Preferences preferences; MediaWikis mediawikis; WebSites webSites; Hunspell hunspell; Transliteration transliteration; Forvo forvo; Programs programs; unsigned lastMainGroupId; // Last used group in main window unsigned lastPopupGroupId; // Last used group in popup window QByteArray popupWindowState; // Binary state saved by QMainWindow QByteArray popupWindowGeometry; // Geometry saved by QMainWindow bool pinPopupWindow; // Last pin status QByteArray mainWindowState; // Binary state saved by QMainWindow QByteArray mainWindowGeometry; // Geometry saved by QMainWindow MutedDictionaries mutedDictionaries; // Disabled via dictionary bar MutedDictionaries popupMutedDictionaries; // Disabled via dictionary bar in popup QDateTime timeForNewReleaseCheck; // Only effective if // preferences.checkForNewReleases is set QString skippedRelease; // Empty by default bool showingDictBarNames; bool usingSmallIconsInToolbars; unsigned short maxDictionaryRefsInContextMenu; Class(): lastMainGroupId( 0 ), lastPopupGroupId( 0 ), pinPopupWindow( false ), showingDictBarNames( false ), usingSmallIconsInToolbars( false ), maxDictionaryRefsInContextMenu( 20 ) {} }; /// Configuration-specific events. Some parts of the program need to react /// to specific changes in configuration. The object of this class is used /// to emit signals when such events happen -- and the listeners connect to /// them to be notified of them. /// This class is separate from the main Class since QObjects can't be copied. class Events: public QObject { Q_OBJECT public: /// Signals that the value of the mutedDictionaries has changed. /// This emits mutedDictionariesChanged() signal, so the subscribers will /// be notified. void signalMutedDictionariesChanged(); signals: /// THe value of the mutedDictionaries has changed. void mutedDictionariesChanged(); private: }; DEF_EX( exError, "Error with the program's configuration", std::exception ) DEF_EX( exCantUseHomeDir, "Can't use home directory to store GoldenDict preferences", exError ) DEF_EX( exCantUseIndexDir, "Can't use index directory to store GoldenDict index files", exError ) DEF_EX( exCantReadConfigFile, "Can't read the configuration file", exError ) DEF_EX( exCantWriteConfigFile, "Can't write the configuration file", exError ) DEF_EX( exMalformedConfigFile, "The configuration file is malformed", exError ) /// Loads the configuration, or creates the default one if none is present Class load() throw( exError ); /// Saves the configuration void save( Class const & ) throw( exError ); /// Returns the main configuration directory. QString getConfigDir() throw( exError ); /// Returns the index directory, where the indices are to be stored. QString getIndexDir() throw( exError ); /// Returns the filename of a .pid file which should store current pid of /// the process. QString getPidFileName() throw( exError ); /// Returns the filename of a history file which stores search history. QString getHistoryFileName() throw( exError ); /// Returns the user .css file name. QString getUserCssFileName() throw( exError ); /// Returns the user .css file name used for printing only. QString getUserCssPrintFileName() throw( exError ); /// Returns the user .css file name for the Qt interface customization. QString getUserQtCssFileName() throw( exError ); /// Returns the program's data dir. Under Linux that would be something like /// /usr/share/apps/goldendict, under Windows C:/Program Files/GoldenDict. QString getProgramDataDir() throw(); /// Returns the directory storing program localizized files (.qm). QString getLocDir() throw(); /// Returns true if the program is configured as a portable version. In that /// mode, all the settings and indices are kept in the program's directory. bool isPortableVersion() throw(); /// Returns directory with dictionaries for portable version. It is content/ /// in the application's directory. QString getPortableVersionDictionaryDir() throw(); /// Returns directory with morpgologies for portable version. It is /// content/morphology in the application's directory. QString getPortableVersionMorphoDir() throw(); } #endif