You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
977 lines
30 KiB
977 lines
30 KiB
15 years ago
|
#include <qlayout.h>
|
||
|
#include <qlabel.h>
|
||
|
#include <qcombobox.h>
|
||
|
#include <qtabwidget.h>
|
||
|
#include <qvgroupbox.h>
|
||
|
#include <qpushbutton.h>
|
||
|
#include <qlistview.h>
|
||
|
#include <qheader.h>
|
||
|
#include <qwhatsthis.h>
|
||
|
#include <qcheckbox.h>
|
||
|
#include <qradiobutton.h>
|
||
|
#include <qlineedit.h>
|
||
|
#include <qlistview.h>
|
||
|
#include <qbuttongroup.h>
|
||
|
#include <qspinbox.h>
|
||
|
|
||
|
#include <kkeydialog.h>
|
||
|
#include <kglobal.h>
|
||
|
#include <kconfig.h>
|
||
|
#include <klocale.h>
|
||
|
#include <kstandarddirs.h>
|
||
|
#include <kdebug.h>
|
||
|
#include <kapplication.h>
|
||
|
#include <kiconloader.h>
|
||
|
|
||
|
#include "extension.h"
|
||
|
#include "kxkbconfig.h"
|
||
|
#include "rules.h"
|
||
|
#include "pixmap.h"
|
||
|
#include "kcmmisc.h"
|
||
|
#include "kcmlayoutwidget.h"
|
||
|
|
||
|
#include "kcmlayout.h"
|
||
|
#include "kcmlayout.moc"
|
||
|
|
||
|
|
||
|
enum {
|
||
|
LAYOUT_COLUMN_FLAG = 0,
|
||
|
LAYOUT_COLUMN_NAME = 1,
|
||
|
LAYOUT_COLUMN_MAP = 2,
|
||
|
LAYOUT_COLUMN_VARIANT = 3,
|
||
|
LAYOUT_COLUMN_INCLUDE = 4,
|
||
|
LAYOUT_COLUMN_DISPLAY_NAME = 5,
|
||
|
SRC_LAYOUT_COLUMN_COUNT = 3,
|
||
|
DST_LAYOUT_COLUMN_COUNT = 6
|
||
|
};
|
||
|
|
||
|
static const QString DEFAULT_VARIANT_NAME("<default>");
|
||
|
|
||
|
|
||
|
class OptionListItem : public QCheckListItem
|
||
|
{
|
||
|
public:
|
||
|
|
||
|
OptionListItem( OptionListItem *parent, const QString &text, Type tt,
|
||
|
const QString &optionName );
|
||
|
OptionListItem( QListView *parent, const QString &text, Type tt,
|
||
|
const QString &optionName );
|
||
|
~OptionListItem() {}
|
||
|
|
||
|
QString optionName() const { return m_OptionName; }
|
||
|
|
||
|
OptionListItem *findChildItem( const QString& text );
|
||
|
|
||
|
protected:
|
||
|
QString m_OptionName;
|
||
|
};
|
||
|
|
||
|
|
||
|
static QString lookupLocalized(const QDict<char> &dict, const QString& text)
|
||
|
{
|
||
|
QDictIterator<char> it(dict);
|
||
|
while (it.current())
|
||
|
{
|
||
|
if ( i18n(it.current()) == text )
|
||
|
return it.currentKey();
|
||
|
++it;
|
||
|
}
|
||
|
|
||
|
return QString::null;
|
||
|
}
|
||
|
|
||
|
static QListViewItem* copyLVI(const QListViewItem* src, QListView* parent)
|
||
|
{
|
||
|
QListViewItem* ret = new QListViewItem(parent);
|
||
|
for(int i = 0; i < SRC_LAYOUT_COLUMN_COUNT; i++)
|
||
|
{
|
||
|
ret->setText(i, src->text(i));
|
||
|
if ( src->pixmap(i) )
|
||
|
ret->setPixmap(i, *src->pixmap(i));
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
LayoutConfig::LayoutConfig(QWidget *parent, const char *name)
|
||
|
: KCModule(parent, name),
|
||
|
m_rules(NULL)
|
||
|
{
|
||
|
QVBoxLayout *main = new QVBoxLayout(this, 0, KDialog::spacingHint());
|
||
|
|
||
|
widget = new LayoutConfigWidget(this, "widget");
|
||
|
main->addWidget(widget);
|
||
|
|
||
|
connect( widget->chkEnable, SIGNAL( toggled( bool )), this, SLOT(changed()));
|
||
|
connect( widget->chkShowSingle, SIGNAL( toggled( bool )), this, SLOT(changed()));
|
||
|
connect( widget->chkShowFlag, SIGNAL( toggled( bool )), this, SLOT(changed()));
|
||
|
connect( widget->comboModel, SIGNAL(activated(int)), this, SLOT(changed()));
|
||
|
|
||
|
connect( widget->listLayoutsSrc, SIGNAL(doubleClicked(QListViewItem*,const QPoint&, int)),
|
||
|
this, SLOT(add()));
|
||
|
connect( widget->btnAdd, SIGNAL(clicked()), this, SLOT(add()));
|
||
|
connect( widget->btnRemove, SIGNAL(clicked()), this, SLOT(remove()));
|
||
|
|
||
|
connect( widget->comboVariant, SIGNAL(activated(int)), this, SLOT(changed()));
|
||
|
connect( widget->comboVariant, SIGNAL(activated(int)), this, SLOT(variantChanged()));
|
||
|
connect( widget->listLayoutsDst, SIGNAL(selectionChanged(QListViewItem *)),
|
||
|
this, SLOT(layoutSelChanged(QListViewItem *)));
|
||
|
|
||
|
connect( widget->editDisplayName, SIGNAL(textChanged(const QString&)), this, SLOT(displayNameChanged(const QString&)));
|
||
|
|
||
|
connect( widget->chkLatin, SIGNAL(clicked()), this, SLOT(changed()));
|
||
|
connect( widget->chkLatin, SIGNAL(clicked()), this, SLOT(latinChanged()));
|
||
|
|
||
|
widget->btnUp->setIconSet(SmallIconSet("1uparrow"));
|
||
|
connect( widget->btnUp, SIGNAL(clicked()), this, SLOT(changed()));
|
||
|
connect( widget->btnUp, SIGNAL(clicked()), this, SLOT(moveUp()));
|
||
|
widget->btnDown->setIconSet(SmallIconSet("1downarrow"));
|
||
|
connect( widget->btnDown, SIGNAL(clicked()), this, SLOT(changed()));
|
||
|
connect( widget->btnDown, SIGNAL(clicked()), this, SLOT(moveDown()));
|
||
|
|
||
|
connect( widget->grpSwitching, SIGNAL( clicked( int ) ), SLOT(changed()));
|
||
|
|
||
|
connect( widget->chkEnableSticky, SIGNAL(toggled(bool)), this, SLOT(changed()));
|
||
|
connect( widget->spinStickyDepth, SIGNAL(valueChanged(int)), this, SLOT(changed()));
|
||
|
|
||
|
widget->listLayoutsSrc->setColumnText(LAYOUT_COLUMN_FLAG, "");
|
||
|
widget->listLayoutsDst->setColumnText(LAYOUT_COLUMN_FLAG, "");
|
||
|
widget->listLayoutsDst->setColumnText(LAYOUT_COLUMN_INCLUDE, "");
|
||
|
// widget->listLayoutsDst->setColumnText(LAYOUT_COLUMN_DISPLAY_NAME, "");
|
||
|
|
||
|
widget->listLayoutsSrc->setColumnWidth(LAYOUT_COLUMN_FLAG, 28);
|
||
|
widget->listLayoutsDst->setColumnWidth(LAYOUT_COLUMN_FLAG, 28);
|
||
|
|
||
|
widget->listLayoutsDst->header()->setResizeEnabled(FALSE, LAYOUT_COLUMN_INCLUDE);
|
||
|
widget->listLayoutsDst->header()->setResizeEnabled(FALSE, LAYOUT_COLUMN_DISPLAY_NAME);
|
||
|
widget->listLayoutsDst->setColumnWidthMode(LAYOUT_COLUMN_INCLUDE, QListView::Manual);
|
||
|
widget->listLayoutsDst->setColumnWidth(LAYOUT_COLUMN_INCLUDE, 0);
|
||
|
// widget->listLayoutsDst->setColumnWidth(LAYOUT_COLUMN_DISPLAY_NAME, 0);
|
||
|
|
||
|
widget->listLayoutsDst->setSorting(-1);
|
||
|
#if 0
|
||
|
widget->listLayoutsDst->setResizeMode(QListView::LastColumn);
|
||
|
widget->listLayoutsSrc->setResizeMode(QListView::LastColumn);
|
||
|
#endif
|
||
|
widget->listLayoutsDst->setResizeMode(QListView::LastColumn);
|
||
|
|
||
|
//Read rules - we _must_ read _before_ creating xkb-options comboboxes
|
||
|
loadRules();
|
||
|
|
||
|
makeOptionsTab();
|
||
|
|
||
|
load();
|
||
|
}
|
||
|
|
||
|
|
||
|
LayoutConfig::~LayoutConfig()
|
||
|
{
|
||
|
delete m_rules;
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::load()
|
||
|
{
|
||
|
m_kxkbConfig.load(KxkbConfig::LOAD_ALL);
|
||
|
|
||
|
initUI();
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::initUI() {
|
||
|
const char* modelName = m_rules->models()[m_kxkbConfig.m_model];
|
||
|
if( modelName == NULL )
|
||
|
modelName = DEFAULT_MODEL;
|
||
|
|
||
|
widget->comboModel->setCurrentText(i18n(modelName));
|
||
|
|
||
|
QValueList<LayoutUnit> otherLayouts = m_kxkbConfig.m_layouts;
|
||
|
widget->listLayoutsDst->clear();
|
||
|
// to optimize we should have gone from it.end to it.begin
|
||
|
QValueList<LayoutUnit>::ConstIterator it;
|
||
|
for (it = otherLayouts.begin(); it != otherLayouts.end(); ++it ) {
|
||
|
QListViewItemIterator src_it( widget->listLayoutsSrc );
|
||
|
LayoutUnit layoutUnit = *it;
|
||
|
|
||
|
for ( ; src_it.current(); ++src_it ) {
|
||
|
QListViewItem* srcItem = src_it.current();
|
||
|
|
||
|
if ( layoutUnit.layout == src_it.current()->text(LAYOUT_COLUMN_MAP) ) { // check if current config knows about this layout
|
||
|
QListViewItem* newItem = copyLVI(srcItem, widget->listLayoutsDst);
|
||
|
|
||
|
newItem->setText(LAYOUT_COLUMN_VARIANT, layoutUnit.variant);
|
||
|
newItem->setText(LAYOUT_COLUMN_INCLUDE, layoutUnit.includeGroup);
|
||
|
newItem->setText(LAYOUT_COLUMN_DISPLAY_NAME, layoutUnit.displayName);
|
||
|
widget->listLayoutsDst->insertItem(newItem);
|
||
|
newItem->moveItem(widget->listLayoutsDst->lastItem());
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// display KXKB switching options
|
||
|
widget->chkShowSingle->setChecked(m_kxkbConfig.m_showSingle);
|
||
|
widget->chkShowFlag->setChecked(m_kxkbConfig.m_showFlag);
|
||
|
|
||
|
widget->chkEnableOptions->setChecked( m_kxkbConfig.m_enableXkbOptions );
|
||
|
widget->checkResetOld->setChecked(m_kxkbConfig.m_resetOldOptions);
|
||
|
|
||
|
switch( m_kxkbConfig.m_switchingPolicy ) {
|
||
|
default:
|
||
|
case SWITCH_POLICY_GLOBAL:
|
||
|
widget->grpSwitching->setButton(0);
|
||
|
break;
|
||
|
case SWITCH_POLICY_WIN_CLASS:
|
||
|
widget->grpSwitching->setButton(1);
|
||
|
break;
|
||
|
case SWITCH_POLICY_WINDOW:
|
||
|
widget->grpSwitching->setButton(2);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
widget->chkEnableSticky->setChecked(m_kxkbConfig.m_stickySwitching);
|
||
|
widget->spinStickyDepth->setEnabled(m_kxkbConfig.m_stickySwitching);
|
||
|
widget->spinStickyDepth->setValue( m_kxkbConfig.m_stickySwitchingDepth);
|
||
|
|
||
|
updateStickyLimit();
|
||
|
|
||
|
widget->chkEnable->setChecked( m_kxkbConfig.m_useKxkb );
|
||
|
widget->grpLayouts->setEnabled( m_kxkbConfig.m_useKxkb );
|
||
|
widget->optionsFrame->setEnabled( m_kxkbConfig.m_useKxkb );
|
||
|
|
||
|
// display xkb options
|
||
|
QStringList options = QStringList::split(',', m_kxkbConfig.m_options);
|
||
|
for (QStringList::ConstIterator it = options.begin(); it != options.end(); ++it)
|
||
|
{
|
||
|
QString option = *it;
|
||
|
QString optionKey = option.mid(0, option.find(':'));
|
||
|
QString optionName = m_rules->options()[option];
|
||
|
OptionListItem *item = m_optionGroups[i18n(optionKey.latin1())];
|
||
|
|
||
|
if (item != NULL) {
|
||
|
OptionListItem *child = item->findChildItem( option );
|
||
|
|
||
|
if ( child )
|
||
|
child->setState( QCheckListItem::On );
|
||
|
else
|
||
|
kdDebug() << "load: Unknown option: " << option << endl;
|
||
|
}
|
||
|
else {
|
||
|
kdDebug() << "load: Unknown option group: " << optionKey << " of " << option << endl;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
updateOptionsCommand();
|
||
|
emit KCModule::changed( false );
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::save()
|
||
|
{
|
||
|
QString model = lookupLocalized(m_rules->models(), widget->comboModel->currentText());
|
||
|
m_kxkbConfig.m_model = model;
|
||
|
|
||
|
m_kxkbConfig.m_enableXkbOptions = widget->chkEnableOptions->isChecked();
|
||
|
m_kxkbConfig.m_resetOldOptions = widget->checkResetOld->isChecked();
|
||
|
m_kxkbConfig.m_options = createOptionString();
|
||
|
|
||
|
QListViewItem *item = widget->listLayoutsDst->firstChild();
|
||
|
QValueList<LayoutUnit> layouts;
|
||
|
while (item) {
|
||
|
QString layout = item->text(LAYOUT_COLUMN_MAP);
|
||
|
QString variant = item->text(LAYOUT_COLUMN_VARIANT);
|
||
|
QString includes = item->text(LAYOUT_COLUMN_INCLUDE);
|
||
|
QString displayName = item->text(LAYOUT_COLUMN_DISPLAY_NAME);
|
||
|
|
||
|
LayoutUnit layoutUnit(layout, variant);
|
||
|
layoutUnit.includeGroup = includes;
|
||
|
layoutUnit.displayName = displayName;
|
||
|
layouts.append( layoutUnit );
|
||
|
|
||
|
item = item->nextSibling();
|
||
|
kdDebug() << "To save: layout " << layoutUnit.toPair()
|
||
|
<< ", inc: " << layoutUnit.includeGroup
|
||
|
<< ", disp: " << layoutUnit.displayName << endl;
|
||
|
}
|
||
|
m_kxkbConfig.m_layouts = layouts;
|
||
|
|
||
|
if( m_kxkbConfig.m_layouts.count() == 0 ) {
|
||
|
m_kxkbConfig.m_layouts.append(LayoutUnit(DEFAULT_LAYOUT_UNIT));
|
||
|
widget->chkEnable->setChecked(false);
|
||
|
}
|
||
|
|
||
|
m_kxkbConfig.m_useKxkb = widget->chkEnable->isChecked();
|
||
|
m_kxkbConfig.m_showSingle = widget->chkShowSingle->isChecked();
|
||
|
m_kxkbConfig.m_showFlag = widget->chkShowFlag->isChecked();
|
||
|
|
||
|
int modeId = widget->grpSwitching->id(widget->grpSwitching->selected());
|
||
|
switch( modeId ) {
|
||
|
default:
|
||
|
case 0:
|
||
|
m_kxkbConfig.m_switchingPolicy = SWITCH_POLICY_GLOBAL;
|
||
|
break;
|
||
|
case 1:
|
||
|
m_kxkbConfig.m_switchingPolicy = SWITCH_POLICY_WIN_CLASS;
|
||
|
break;
|
||
|
case 2:
|
||
|
m_kxkbConfig.m_switchingPolicy = SWITCH_POLICY_WINDOW;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
m_kxkbConfig.m_stickySwitching = widget->chkEnableSticky->isChecked();
|
||
|
m_kxkbConfig.m_stickySwitchingDepth = widget->spinStickyDepth->value();
|
||
|
|
||
|
m_kxkbConfig.save();
|
||
|
|
||
|
kapp->kdeinitExec("kxkb");
|
||
|
emit KCModule::changed( false );
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::updateStickyLimit()
|
||
|
{
|
||
|
int layoutsCnt = widget->listLayoutsDst->childCount();
|
||
|
int maxDepth = layoutsCnt - 1;
|
||
|
|
||
|
if( maxDepth < 2 ) {
|
||
|
maxDepth = 2;
|
||
|
}
|
||
|
|
||
|
widget->spinStickyDepth->setMaxValue(maxDepth);
|
||
|
/* if( value > maxDepth )
|
||
|
setValue(maxDepth);*/
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::add()
|
||
|
{
|
||
|
QListViewItem* sel = widget->listLayoutsSrc->selectedItem();
|
||
|
if( sel == 0 )
|
||
|
return;
|
||
|
|
||
|
// Create a copy of the sel widget, as one might add the same layout more
|
||
|
// than one time, with different variants.
|
||
|
QListViewItem* toadd = copyLVI(sel, widget->listLayoutsDst);
|
||
|
|
||
|
widget->listLayoutsDst->insertItem(toadd);
|
||
|
if( widget->listLayoutsDst->childCount() > 1 )
|
||
|
toadd->moveItem(widget->listLayoutsDst->lastItem());
|
||
|
// disabling temporary: does not work reliable in Qt :(
|
||
|
// widget->listLayoutsDst->setSelected(sel, true);
|
||
|
// layoutSelChanged(sel);
|
||
|
|
||
|
updateStickyLimit();
|
||
|
changed();
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::remove()
|
||
|
{
|
||
|
QListViewItem* sel = widget->listLayoutsDst->selectedItem();
|
||
|
QListViewItem* newSel = 0;
|
||
|
|
||
|
if( sel == 0 )
|
||
|
return;
|
||
|
|
||
|
if( sel->itemBelow() )
|
||
|
newSel = sel->itemBelow();
|
||
|
else
|
||
|
if( sel->itemAbove() )
|
||
|
newSel = sel->itemAbove();
|
||
|
|
||
|
delete sel;
|
||
|
if( newSel )
|
||
|
widget->listLayoutsSrc->setSelected(newSel, true);
|
||
|
layoutSelChanged(newSel);
|
||
|
|
||
|
updateStickyLimit();
|
||
|
changed();
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::moveUp()
|
||
|
{
|
||
|
QListViewItem* sel = widget->listLayoutsDst->selectedItem();
|
||
|
if( sel == 0 || sel->itemAbove() == 0 )
|
||
|
return;
|
||
|
|
||
|
if( sel->itemAbove()->itemAbove() == 0 ) {
|
||
|
widget->listLayoutsDst->takeItem(sel);
|
||
|
widget->listLayoutsDst->insertItem(sel);
|
||
|
widget->listLayoutsDst->setSelected(sel, true);
|
||
|
}
|
||
|
else
|
||
|
sel->moveItem(sel->itemAbove()->itemAbove());
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::moveDown()
|
||
|
{
|
||
|
QListViewItem* sel = widget->listLayoutsDst->selectedItem();
|
||
|
if( sel == 0 || sel->itemBelow() == 0 )
|
||
|
return;
|
||
|
|
||
|
sel->moveItem(sel->itemBelow());
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::variantChanged()
|
||
|
{
|
||
|
QListViewItem* selLayout = widget->listLayoutsDst->selectedItem();
|
||
|
if( selLayout == NULL ) {
|
||
|
widget->comboVariant->clear();
|
||
|
widget->comboVariant->setEnabled(false);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
QString selectedVariant = widget->comboVariant->currentText();
|
||
|
if( selectedVariant == DEFAULT_VARIANT_NAME )
|
||
|
selectedVariant = "";
|
||
|
selLayout->setText(LAYOUT_COLUMN_VARIANT, selectedVariant);
|
||
|
}
|
||
|
|
||
|
// helper
|
||
|
LayoutUnit LayoutConfig::getLayoutUnitKey(QListViewItem *sel)
|
||
|
{
|
||
|
QString kbdLayout = sel->text(LAYOUT_COLUMN_MAP);
|
||
|
QString kbdVariant = sel->text(LAYOUT_COLUMN_VARIANT);
|
||
|
return LayoutUnit(kbdLayout, kbdVariant);
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::displayNameChanged(const QString& newDisplayName)
|
||
|
{
|
||
|
QListViewItem* selLayout = widget->listLayoutsDst->selectedItem();
|
||
|
if( selLayout == NULL )
|
||
|
return;
|
||
|
|
||
|
const LayoutUnit layoutUnitKey = getLayoutUnitKey( selLayout );
|
||
|
LayoutUnit& layoutUnit = *m_kxkbConfig.m_layouts.find(layoutUnitKey);
|
||
|
|
||
|
QString oldName = selLayout->text(LAYOUT_COLUMN_DISPLAY_NAME);
|
||
|
|
||
|
if( oldName.isEmpty() )
|
||
|
oldName = KxkbConfig::getDefaultDisplayName( layoutUnit );
|
||
|
|
||
|
if( oldName != newDisplayName ) {
|
||
|
kdDebug() << "setting label for " << layoutUnit.toPair() << " : " << newDisplayName << endl;
|
||
|
selLayout->setText(LAYOUT_COLUMN_DISPLAY_NAME, newDisplayName);
|
||
|
updateIndicator(selLayout);
|
||
|
emit changed();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/** will update flag with label if layout label has been edited
|
||
|
*/
|
||
|
void LayoutConfig::updateIndicator(QListViewItem* selLayout)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::latinChanged()
|
||
|
{
|
||
|
QListViewItem* selLayout = widget->listLayoutsDst->selectedItem();
|
||
|
if ( !selLayout ) {
|
||
|
widget->chkLatin->setChecked( false );
|
||
|
widget->chkLatin->setEnabled( false );
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
QString include;
|
||
|
if( widget->chkLatin->isChecked() )
|
||
|
include = "us";
|
||
|
else
|
||
|
include = "";
|
||
|
selLayout->setText(LAYOUT_COLUMN_INCLUDE, include);
|
||
|
|
||
|
LayoutUnit layoutUnitKey = getLayoutUnitKey(selLayout);
|
||
|
kdDebug() << "layout " << layoutUnitKey.toPair() << ", inc: " << include << endl;
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::layoutSelChanged(QListViewItem *sel)
|
||
|
{
|
||
|
widget->comboVariant->clear();
|
||
|
widget->comboVariant->setEnabled( sel != NULL );
|
||
|
widget->chkLatin->setChecked( false );
|
||
|
widget->chkLatin->setEnabled( sel != NULL );
|
||
|
|
||
|
if( sel == NULL ) {
|
||
|
updateLayoutCommand();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
LayoutUnit layoutUnitKey = getLayoutUnitKey(sel);
|
||
|
QString kbdLayout = layoutUnitKey.layout;
|
||
|
|
||
|
// TODO: need better algorithm here for determining if needs us group
|
||
|
if ( ! m_rules->isSingleGroup(kbdLayout)
|
||
|
|| kbdLayout.startsWith("us") || kbdLayout.startsWith("en") ) {
|
||
|
widget->chkLatin->setEnabled( false );
|
||
|
}
|
||
|
else {
|
||
|
QString inc = sel->text(LAYOUT_COLUMN_INCLUDE);
|
||
|
if ( inc.startsWith("us") || inc.startsWith("en") ) {
|
||
|
widget->chkLatin->setChecked(true);
|
||
|
}
|
||
|
else {
|
||
|
widget->chkLatin->setChecked(false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
QStringList vars = m_rules->getAvailableVariants(kbdLayout);
|
||
|
kdDebug() << "layout " << kbdLayout << " has " << vars.count() << " variants" << endl;
|
||
|
|
||
|
if( vars.count() > 0 ) {
|
||
|
vars.prepend(DEFAULT_VARIANT_NAME);
|
||
|
widget->comboVariant->insertStringList(vars);
|
||
|
|
||
|
QString variant = sel->text(LAYOUT_COLUMN_VARIANT);
|
||
|
if( variant != NULL && variant.isEmpty() == false ) {
|
||
|
widget->comboVariant->setCurrentText(variant);
|
||
|
}
|
||
|
else {
|
||
|
widget->comboVariant->setCurrentItem(0);
|
||
|
}
|
||
|
}
|
||
|
updateLayoutCommand();
|
||
|
}
|
||
|
|
||
|
QWidget* LayoutConfig::makeOptionsTab()
|
||
|
{
|
||
|
QListView *listView = widget->listOptions;
|
||
|
|
||
|
listView->setMinimumHeight(150);
|
||
|
listView->setSortColumn( -1 );
|
||
|
listView->setColumnText( 0, i18n( "Options" ) );
|
||
|
listView->clear();
|
||
|
|
||
|
connect(listView, SIGNAL(clicked(QListViewItem *)), SLOT(changed()));
|
||
|
connect(listView, SIGNAL(clicked(QListViewItem *)), SLOT(updateOptionsCommand()));
|
||
|
|
||
|
connect(widget->chkEnableOptions, SIGNAL(toggled(bool)), SLOT(changed()));
|
||
|
|
||
|
connect(widget->checkResetOld, SIGNAL(toggled(bool)), SLOT(changed()));
|
||
|
connect(widget->checkResetOld, SIGNAL(toggled(bool)), SLOT(updateOptionsCommand()));
|
||
|
|
||
|
//Create controllers for all options
|
||
|
QDictIterator<char> it(m_rules->options());
|
||
|
OptionListItem *parent;
|
||
|
for (; it.current(); ++it)
|
||
|
{
|
||
|
if (!it.currentKey().contains(':'))
|
||
|
{
|
||
|
if( it.currentKey() == "ctrl" || it.currentKey() == "caps"
|
||
|
|| it.currentKey() == "altwin" ) {
|
||
|
parent = new OptionListItem(listView, i18n( it.current() ),
|
||
|
QCheckListItem::RadioButtonController, it.currentKey());
|
||
|
OptionListItem *item = new OptionListItem(parent, i18n( "None" ),
|
||
|
QCheckListItem::RadioButton, "none");
|
||
|
item->setState(QCheckListItem::On);
|
||
|
}
|
||
|
else {
|
||
|
parent = new OptionListItem(listView, i18n( it.current() ),
|
||
|
QCheckListItem::CheckBoxController, it.currentKey());
|
||
|
}
|
||
|
parent->setOpen(true);
|
||
|
m_optionGroups.insert(i18n(it.currentKey().local8Bit()), parent);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
it.toFirst();
|
||
|
for( ; it.current(); ++it)
|
||
|
{
|
||
|
QString key = it.currentKey();
|
||
|
int pos = key.find(':');
|
||
|
if (pos >= 0)
|
||
|
{
|
||
|
OptionListItem *parent = m_optionGroups[key.left(pos)];
|
||
|
if (parent == NULL )
|
||
|
parent = m_optionGroups["misc"];
|
||
|
if (parent != NULL) {
|
||
|
// workaroung for mistake in rules file for xkb options in XFree 4.2.0
|
||
|
QString text(it.current());
|
||
|
text = text.replace( "Cap$", "Caps." );
|
||
|
if( parent->type() == QCheckListItem::RadioButtonController )
|
||
|
new OptionListItem(parent, i18n(text.utf8()),
|
||
|
QCheckListItem::RadioButton, key);
|
||
|
else
|
||
|
new OptionListItem(parent, i18n(text.utf8()),
|
||
|
QCheckListItem::CheckBox, key);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//scroll->setMinimumSize(450, 330);
|
||
|
|
||
|
return listView;
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::updateOptionsCommand()
|
||
|
{
|
||
|
QString setxkbmap;
|
||
|
QString options = createOptionString();
|
||
|
|
||
|
if( !options.isEmpty() ) {
|
||
|
setxkbmap = "setxkbmap -option "; //-rules " + m_rule
|
||
|
if( widget->checkResetOld->isChecked() )
|
||
|
setxkbmap += "-option ";
|
||
|
setxkbmap += options;
|
||
|
}
|
||
|
widget->editCmdLineOpt->setText(setxkbmap);
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::updateLayoutCommand()
|
||
|
{
|
||
|
QString setxkbmap;
|
||
|
QString layoutDisplayName;
|
||
|
QListViewItem* sel = widget->listLayoutsDst->selectedItem();
|
||
|
|
||
|
if( sel != NULL ) {
|
||
|
QString kbdLayout = sel->text(LAYOUT_COLUMN_MAP);
|
||
|
QString variant = widget->comboVariant->currentText();
|
||
|
if( variant == DEFAULT_VARIANT_NAME )
|
||
|
variant = "";
|
||
|
|
||
|
setxkbmap = "setxkbmap"; //-rules " + m_rule
|
||
|
setxkbmap += " -model " + lookupLocalized(m_rules->models(), widget->comboModel->currentText())
|
||
|
+ " -layout ";
|
||
|
setxkbmap += kbdLayout;
|
||
|
if( widget->chkLatin->isChecked() )
|
||
|
setxkbmap += ",us";
|
||
|
|
||
|
/* LayoutUnit layoutUnitKey = getLayoutUnitKey(sel);
|
||
|
layoutDisplayName = m_kxkbConfig.getLayoutDisplayName( *m_kxkbConfig.m_layouts.find(layoutUnitKey) );*/
|
||
|
layoutDisplayName = sel->text(LAYOUT_COLUMN_DISPLAY_NAME);
|
||
|
if( layoutDisplayName.isEmpty() ) {
|
||
|
int count = 0;
|
||
|
QListViewItem *item = widget->listLayoutsDst->firstChild();
|
||
|
while (item) {
|
||
|
QString layout_ = item->text(LAYOUT_COLUMN_MAP);
|
||
|
if( layout_ == kbdLayout )
|
||
|
++count;
|
||
|
item = item->nextSibling();
|
||
|
}
|
||
|
bool single = count < 2;
|
||
|
layoutDisplayName = m_kxkbConfig.getDefaultDisplayName(LayoutUnit(kbdLayout, variant), single);
|
||
|
}
|
||
|
kdDebug() << "disp: '" << layoutDisplayName << "'" << endl;
|
||
|
|
||
|
if( !variant.isEmpty() ) {
|
||
|
setxkbmap += " -variant ";
|
||
|
if( widget->chkLatin->isChecked() )
|
||
|
setxkbmap += ",";
|
||
|
setxkbmap += variant;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
widget->editCmdLine->setText(setxkbmap);
|
||
|
|
||
|
widget->editDisplayName->setEnabled( sel != NULL );
|
||
|
widget->editDisplayName->setText(layoutDisplayName);
|
||
|
}
|
||
|
|
||
|
void LayoutConfig::changed()
|
||
|
{
|
||
|
updateLayoutCommand();
|
||
|
emit KCModule::changed( true );
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::loadRules()
|
||
|
{
|
||
|
// do we need this ?
|
||
|
// this could obly be used if rules are changed and 'Defaults' is pressed
|
||
|
delete m_rules;
|
||
|
m_rules = new XkbRules();
|
||
|
|
||
|
QStringList modelsList;
|
||
|
QDictIterator<char> it(m_rules->models());
|
||
|
while (it.current()) {
|
||
|
modelsList.append(i18n(it.current()));
|
||
|
++it;
|
||
|
}
|
||
|
modelsList.sort();
|
||
|
|
||
|
widget->comboModel->clear();
|
||
|
widget->comboModel->insertStringList(modelsList);
|
||
|
widget->comboModel->setCurrentItem(0);
|
||
|
|
||
|
// fill in the additional layouts
|
||
|
widget->listLayoutsSrc->clear();
|
||
|
widget->listLayoutsDst->clear();
|
||
|
QDictIterator<char> it2(m_rules->layouts());
|
||
|
|
||
|
while (it2.current())
|
||
|
{
|
||
|
QString layout = it2.currentKey();
|
||
|
QString layoutName = it2.current();
|
||
|
QListViewItem *item = new QListViewItem(widget->listLayoutsSrc);
|
||
|
|
||
|
item->setPixmap(LAYOUT_COLUMN_FLAG, LayoutIcon::getInstance().findPixmap(layout, true));
|
||
|
item->setText(LAYOUT_COLUMN_NAME, i18n(layoutName.latin1()));
|
||
|
item->setText(LAYOUT_COLUMN_MAP, layout);
|
||
|
++it2;
|
||
|
}
|
||
|
widget->listLayoutsSrc->setSorting(LAYOUT_COLUMN_NAME); // from Qt3 QListView sorts by language
|
||
|
|
||
|
//TODO: reset options and xkb options
|
||
|
}
|
||
|
|
||
|
|
||
|
QString LayoutConfig::createOptionString()
|
||
|
{
|
||
|
QString options;
|
||
|
for (QDictIterator<char> it(m_rules->options()); it.current(); ++it)
|
||
|
{
|
||
|
QString option(it.currentKey());
|
||
|
|
||
|
if (option.contains(':')) {
|
||
|
|
||
|
QString optionKey = option.mid(0, option.find(':'));
|
||
|
OptionListItem *item = m_optionGroups[optionKey];
|
||
|
|
||
|
if( !item ) {
|
||
|
kdDebug() << "WARNING: skipping empty group for " << it.currentKey()
|
||
|
<< endl;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
OptionListItem *child = item->findChildItem( option );
|
||
|
|
||
|
if ( child ) {
|
||
|
if ( child->state() == QCheckListItem::On ) {
|
||
|
QString selectedName = child->optionName();
|
||
|
if ( !selectedName.isEmpty() && selectedName != "none" ) {
|
||
|
if (!options.isEmpty())
|
||
|
options.append(',');
|
||
|
options.append(selectedName);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
kdDebug() << "Empty option button for group " << it.currentKey() << endl;
|
||
|
}
|
||
|
}
|
||
|
return options;
|
||
|
}
|
||
|
|
||
|
|
||
|
void LayoutConfig::defaults()
|
||
|
{
|
||
|
loadRules();
|
||
|
m_kxkbConfig.setDefaults();
|
||
|
|
||
|
initUI();
|
||
|
|
||
|
emit KCModule::changed( true );
|
||
|
}
|
||
|
|
||
|
|
||
|
OptionListItem::OptionListItem( OptionListItem *parent, const QString &text,
|
||
|
Type tt, const QString &optionName )
|
||
|
: QCheckListItem( parent, text, tt ), m_OptionName( optionName )
|
||
|
{
|
||
|
}
|
||
|
|
||
|
OptionListItem::OptionListItem( QListView *parent, const QString &text,
|
||
|
Type tt, const QString &optionName )
|
||
|
: QCheckListItem( parent, text, tt ), m_OptionName( optionName )
|
||
|
{
|
||
|
}
|
||
|
|
||
|
OptionListItem * OptionListItem::findChildItem( const QString& optionName )
|
||
|
{
|
||
|
OptionListItem *child = static_cast<OptionListItem *>( firstChild() );
|
||
|
|
||
|
while ( child )
|
||
|
{
|
||
|
if ( child->optionName() == optionName )
|
||
|
break;
|
||
|
child = static_cast<OptionListItem *>( child->nextSibling() );
|
||
|
}
|
||
|
|
||
|
return child;
|
||
|
}
|
||
|
|
||
|
|
||
|
extern "C"
|
||
|
{
|
||
|
KDE_EXPORT KCModule *create_keyboard_layout(QWidget *parent, const char *)
|
||
|
{
|
||
|
return new LayoutConfig(parent, "kcmlayout");
|
||
|
}
|
||
|
|
||
|
KDE_EXPORT KCModule *create_keyboard(QWidget *parent, const char *)
|
||
|
{
|
||
|
return new KeyboardConfig(parent, "kcmlayout");
|
||
|
}
|
||
|
|
||
|
KDE_EXPORT void init_keyboard()
|
||
|
{
|
||
|
KeyboardConfig::init_keyboard();
|
||
|
|
||
|
KxkbConfig m_kxkbConfig;
|
||
|
m_kxkbConfig.load(KxkbConfig::LOAD_INIT_OPTIONS);
|
||
|
|
||
|
if( m_kxkbConfig.m_useKxkb == true ) {
|
||
|
kapp->startServiceByDesktopName("kxkb");
|
||
|
}
|
||
|
else {
|
||
|
// Even if the layouts have been disabled we still want to set Xkb options
|
||
|
// user can always switch them off now in the "Options" tab
|
||
|
if( m_kxkbConfig.m_enableXkbOptions ) {
|
||
|
if( !XKBExtension::setXkbOptions(m_kxkbConfig.m_options, m_kxkbConfig.m_resetOldOptions) ) {
|
||
|
kdDebug() << "Setting XKB options failed!" << endl;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
#if 0// do not remove!
|
||
|
// please don't change/fix messages below
|
||
|
// they're taken from XFree86 as is and should stay the same
|
||
|
I18N_NOOP("Brazilian ABNT2");
|
||
|
I18N_NOOP("Dell 101-key PC");
|
||
|
I18N_NOOP("Everex STEPnote");
|
||
|
I18N_NOOP("Generic 101-key PC");
|
||
|
I18N_NOOP("Generic 102-key (Intl) PC");
|
||
|
I18N_NOOP("Generic 104-key PC");
|
||
|
I18N_NOOP("Generic 105-key (Intl) PC");
|
||
|
I18N_NOOP("Japanese 106-key");
|
||
|
I18N_NOOP("Microsoft Natural");
|
||
|
I18N_NOOP("Northgate OmniKey 101");
|
||
|
I18N_NOOP("Keytronic FlexPro");
|
||
|
I18N_NOOP("Winbook Model XP5");
|
||
|
|
||
|
// These options are from XFree 4.1.0
|
||
|
I18N_NOOP("Group Shift/Lock behavior");
|
||
|
I18N_NOOP("R-Alt switches group while pressed");
|
||
|
I18N_NOOP("Right Alt key changes group");
|
||
|
I18N_NOOP("Caps Lock key changes group");
|
||
|
I18N_NOOP("Menu key changes group");
|
||
|
I18N_NOOP("Both Shift keys together change group");
|
||
|
I18N_NOOP("Control+Shift changes group");
|
||
|
I18N_NOOP("Alt+Control changes group");
|
||
|
I18N_NOOP("Alt+Shift changes group");
|
||
|
I18N_NOOP("Control Key Position");
|
||
|
I18N_NOOP("Make CapsLock an additional Control");
|
||
|
I18N_NOOP("Swap Control and Caps Lock");
|
||
|
I18N_NOOP("Control key at left of 'A'");
|
||
|
I18N_NOOP("Control key at bottom left");
|
||
|
I18N_NOOP("Use keyboard LED to show alternative group");
|
||
|
I18N_NOOP("Num_Lock LED shows alternative group");
|
||
|
I18N_NOOP("Caps_Lock LED shows alternative group");
|
||
|
I18N_NOOP("Scroll_Lock LED shows alternative group");
|
||
|
|
||
|
//these seem to be new in XFree86 4.2.0
|
||
|
I18N_NOOP("Left Win-key switches group while pressed");
|
||
|
I18N_NOOP("Right Win-key switches group while pressed");
|
||
|
I18N_NOOP("Both Win-keys switch group while pressed");
|
||
|
I18N_NOOP("Left Win-key changes group");
|
||
|
I18N_NOOP("Right Win-key changes group");
|
||
|
I18N_NOOP("Third level choosers");
|
||
|
I18N_NOOP("Press Right Control to choose 3rd level");
|
||
|
I18N_NOOP("Press Menu key to choose 3rd level");
|
||
|
I18N_NOOP("Press any of Win-keys to choose 3rd level");
|
||
|
I18N_NOOP("Press Left Win-key to choose 3rd level");
|
||
|
I18N_NOOP("Press Right Win-key to choose 3rd level");
|
||
|
I18N_NOOP("CapsLock key behavior");
|
||
|
I18N_NOOP("uses internal capitalization. Shift cancels Caps.");
|
||
|
I18N_NOOP("uses internal capitalization. Shift doesn't cancel Caps.");
|
||
|
I18N_NOOP("acts as Shift with locking. Shift cancels Caps.");
|
||
|
I18N_NOOP("acts as Shift with locking. Shift doesn't cancel Caps.");
|
||
|
I18N_NOOP("Alt/Win key behavior");
|
||
|
I18N_NOOP("Add the standard behavior to Menu key.");
|
||
|
I18N_NOOP("Alt and Meta on the Alt keys (default).");
|
||
|
I18N_NOOP("Meta is mapped to the Win-keys.");
|
||
|
I18N_NOOP("Meta is mapped to the left Win-key.");
|
||
|
I18N_NOOP("Super is mapped to the Win-keys (default).");
|
||
|
I18N_NOOP("Hyper is mapped to the Win-keys.");
|
||
|
I18N_NOOP("Right Alt is Compose");
|
||
|
I18N_NOOP("Right Win-key is Compose");
|
||
|
I18N_NOOP("Menu is Compose");
|
||
|
|
||
|
//these seem to be new in XFree86 4.3.0
|
||
|
I18N_NOOP( "Both Ctrl keys together change group" );
|
||
|
I18N_NOOP( "Both Alt keys together change group" );
|
||
|
I18N_NOOP( "Left Shift key changes group" );
|
||
|
I18N_NOOP( "Right Shift key changes group" );
|
||
|
I18N_NOOP( "Right Ctrl key changes group" );
|
||
|
I18N_NOOP( "Left Alt key changes group" );
|
||
|
I18N_NOOP( "Left Ctrl key changes group" );
|
||
|
I18N_NOOP( "Compose Key" );
|
||
|
|
||
|
//these seem to be new in XFree86 4.4.0
|
||
|
I18N_NOOP("Shift with numpad keys works as in MS Windows.");
|
||
|
I18N_NOOP("Special keys (Ctrl+Alt+<key>) handled in a server.");
|
||
|
I18N_NOOP("Miscellaneous compatibility options");
|
||
|
I18N_NOOP("Right Control key works as Right Alt");
|
||
|
|
||
|
//these seem to be in x.org and Debian XFree86 4.3
|
||
|
I18N_NOOP("Right Alt key switches group while pressed");
|
||
|
I18N_NOOP("Left Alt key switches group while pressed");
|
||
|
I18N_NOOP("Press Right Alt-key to choose 3rd level");
|
||
|
|
||
|
//new in Xorg 6.9
|
||
|
I18N_NOOP("R-Alt switches group while pressed.");
|
||
|
I18N_NOOP("Left Alt key switches group while pressed.");
|
||
|
I18N_NOOP("Left Win-key switches group while pressed.");
|
||
|
I18N_NOOP("Right Win-key switches group while pressed.");
|
||
|
I18N_NOOP("Both Win-keys switch group while pressed.");
|
||
|
I18N_NOOP("Right Ctrl key switches group while pressed.");
|
||
|
I18N_NOOP("Right Alt key changes group.");
|
||
|
I18N_NOOP("Left Alt key changes group.");
|
||
|
I18N_NOOP("CapsLock key changes group.");
|
||
|
I18N_NOOP("Shift+CapsLock changes group.");
|
||
|
I18N_NOOP("Both Shift keys together change group.");
|
||
|
I18N_NOOP("Both Alt keys together change group.");
|
||
|
I18N_NOOP("Both Ctrl keys together change group.");
|
||
|
I18N_NOOP("Ctrl+Shift changes group.");
|
||
|
I18N_NOOP("Alt+Ctrl changes group.");
|
||
|
I18N_NOOP("Alt+Shift changes group.");
|
||
|
I18N_NOOP("Menu key changes group.");
|
||
|
I18N_NOOP("Left Win-key changes group.");
|
||
|
I18N_NOOP("Right Win-key changes group.");
|
||
|
I18N_NOOP("Left Shift key changes group.");
|
||
|
I18N_NOOP("Right Shift key changes group.");
|
||
|
I18N_NOOP("Left Ctrl key changes group.");
|
||
|
I18N_NOOP("Right Ctrl key changes group.");
|
||
|
I18N_NOOP("Press Right Ctrl to choose 3rd level.");
|
||
|
I18N_NOOP("Press Menu key to choose 3rd level.");
|
||
|
I18N_NOOP("Press any of Win-keys to choose 3rd level.");
|
||
|
I18N_NOOP("Press Left Win-key to choose 3rd level.");
|
||
|
I18N_NOOP("Press Right Win-key to choose 3rd level.");
|
||
|
I18N_NOOP("Press any of Alt keys to choose 3rd level.");
|
||
|
I18N_NOOP("Press Left Alt key to choose 3rd level.");
|
||
|
I18N_NOOP("Press Right Alt key to choose 3rd level.");
|
||
|
I18N_NOOP("Ctrl key position");
|
||
|
I18N_NOOP("Make CapsLock an additional Ctrl.");
|
||
|
I18N_NOOP("Swap Ctrl and CapsLock.");
|
||
|
I18N_NOOP("Ctrl key at left of 'A'");
|
||
|
I18N_NOOP("Ctrl key at bottom left");
|
||
|
I18N_NOOP("Right Ctrl key works as Right Alt.");
|
||
|
I18N_NOOP("Use keyboard LED to show alternative group.");
|
||
|
I18N_NOOP("NumLock LED shows alternative group.");
|
||
|
I18N_NOOP("CapsLock LED shows alternative group.");
|
||
|
I18N_NOOP("ScrollLock LED shows alternative group.");
|
||
|
I18N_NOOP("CapsLock uses internal capitalization. Shift cancels CapsLock.");
|
||
|
I18N_NOOP("CapsLock uses internal capitalization. Shift doesn't cancel CapsLock.");
|
||
|
I18N_NOOP("CapsLock acts as Shift with locking. Shift cancels CapsLock.");
|
||
|
I18N_NOOP("CapsLock acts as Shift with locking. Shift doesn't cancel CapsLock.");
|
||
|
I18N_NOOP("CapsLock just locks the Shift modifier.");
|
||
|
I18N_NOOP("CapsLock toggles normal capitalization of alphabetic characters.");
|
||
|
I18N_NOOP("CapsLock toggles Shift so all keys are affected.");
|
||
|
I18N_NOOP("Alt and Meta are on the Alt keys (default).");
|
||
|
I18N_NOOP("Alt is mapped to the right Win-key and Super to Menu.");
|
||
|
I18N_NOOP("Compose key position");
|
||
|
I18N_NOOP("Right Alt is Compose.");
|
||
|
I18N_NOOP("Right Win-key is Compose.");
|
||
|
I18N_NOOP("Menu is Compose.");
|
||
|
I18N_NOOP("Right Ctrl is Compose.");
|
||
|
I18N_NOOP("Caps Lock is Compose.");
|
||
|
I18N_NOOP("Special keys (Ctrl+Alt+<key>) handled in a server.");
|
||
|
I18N_NOOP("Adding the EuroSign to certain keys");
|
||
|
I18N_NOOP("Add the EuroSign to the E key.");
|
||
|
I18N_NOOP("Add the EuroSign to the 5 key.");
|
||
|
I18N_NOOP("Add the EuroSign to the 2 key.");
|
||
|
#endif
|