2020-03-25 16:08:12 +01:00
|
|
|
/* === This file is part of Calamares - <https://github.com/calamares> ===
|
|
|
|
*
|
|
|
|
* Copyright 2019-2020, Adriaan de Groot <groot@kde.org>
|
2020-04-03 10:18:07 +02:00
|
|
|
* Copyright 2020, Camilo Higuita <milo.h@aol.com> *
|
2020-03-25 16:08:12 +01:00
|
|
|
*
|
|
|
|
* Calamares is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Calamares 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 Calamares. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Config.h"
|
|
|
|
|
|
|
|
#include "SetKeyboardLayoutJob.h"
|
2020-03-26 15:57:02 +01:00
|
|
|
#include "keyboardwidget/keyboardpreview.h"
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
#include "GlobalStorage.h"
|
|
|
|
#include "JobQueue.h"
|
|
|
|
#include "utils/Logger.h"
|
|
|
|
#include "utils/Retranslator.h"
|
2020-06-23 11:13:55 +02:00
|
|
|
#include "utils/String.h"
|
2020-03-25 16:08:12 +01:00
|
|
|
|
2020-04-03 10:18:07 +02:00
|
|
|
#include <QApplication>
|
2020-03-26 15:57:02 +01:00
|
|
|
#include <QProcess>
|
2020-04-03 10:18:07 +02:00
|
|
|
#include <QTimer>
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardModelsModel::KeyboardModelsModel( QObject* parent )
|
|
|
|
: QAbstractListModel( parent )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
detectModels();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
KeyboardModelsModel::detectModels()
|
|
|
|
{
|
|
|
|
beginResetModel();
|
2020-03-26 15:57:02 +01:00
|
|
|
const auto models = KeyboardGlobal::getKeyboardModels();
|
2020-03-25 16:08:12 +01:00
|
|
|
auto index = -1;
|
2020-03-26 15:57:02 +01:00
|
|
|
for ( const auto& key : models.keys() )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
index++;
|
2020-03-26 15:57:02 +01:00
|
|
|
m_list << QMap< QString, QString > { { "label", key }, { "key", models[ key ] } };
|
|
|
|
if ( models[ key ] == "pc105" )
|
|
|
|
{
|
|
|
|
this->setCurrentIndex( index );
|
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
endResetModel();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
KeyboardModelsModel::refresh()
|
|
|
|
{
|
|
|
|
m_list.clear();
|
2020-03-26 15:57:02 +01:00
|
|
|
setCurrentIndex( -1 );
|
2020-03-25 16:08:12 +01:00
|
|
|
detectModels();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardModelsModel::data( const QModelIndex& index, int role ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( !index.isValid() )
|
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
return QVariant();
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
|
|
|
const auto item = m_list.at( index.row() );
|
|
|
|
return role == Qt::DisplayRole ? item[ "label" ] : item[ "key" ];
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardModelsModel::rowCount( const QModelIndex& ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
return m_list.count();
|
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
QHash< int, QByteArray >
|
2020-03-25 16:08:12 +01:00
|
|
|
KeyboardModelsModel::roleNames() const
|
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
return { { Qt::DisplayRole, "label" }, { Qt::UserRole, "key" } };
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
KeyboardModelsModel::currentIndex() const
|
|
|
|
{
|
|
|
|
return m_currentIndex;
|
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
const QMap< QString, QString >
|
|
|
|
KeyboardModelsModel::item( const int& index ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( index >= m_list.count() || index < 0 )
|
|
|
|
{
|
|
|
|
return QMap< QString, QString >();
|
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
return m_list.at( index );
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
const QMap< QString, QString >
|
|
|
|
KeyboardVariantsModel::item( const int& index ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( index >= m_list.count() || index < 0 )
|
|
|
|
{
|
|
|
|
return QMap< QString, QString >();
|
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
return m_list.at( index );
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardModelsModel::setCurrentIndex( const int& index )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( index >= m_list.count() || index < 0 )
|
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
return;
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
m_currentIndex = index;
|
2020-03-26 15:57:02 +01:00
|
|
|
emit currentIndexChanged( m_currentIndex );
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardVariantsModel::KeyboardVariantsModel( QObject* parent )
|
|
|
|
: QAbstractListModel( parent )
|
|
|
|
{
|
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
int
|
|
|
|
KeyboardVariantsModel::currentIndex() const
|
|
|
|
{
|
|
|
|
return m_currentIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardVariantsModel::setCurrentIndex( const int& index )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( index >= m_list.count() || index < 0 )
|
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
return;
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
m_currentIndex = index;
|
2020-03-26 15:57:02 +01:00
|
|
|
emit currentIndexChanged( m_currentIndex );
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardVariantsModel::data( const QModelIndex& index, int role ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( !index.isValid() )
|
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
return QVariant();
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
|
|
|
const auto item = m_list.at( index.row() );
|
|
|
|
return role == Qt::DisplayRole ? item[ "label" ] : item[ "key" ];
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardVariantsModel::rowCount( const QModelIndex& ) const
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
return m_list.count();
|
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
QHash< int, QByteArray >
|
2020-03-25 16:08:12 +01:00
|
|
|
KeyboardVariantsModel::roleNames() const
|
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
return { { Qt::DisplayRole, "label" }, { Qt::UserRole, "key" } };
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-03-26 15:57:02 +01:00
|
|
|
KeyboardVariantsModel::setVariants( QMap< QString, QString > variants )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
m_list.clear();
|
|
|
|
beginResetModel();
|
2020-03-26 15:57:02 +01:00
|
|
|
for ( const auto& key : variants.keys() )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
const auto item = QMap< QString, QString > { { "label", key }, { "key", variants[ key ] } };
|
2020-03-25 16:08:12 +01:00
|
|
|
m_list << item;
|
|
|
|
}
|
|
|
|
endResetModel();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns stringlist with suitable setxkbmap command-line arguments
|
|
|
|
* to set the given @p layout and @p variant.
|
|
|
|
*/
|
2020-03-26 15:57:02 +01:00
|
|
|
static inline QStringList
|
|
|
|
xkbmap_args( const QString& layout, const QString& variant )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
QStringList r { "-layout", layout };
|
2020-03-25 16:08:12 +01:00
|
|
|
if ( !variant.isEmpty() )
|
2020-03-26 15:57:02 +01:00
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
r << "-variant" << variant;
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
Config::Config( QObject* parent )
|
|
|
|
: QObject( parent )
|
|
|
|
, m_keyboardModelsModel( new KeyboardModelsModel( this ) )
|
|
|
|
, m_keyboardLayoutsModel( new KeyboardLayoutModel( this ) )
|
|
|
|
, m_keyboardVariantsModel( new KeyboardVariantsModel( this ) )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
m_setxkbmapTimer.setSingleShot( true );
|
|
|
|
|
|
|
|
// Connect signals and slots
|
2020-03-26 15:57:02 +01:00
|
|
|
connect( m_keyboardModelsModel, &KeyboardModelsModel::currentIndexChanged, [&]( int index ) {
|
|
|
|
m_selectedModel = m_keyboardModelsModel->item( index ).value( "key", "pc105" );
|
|
|
|
// Set Xorg keyboard model
|
|
|
|
QProcess::execute( "setxkbmap", QStringList { "-model", m_selectedModel } );
|
|
|
|
emit prettyStatusChanged();
|
|
|
|
} );
|
|
|
|
|
|
|
|
connect( m_keyboardLayoutsModel, &KeyboardLayoutModel::currentIndexChanged, [&]( int index ) {
|
|
|
|
m_selectedLayout = m_keyboardLayoutsModel->item( index ).first;
|
|
|
|
updateVariants( QPersistentModelIndex( m_keyboardLayoutsModel->index( index ) ) );
|
|
|
|
emit prettyStatusChanged();
|
|
|
|
} );
|
|
|
|
|
|
|
|
connect( m_keyboardVariantsModel, &KeyboardVariantsModel::currentIndexChanged, [&]( int index ) {
|
|
|
|
m_selectedVariant = m_keyboardVariantsModel->item( index )[ "key" ];
|
|
|
|
|
|
|
|
// Set Xorg keyboard layout
|
|
|
|
if ( m_setxkbmapTimer.isActive() )
|
|
|
|
{
|
|
|
|
m_setxkbmapTimer.stop();
|
|
|
|
m_setxkbmapTimer.disconnect( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
connect( &m_setxkbmapTimer, &QTimer::timeout, this, [=] {
|
|
|
|
QProcess::execute( "setxkbmap", xkbmap_args( m_selectedLayout, m_selectedVariant ) );
|
|
|
|
cDebug() << "xkbmap selection changed to: " << m_selectedLayout << '-' << m_selectedVariant;
|
|
|
|
m_setxkbmapTimer.disconnect( this );
|
|
|
|
} );
|
|
|
|
m_setxkbmapTimer.start( QApplication::keyboardInputInterval() );
|
|
|
|
emit prettyStatusChanged();
|
|
|
|
} );
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
KeyboardModelsModel*
|
|
|
|
Config::keyboardModels() const
|
|
|
|
{
|
|
|
|
return m_keyboardModelsModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyboardLayoutModel*
|
|
|
|
Config::keyboardLayouts() const
|
|
|
|
{
|
|
|
|
return m_keyboardLayoutsModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyboardVariantsModel*
|
|
|
|
Config::keyboardVariants() const
|
|
|
|
{
|
|
|
|
return m_keyboardVariantsModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
static QPersistentModelIndex
|
|
|
|
findLayout( const KeyboardLayoutModel* klm, const QString& currentLayout )
|
|
|
|
{
|
|
|
|
QPersistentModelIndex currentLayoutItem;
|
|
|
|
|
|
|
|
for ( int i = 0; i < klm->rowCount(); ++i )
|
|
|
|
{
|
|
|
|
QModelIndex idx = klm->index( i );
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( idx.isValid() && idx.data( KeyboardLayoutModel::KeyboardLayoutKeyRole ).toString() == currentLayout )
|
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
currentLayoutItem = idx;
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return currentLayoutItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Config::init()
|
|
|
|
{
|
|
|
|
//### Detect current keyboard layout and variant
|
|
|
|
QString currentLayout;
|
|
|
|
QString currentVariant;
|
|
|
|
QProcess process;
|
|
|
|
process.start( "setxkbmap", QStringList() << "-print" );
|
|
|
|
|
|
|
|
if ( process.waitForFinished() )
|
|
|
|
{
|
2020-06-23 11:13:55 +02:00
|
|
|
const QStringList list = QString( process.readAll() ).split( "\n", SplitSkipEmptyParts );
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
for ( QString line : list )
|
|
|
|
{
|
|
|
|
line = line.trimmed();
|
|
|
|
if ( !line.startsWith( "xkb_symbols" ) )
|
2020-03-26 15:57:02 +01:00
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
continue;
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
line = line.remove( "}" ).remove( "{" ).remove( ";" );
|
2020-03-25 16:08:12 +01:00
|
|
|
line = line.mid( line.indexOf( "\"" ) + 1 );
|
|
|
|
|
2020-06-23 11:13:55 +02:00
|
|
|
QStringList split = line.split( "+", SplitSkipEmptyParts );
|
2020-03-25 16:08:12 +01:00
|
|
|
if ( split.size() >= 2 )
|
|
|
|
{
|
|
|
|
currentLayout = split.at( 1 );
|
|
|
|
|
|
|
|
if ( currentLayout.contains( "(" ) )
|
|
|
|
{
|
|
|
|
int parenthesisIndex = currentLayout.indexOf( "(" );
|
2020-03-26 15:57:02 +01:00
|
|
|
currentVariant = currentLayout.mid( parenthesisIndex + 1 ).trimmed();
|
2020-03-25 16:08:12 +01:00
|
|
|
currentVariant.chop( 1 );
|
2020-03-26 15:57:02 +01:00
|
|
|
currentLayout = currentLayout.mid( 0, parenthesisIndex ).trimmed();
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//### Layouts and Variants
|
|
|
|
QPersistentModelIndex currentLayoutItem = findLayout( m_keyboardLayoutsModel, currentLayout );
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( !currentLayoutItem.isValid() && ( ( currentLayout == "latin" ) || ( currentLayout == "pc" ) ) )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
currentLayout = "us";
|
|
|
|
currentLayoutItem = findLayout( m_keyboardLayoutsModel, currentLayout );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set current layout and variant
|
|
|
|
if ( currentLayoutItem.isValid() )
|
|
|
|
{
|
|
|
|
m_keyboardLayoutsModel->setCurrentIndex( currentLayoutItem.row() );
|
|
|
|
updateVariants( currentLayoutItem, currentVariant );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Default to the first available layout if none was set
|
|
|
|
// Do this after unblocking signals so we get the default variant handling.
|
|
|
|
if ( !currentLayoutItem.isValid() && m_keyboardLayoutsModel->rowCount() > 0 )
|
2020-03-26 15:57:02 +01:00
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
m_keyboardLayoutsModel->setCurrentIndex( m_keyboardLayoutsModel->index( 0 ).row() );
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
Config::prettyStatus() const
|
|
|
|
{
|
|
|
|
QString status;
|
2020-03-26 15:57:02 +01:00
|
|
|
status += tr( "Set keyboard model to %1.<br/>" )
|
|
|
|
.arg( m_keyboardModelsModel->item( m_keyboardModelsModel->currentIndex() )[ "label" ] );
|
2020-03-25 16:08:12 +01:00
|
|
|
|
2020-03-26 15:57:02 +01:00
|
|
|
QString layout = m_keyboardLayoutsModel->item( m_keyboardLayoutsModel->currentIndex() ).second.description;
|
|
|
|
QString variant = m_keyboardVariantsModel->currentIndex() >= 0
|
|
|
|
? m_keyboardVariantsModel->item( m_keyboardVariantsModel->currentIndex() )[ "label" ]
|
|
|
|
: QString( "<default>" );
|
2020-03-25 16:08:12 +01:00
|
|
|
status += tr( "Set keyboard layout to %1/%2." ).arg( layout, variant );
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Calamares::JobList
|
2020-03-26 15:57:02 +01:00
|
|
|
Config::createJobs( const QString& xOrgConfFileName, const QString& convertedKeymapPath, bool writeEtcDefaultKeyboard )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
QList< Calamares::job_ptr > list;
|
|
|
|
|
|
|
|
Calamares::Job* j = new SetKeyboardLayoutJob( m_selectedModel,
|
|
|
|
m_selectedLayout,
|
|
|
|
m_selectedVariant,
|
|
|
|
xOrgConfFileName,
|
|
|
|
convertedKeymapPath,
|
|
|
|
writeEtcDefaultKeyboard );
|
|
|
|
list.append( Calamares::job_ptr( j ) );
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-03-26 15:57:02 +01:00
|
|
|
Config::guessLayout( const QStringList& langParts )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
bool foundCountryPart = false;
|
|
|
|
for ( auto countryPart = langParts.rbegin(); !foundCountryPart && countryPart != langParts.rend(); ++countryPart )
|
|
|
|
{
|
|
|
|
cDebug() << Logger::SubEntry << "looking for locale part" << *countryPart;
|
|
|
|
for ( int i = 0; i < m_keyboardLayoutsModel->rowCount(); ++i )
|
|
|
|
{
|
|
|
|
QModelIndex idx = m_keyboardLayoutsModel->index( i );
|
2020-03-26 15:57:02 +01:00
|
|
|
QString name
|
|
|
|
= idx.isValid() ? idx.data( KeyboardLayoutModel::KeyboardLayoutKeyRole ).toString() : QString();
|
2020-03-25 16:08:12 +01:00
|
|
|
if ( idx.isValid() && ( name.compare( *countryPart, Qt::CaseInsensitive ) == 0 ) )
|
|
|
|
{
|
|
|
|
cDebug() << Logger::SubEntry << "matched" << name;
|
|
|
|
m_keyboardLayoutsModel->setCurrentIndex( i );
|
|
|
|
foundCountryPart = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( foundCountryPart )
|
|
|
|
{
|
|
|
|
++countryPart;
|
|
|
|
if ( countryPart != langParts.rend() )
|
|
|
|
{
|
|
|
|
cDebug() << "Next level:" << *countryPart;
|
2020-03-26 15:57:02 +01:00
|
|
|
for ( int variantnumber = 0; variantnumber < m_keyboardVariantsModel->rowCount(); ++variantnumber )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( m_keyboardVariantsModel->item( variantnumber )[ "key" ].compare( *countryPart,
|
|
|
|
Qt::CaseInsensitive ) )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
m_keyboardVariantsModel->setCurrentIndex( variantnumber );
|
2020-03-26 15:57:02 +01:00
|
|
|
cDebug() << Logger::SubEntry << "matched variant"
|
|
|
|
<< m_keyboardVariantsModel->item( variantnumber )[ "key" ] << ' '
|
|
|
|
<< m_keyboardVariantsModel->item( variantnumber )[ "key" ];
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Config::onActivate()
|
|
|
|
{
|
|
|
|
/* Guessing a keyboard layout based on the locale means
|
|
|
|
* mapping between language identifiers in <lang>_<country>
|
|
|
|
* format to keyboard mappings, which are <country>_<layout>
|
|
|
|
* format; in addition, some countries have multiple languages,
|
|
|
|
* so fr_BE and nl_BE want different layouts (both Belgian)
|
|
|
|
* and sometimes the language-country name doesn't match the
|
|
|
|
* keyboard-country name at all (e.g. Ellas vs. Greek).
|
|
|
|
*
|
|
|
|
* This is a table of language-to-keyboard mappings. The
|
|
|
|
* language identifier is the key, while the value is
|
|
|
|
* a string that is used instead of the real language
|
|
|
|
* identifier in guessing -- so it should be something
|
|
|
|
* like <layout>_<country>.
|
|
|
|
*/
|
|
|
|
static constexpr char arabic[] = "ara";
|
2020-03-26 15:57:02 +01:00
|
|
|
static const auto specialCaseMap = QMap< std::string, std::string >( {
|
2020-03-25 16:08:12 +01:00
|
|
|
/* Most Arab countries map to Arabic keyboard (Default) */
|
|
|
|
{ "ar_AE", arabic },
|
|
|
|
{ "ar_BH", arabic },
|
|
|
|
{ "ar_DZ", arabic },
|
|
|
|
{ "ar_EG", arabic },
|
|
|
|
{ "ar_IN", arabic },
|
|
|
|
{ "ar_IQ", arabic },
|
|
|
|
{ "ar_JO", arabic },
|
|
|
|
{ "ar_KW", arabic },
|
|
|
|
{ "ar_LB", arabic },
|
|
|
|
{ "ar_LY", arabic },
|
|
|
|
/* Not Morocco: use layout ma */
|
|
|
|
{ "ar_OM", arabic },
|
|
|
|
{ "ar_QA", arabic },
|
|
|
|
{ "ar_SA", arabic },
|
|
|
|
{ "ar_SD", arabic },
|
|
|
|
{ "ar_SS", arabic },
|
|
|
|
/* Not Syria: use layout sy */
|
|
|
|
{ "ar_TN", arabic },
|
|
|
|
{ "ar_YE", arabic },
|
|
|
|
{ "ca_ES", "cat_ES" }, /* Catalan */
|
|
|
|
{ "as_ES", "ast_ES" }, /* Asturian */
|
2020-06-03 13:42:15 +02:00
|
|
|
{ "en_CA", "us" }, /* Canadian English */
|
2020-03-26 15:57:02 +01:00
|
|
|
{ "el_CY", "gr" }, /* Greek in Cyprus */
|
|
|
|
{ "el_GR", "gr" }, /* Greek in Greeze */
|
2020-03-25 16:08:12 +01:00
|
|
|
{ "ig_NG", "igbo_NG" }, /* Igbo in Nigeria */
|
|
|
|
{ "ha_NG", "hausa_NG" } /* Hausa */
|
|
|
|
} );
|
|
|
|
|
|
|
|
// Try to preselect a layout, depending on language and locale
|
|
|
|
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
|
|
|
|
QString lang = gs->value( "localeConf" ).toMap().value( "LANG" ).toString();
|
|
|
|
|
|
|
|
cDebug() << "Got locale language" << lang;
|
|
|
|
if ( !lang.isEmpty() )
|
|
|
|
{
|
|
|
|
// Chop off .codeset and @modifier
|
|
|
|
int index = lang.indexOf( '.' );
|
|
|
|
if ( index >= 0 )
|
2020-03-26 15:57:02 +01:00
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
lang.truncate( index );
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
index = lang.indexOf( '@' );
|
|
|
|
if ( index >= 0 )
|
2020-03-26 15:57:02 +01:00
|
|
|
{
|
2020-03-25 16:08:12 +01:00
|
|
|
lang.truncate( index );
|
2020-03-26 15:57:02 +01:00
|
|
|
}
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
lang.replace( '-', '_' ); // Normalize separators
|
|
|
|
}
|
|
|
|
if ( !lang.isEmpty() )
|
|
|
|
{
|
|
|
|
std::string lang_s = lang.toStdString();
|
|
|
|
if ( specialCaseMap.contains( lang_s ) )
|
|
|
|
{
|
|
|
|
QString newLang = QString::fromStdString( specialCaseMap.value( lang_s ) );
|
|
|
|
cDebug() << Logger::SubEntry << "special case language" << lang << "becomes" << newLang;
|
|
|
|
lang = newLang;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( !lang.isEmpty() )
|
|
|
|
{
|
2020-06-23 11:13:55 +02:00
|
|
|
const auto langParts = lang.split( '_', SplitSkipEmptyParts );
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
// Note that this his string is not fit for display purposes!
|
|
|
|
// It doesn't come from QLocale::nativeCountryName.
|
|
|
|
QString country = QLocale::countryToString( QLocale( lang ).country() );
|
|
|
|
cDebug() << Logger::SubEntry << "extracted country" << country << "::" << langParts;
|
|
|
|
|
|
|
|
guessLayout( langParts );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Config::finalize()
|
|
|
|
{
|
|
|
|
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
|
|
|
|
if ( !m_selectedLayout.isEmpty() )
|
|
|
|
{
|
|
|
|
gs->insert( "keyboardLayout", m_selectedLayout );
|
2020-03-26 15:57:02 +01:00
|
|
|
gs->insert( "keyboardVariant", m_selectedVariant ); //empty means default variant
|
2020-03-25 16:08:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//FIXME: also store keyboard model for something?
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-03-26 15:57:02 +01:00
|
|
|
Config::updateVariants( const QPersistentModelIndex& currentItem, QString currentVariant )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
const auto variants = m_keyboardLayoutsModel->item( currentItem.row() ).second.variants;
|
|
|
|
m_keyboardVariantsModel->setVariants( variants );
|
2020-03-25 16:08:12 +01:00
|
|
|
|
|
|
|
auto index = -1;
|
2020-03-26 15:57:02 +01:00
|
|
|
for ( const auto& key : variants.keys() )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
|
|
|
index++;
|
2020-03-26 15:57:02 +01:00
|
|
|
if ( variants[ key ] == currentVariant )
|
2020-03-25 16:08:12 +01:00
|
|
|
{
|
2020-03-26 15:57:02 +01:00
|
|
|
m_keyboardVariantsModel->setCurrentIndex( index );
|
2020-03-25 16:08:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|