/* This file is (c) 2008-2009 Konstantin Isakov * Part of GoldenDict. Licensed under GPLv3 or later, see the LICENSE file */ #include "sources.hh" #include #include #include #include Sources::Sources( QWidget * parent, Config::Paths const & paths, Config::SoundDirs const & soundDirs, Config::Hunspell const & hunspell, Config::Transliteration const & tr, Config::MediaWikis const & mediawikis ): QWidget( parent ), mediawikisModel( this, mediawikis ), pathsModel( this, paths ), soundDirsModel( this, soundDirs ), hunspellDictsModel( this, hunspell ) { ui.setupUi( this ); ui.mediaWikis->setTabKeyNavigation( true ); ui.mediaWikis->setModel( &mediawikisModel ); ui.mediaWikis->resizeColumnToContents( 0 ); ui.mediaWikis->resizeColumnToContents( 1 ); ui.mediaWikis->resizeColumnToContents( 2 ); ui.paths->setTabKeyNavigation( true ); ui.paths->setModel( &pathsModel ); fitPathsColumns(); ui.soundDirs->setTabKeyNavigation( true ); ui.soundDirs->setModel( &soundDirsModel ); fitSoundDirsColumns(); ui.hunspellPath->setText( hunspell.dictionariesPath ); ui.hunspellDictionaries->setTabKeyNavigation( true ); ui.hunspellDictionaries->setModel( &hunspellDictsModel ); fitHunspellDictsColumns(); ui.enableRussianTransliteration->setChecked( tr.enableRussianTransliteration ); ui.enableRomaji->setChecked( tr.romaji.enable ); ui.enableHepburn->setChecked( tr.romaji.enableHepburn ); ui.enableNihonShiki->setChecked( tr.romaji.enableNihonShiki ); ui.enableKunreiShiki->setChecked( tr.romaji.enableKunreiShiki ); ui.enableHiragana->setChecked( tr.romaji.enableHiragana ); ui.enableKatakana->setChecked( tr.romaji.enableKatakana ); } void Sources::fitPathsColumns() { ui.paths->resizeColumnToContents( 0 ); ui.paths->resizeColumnToContents( 1 ); } void Sources::fitSoundDirsColumns() { ui.soundDirs->resizeColumnToContents( 0 ); ui.soundDirs->resizeColumnToContents( 1 ); } void Sources::fitHunspellDictsColumns() { ui.hunspellDictionaries->resizeColumnToContents( 0 ); ui.hunspellDictionaries->resizeColumnToContents( 1 ); } void Sources::on_addPath_clicked() { QString dir = QFileDialog::getExistingDirectory( this, tr( "Choose a directory" ) ); if ( !dir.isEmpty() ) { pathsModel.addNewPath( dir ); fitPathsColumns(); } } void Sources::on_removePath_clicked() { QModelIndex current = ui.paths->currentIndex(); if ( current.isValid() && QMessageBox::question( this, tr( "Confirm removal" ), tr( "Remove directory %1 from the list?" ).arg( pathsModel.getCurrentPaths()[ current.row() ].path ), QMessageBox::Ok, QMessageBox::Cancel ) == QMessageBox::Ok ) { pathsModel.removePath( current.row() ); fitPathsColumns(); } } void Sources::on_addSoundDir_clicked() { QString dir = QFileDialog::getExistingDirectory( this, tr( "Choose a directory" ) ); if ( !dir.isEmpty() ) { soundDirsModel.addNewSoundDir( dir, QDir( dir ).dirName() ); fitSoundDirsColumns(); } } void Sources::on_removeSoundDir_clicked() { QModelIndex current = ui.soundDirs->currentIndex(); if ( current.isValid() && QMessageBox::question( this, tr( "Confirm removal" ), tr( "Remove directory %1 from the list?" ).arg( soundDirsModel.getCurrentSoundDirs()[ current.row() ].path ), QMessageBox::Ok, QMessageBox::Cancel ) == QMessageBox::Ok ) { soundDirsModel.removeSoundDir( current.row() ); fitSoundDirsColumns(); } } void Sources::on_changeHunspellPath_clicked() { QString dir = QFileDialog::getExistingDirectory( this, tr( "Choose a directory" ) ); if ( !dir.isEmpty() ) { ui.hunspellPath->setText( dir ); hunspellDictsModel.changePath( dir ); fitHunspellDictsColumns(); } } void Sources::on_addMediaWiki_clicked() { mediawikisModel.addNewWiki(); QModelIndex result = mediawikisModel.index( mediawikisModel.rowCount( QModelIndex() ) - 1, 1, QModelIndex() ); ui.mediaWikis->scrollTo( result ); //ui.mediaWikis->setCurrentIndex( result ); ui.mediaWikis->edit( result ); } void Sources::on_removeMediaWiki_clicked() { QModelIndex current = ui.mediaWikis->currentIndex(); if ( current.isValid() && QMessageBox::question( this, tr( "Confirm removal" ), tr( "Remove site %1 from the list?" ).arg( mediawikisModel.getCurrentWikis()[ current.row() ].name ), QMessageBox::Ok, QMessageBox::Cancel ) == QMessageBox::Ok ) mediawikisModel.removeWiki( current.row() ); } Config::Hunspell Sources::getHunspell() const { Config::Hunspell h; h.dictionariesPath = ui.hunspellPath->text(); h.enabledDictionaries = hunspellDictsModel.getEnabledDictionaries(); return h; } Config::Transliteration Sources::getTransliteration() const { Config::Transliteration tr; tr.enableRussianTransliteration = ui.enableRussianTransliteration->isChecked(); tr.romaji.enable = ui.enableRomaji->isChecked(); tr.romaji.enableHepburn = ui.enableHepburn->isChecked(); tr.romaji.enableNihonShiki = ui.enableNihonShiki->isChecked(); tr.romaji.enableKunreiShiki = ui.enableKunreiShiki->isChecked(); tr.romaji.enableHiragana = ui.enableHiragana->isChecked(); tr.romaji.enableKatakana = ui.enableKatakana->isChecked(); return tr; } ////////// MediaWikisModel MediaWikisModel::MediaWikisModel( QWidget * parent, Config::MediaWikis const & mediawikis_ ): QAbstractItemModel( parent ), mediawikis( mediawikis_ ) { } void MediaWikisModel::removeWiki( int index ) { beginRemoveRows( QModelIndex(), index, index ); mediawikis.erase( mediawikis.begin() + index ); endRemoveRows(); } void MediaWikisModel::addNewWiki() { Config::MediaWiki w; w.enabled = false; // That's quite some rng w.id = QString( QCryptographicHash::hash( QDateTime::currentDateTime().toString( "\"MediaWiki\"dd.MM.yyyy hh:mm:ss.zzz" ).toUtf8(), QCryptographicHash::Md5 ).toHex() ); w.url = "http://"; beginInsertRows( QModelIndex(), mediawikis.size(), mediawikis.size() ); mediawikis.push_back( w ); endInsertRows(); } QModelIndex MediaWikisModel::index( int row, int column, QModelIndex const & /*parent*/ ) const { return createIndex( row, column, 0 ); } QModelIndex MediaWikisModel::parent( QModelIndex const & /*parent*/ ) const { return QModelIndex(); } Qt::ItemFlags MediaWikisModel::flags( QModelIndex const & index ) const { Qt::ItemFlags result = QAbstractItemModel::flags( index ); if ( index.isValid() ) { if ( !index.column() ) result |= Qt::ItemIsUserCheckable; else result |= Qt::ItemIsEditable; } return result; } int MediaWikisModel::rowCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return mediawikis.size(); } int MediaWikisModel::columnCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return 3; } QVariant MediaWikisModel::headerData( int section, Qt::Orientation /*orientation*/, int role ) const { if ( role == Qt::DisplayRole ) switch( section ) { case 0: return tr( "Enabled" ); case 1: return tr( "Name" ); case 2: return tr( "Address" ); default: return QVariant(); } return QVariant(); } QVariant MediaWikisModel::data( QModelIndex const & index, int role ) const { if ( (unsigned)index.row() >= mediawikis.size() ) return QVariant(); if ( role == Qt::DisplayRole || role == Qt::EditRole ) { switch( index.column() ) { case 1: return mediawikis[ index.row() ].name; case 2: return mediawikis[ index.row() ].url; default: return QVariant(); } } if ( role == Qt::CheckStateRole && !index.column() ) return mediawikis[ index.row() ].enabled; return QVariant(); } bool MediaWikisModel::setData( QModelIndex const & index, const QVariant & value, int role ) { if ( (unsigned)index.row() >= mediawikis.size() ) return false; if ( role == Qt::CheckStateRole && !index.column() ) { //printf( "type = %d\n", (int)value.type() ); //printf( "value = %d\n", (int)value.toInt() ); // XXX it seems to be always passing Int( 2 ) as a value, so we just toggle mediawikis[ index.row() ].enabled = !mediawikis[ index.row() ].enabled; dataChanged( index, index ); return true; } if ( role == Qt::DisplayRole || role == Qt::EditRole ) switch( index.column() ) { case 1: mediawikis[ index.row() ].name = value.toString(); dataChanged( index, index ); return true; case 2: mediawikis[ index.row() ].url = value.toString(); dataChanged( index, index ); return true; default: return false; } return false; } ////////// PathsModel PathsModel::PathsModel( QWidget * parent, Config::Paths const & paths_ ): QAbstractItemModel( parent ), paths( paths_ ) { } void PathsModel::removePath( int index ) { beginRemoveRows( QModelIndex(), index, index ); paths.erase( paths.begin() + index ); endRemoveRows(); } void PathsModel::addNewPath( QString const & path ) { beginInsertRows( QModelIndex(), paths.size(), paths.size() ); paths.push_back( Config::Path( path, false ) ); endInsertRows(); } QModelIndex PathsModel::index( int row, int column, QModelIndex const & /*parent*/ ) const { return createIndex( row, column, 0 ); } QModelIndex PathsModel::parent( QModelIndex const & /*parent*/ ) const { return QModelIndex(); } Qt::ItemFlags PathsModel::flags( QModelIndex const & index ) const { Qt::ItemFlags result = QAbstractItemModel::flags( index ); if ( index.isValid() && index.column() == 1 ) result |= Qt::ItemIsUserCheckable; return result; } int PathsModel::rowCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return paths.size(); } int PathsModel::columnCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return 2; } QVariant PathsModel::headerData( int section, Qt::Orientation /*orientation*/, int role ) const { if ( role == Qt::DisplayRole ) switch( section ) { case 0: return tr( "Path" ); case 1: return tr( "Recursive" ); default: return QVariant(); } return QVariant(); } QVariant PathsModel::data( QModelIndex const & index, int role ) const { if ( (unsigned)index.row() >= paths.size() ) return QVariant(); if ( ( role == Qt::DisplayRole || role == Qt::EditRole ) && !index.column() ) return paths[ index.row() ].path; if ( role == Qt::CheckStateRole && index.column() == 1 ) return paths[ index.row() ].recursive; return QVariant(); } bool PathsModel::setData( QModelIndex const & index, const QVariant & /*value*/, int role ) { if ( (unsigned)index.row() >= paths.size() ) return false; if ( role == Qt::CheckStateRole && index.column() == 1 ) { paths[ index.row() ].recursive = !paths[ index.row() ].recursive; dataChanged( index, index ); return true; } return false; } ////////// SoundDirsModel SoundDirsModel::SoundDirsModel( QWidget * parent, Config::SoundDirs const & soundDirs_ ): QAbstractItemModel( parent ), soundDirs( soundDirs_ ) { } void SoundDirsModel::removeSoundDir( int index ) { beginRemoveRows( QModelIndex(), index, index ); soundDirs.erase( soundDirs.begin() + index ); endRemoveRows(); } void SoundDirsModel::addNewSoundDir( QString const & path, QString const & name ) { beginInsertRows( QModelIndex(), soundDirs.size(), soundDirs.size() ); soundDirs.push_back( Config::SoundDir( path, name ) ); endInsertRows(); } QModelIndex SoundDirsModel::index( int row, int column, QModelIndex const & /*parent*/ ) const { return createIndex( row, column, 0 ); } QModelIndex SoundDirsModel::parent( QModelIndex const & /*parent*/ ) const { return QModelIndex(); } Qt::ItemFlags SoundDirsModel::flags( QModelIndex const & index ) const { Qt::ItemFlags result = QAbstractItemModel::flags( index ); if ( index.isValid() && index.column() < 2 ) result |= Qt::ItemIsEditable; return result; } int SoundDirsModel::rowCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return soundDirs.size(); } int SoundDirsModel::columnCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return 2; } QVariant SoundDirsModel::headerData( int section, Qt::Orientation /*orientation*/, int role ) const { if ( role == Qt::DisplayRole ) switch( section ) { case 0: return tr( "Path" ); case 1: return tr( "Name" ); default: return QVariant(); } return QVariant(); } QVariant SoundDirsModel::data( QModelIndex const & index, int role ) const { if ( (unsigned)index.row() >= soundDirs.size() ) return QVariant(); if ( ( role == Qt::DisplayRole || role == Qt::EditRole ) && !index.column() ) return soundDirs[ index.row() ].path; if ( ( role == Qt::DisplayRole || role == Qt::EditRole ) && index.column() == 1 ) return soundDirs[ index.row() ].name; return QVariant(); } bool SoundDirsModel::setData( QModelIndex const & index, const QVariant & value, int role ) { if ( (unsigned)index.row() >= soundDirs.size() ) return false; if ( ( role == Qt::DisplayRole || role == Qt::EditRole ) && index.column() < 2 ) { if ( !index.column() ) soundDirs[ index.row() ].path = value.toString(); else soundDirs[ index.row() ].name = value.toString(); dataChanged( index, index ); return true; } return false; } ////////// HunspellDictsModel HunspellDictsModel::HunspellDictsModel( QWidget * parent, Config::Hunspell const & hunspell ): QAbstractItemModel( parent ), enabledDictionaries( hunspell.enabledDictionaries ) { changePath( hunspell.dictionariesPath ); } void HunspellDictsModel::changePath( QString const & newPath ) { dataFiles = HunspellMorpho::findDataFiles( newPath ); reset(); } QModelIndex HunspellDictsModel::index( int row, int column, QModelIndex const & /*parent*/ ) const { return createIndex( row, column, 0 ); } QModelIndex HunspellDictsModel::parent( QModelIndex const & /*parent*/ ) const { return QModelIndex(); } Qt::ItemFlags HunspellDictsModel::flags( QModelIndex const & index ) const { Qt::ItemFlags result = QAbstractItemModel::flags( index ); if ( index.isValid() ) { if ( !index.column() ) result |= Qt::ItemIsUserCheckable; } return result; } int HunspellDictsModel::rowCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return dataFiles.size(); } int HunspellDictsModel::columnCount( QModelIndex const & parent ) const { if ( parent.isValid() ) return 0; else return 2; } QVariant HunspellDictsModel::headerData( int section, Qt::Orientation /*orientation*/, int role ) const { if ( role == Qt::DisplayRole ) switch( section ) { case 0: return tr( "Enabled" ); case 1: return tr( "Name" ); default: return QVariant(); } return QVariant(); } QVariant HunspellDictsModel::data( QModelIndex const & index, int role ) const { if ( (unsigned)index.row() >= dataFiles.size() ) return QVariant(); if ( role == Qt::DisplayRole && index.column() == 1 ) return dataFiles[ index.row() ].dictName; if ( role == Qt::CheckStateRole && !index.column() ) { for( unsigned x = enabledDictionaries.size(); x--; ) { if ( enabledDictionaries[ x ] == dataFiles[ index.row() ].dictId ) return true; } return false; } return QVariant(); } bool HunspellDictsModel::setData( QModelIndex const & index, const QVariant & /*value*/, int role ) { if ( (unsigned)index.row() >= dataFiles.size() ) return false; if ( role == Qt::CheckStateRole && !index.column() ) { for( unsigned x = enabledDictionaries.size(); x--; ) { if ( enabledDictionaries[ x ] == dataFiles[ index.row() ].dictId ) { // Remove it now enabledDictionaries.erase( enabledDictionaries.begin() + x ); dataChanged( index, index ); return true; } } // Add it enabledDictionaries.push_back( dataFiles[ index.row() ].dictId ); dataChanged( index, index ); return true; } return false; } void Sources::on_rescan_clicked() { emit rescan(); }