diff options
author | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
---|---|---|
committer | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
commit | 4aed2c8219774f5d797760606b8489a92ddc5163 (patch) | |
tree | 3f8c130f7d269626bf6a9447407ef6c35954426a /kxkb/kcmlayout.cpp | |
download | tdebase-4aed2c8219774f5d797760606b8489a92ddc5163.tar.gz tdebase-4aed2c8219774f5d797760606b8489a92ddc5163.zip |
Copy the KDE 3.5 branch to branches/trinity for new KDE 3.5 features.
BUG:215923
git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdebase@1054174 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'kxkb/kcmlayout.cpp')
-rw-r--r-- | kxkb/kcmlayout.cpp | 976 |
1 files changed, 976 insertions, 0 deletions
diff --git a/kxkb/kcmlayout.cpp b/kxkb/kcmlayout.cpp new file mode 100644 index 000000000..5f5b1ff84 --- /dev/null +++ b/kxkb/kcmlayout.cpp @@ -0,0 +1,976 @@ +#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 |