Kxkb: improve TDE Control Centre module.

1) Add conflicts check for hotkey checkboxes

Due to the ability to set multiple keyboard layout switching hotkeys,
it is important to inform the user of conflicting options that are
not handled properly by the X.org server when set at the same time
(e.g. Win+Space and Alt+Space).

This change adds a warning that informs the user about the problem and
the conflicting options. This warning is shown only when setting
multiple hotkeys via the Xkb options tab, which is for the advanced
user. Most users will ever need only one hotkey, and the combobox on
the first tab should be more than enough.

2) Add "none" item to layout switching options

3) Replace Reset old options checkbox with radio buttons

As per discussion, this makes the function of the option more
apparent. WhatIs hints have been added for additional clarity.

4) Update hotkey combobox per server options

5) Avoid duplication of options by querying Xkb for already set options.

This was a problem in Append Mode in which `setxkbmap` strings would
get too long due to setting already set options. This code checks for
already set options and omits them from the new `setxkbmap` call.

This does not apply to Overwrite Mode.

6) Overwrite previous grp: options when using the combobox

See previous commit message about the addition of hotkeys combobox.

Signed-off-by: Mavridis Philippe <mavridisf@gmail.com>
pull/304/head
Mavridis Philippe 1 year ago
parent a67db2d484
commit e8208c1dfb
No known key found for this signature in database
GPG Key ID: F8D2D7E2F989A494

@ -15,6 +15,7 @@
#include <X11/Xlib.h>
#include <X11/XKBlib.h>
#include <X11/extensions/XKBfile.h>
#include <X11/extensions/XKBrules.h>
#include <X11/extensions/XKBgeom.h>
#include <X11/extensions/XKM.h>
@ -89,11 +90,17 @@ bool XKBExtension::setXkbOptions(const XkbOptions options)
TDEProcess p;
p << exe;
p << "-layout";
p << options.layouts;
p << "-variant";
p << options.variants;
if (!options.layouts.isEmpty())
{
p << "-layout";
p << options.layouts;
}
if (!options.variants.isEmpty())
{
p << "-variant";
p << options.variants;
}
if (!options.model.isEmpty()) {
p << "-model";
@ -103,17 +110,50 @@ bool XKBExtension::setXkbOptions(const XkbOptions options)
if (options.resetOld) {
p << "-option";
}
if (!options.options.isEmpty()) {
p << "-option" << options.options;
p << "-option";
if (options.resetOld)
{
p << options.options;
}
else
{
// Avoid duplication of options in Append mode
TQStringList srvOptions = TQStringList::split(",", XKBExtension::getServerOptions());
TQStringList kxkbOptions = TQStringList::split(",", options.options);
TQStringList newOptions;
for (TQStringList::Iterator it = kxkbOptions.begin(); it != kxkbOptions.end(); ++it)
{
TQString option(*it);
if (!srvOptions.contains(option))
{
newOptions << option;
}
}
p << newOptions.join(",");
}
}
kdDebug() << "[kxkb-extension] Command: " << p.args() << endl;
kdDebug() << "[setXkbOptions] Command: " << p.args() << endl;
p.start(TDEProcess::Block);
return p.normalExit() && (p.exitStatus() == 0);
}
TQString XKBExtension::getServerOptions()
{
XkbRF_VarDefsRec vd;
if (XkbRF_GetNamesProp(tqt_xdisplay(), nullptr, &vd) && vd.options)
{
kdDebug() << "[kxkb-extension] Got server options " << vd.options << endl;
return TQString(vd.options);
}
return TQString::null;
}
bool XKBExtension::setGroup(unsigned int group)
{
kdDebug() << "[kxkb-extension] Setting group " << group << endl;
@ -136,4 +176,4 @@ void XKBExtension::processXEvent(XEvent *event) {
}
}
#include "extension.moc"
#include "extension.moc"

@ -16,6 +16,7 @@ public:
bool init();
static bool setXkbOptions(const XkbOptions options);
static TQString getServerOptions();
bool setGroup(unsigned int group);
unsigned int getGroup() const;
void processXEvent(XEvent *ev);

@ -24,6 +24,7 @@
#include <kdebug.h>
#include <tdeapplication.h>
#include <kiconloader.h>
#include <tdemessagebox.h>
#include <dcopref.h>
#include <dcopclient.h>
@ -99,7 +100,8 @@ static TQListViewItem* copyLVI(const TQListViewItem* src, TQListView* parent)
LayoutConfig::LayoutConfig(TQWidget *parent, const char *name)
: TDECModule(parent, name),
m_rules(NULL)
m_rules(NULL),
m_forceGrpOverwrite(false)
{
TQVBoxLayout *main = new TQVBoxLayout(this, 0, KDialog::spacingHint());
@ -232,6 +234,7 @@ void LayoutConfig::initUI() {
widget->comboHotkey->insertItem(i18n(hkDesc));
}
}
widget->comboHotkey->insertItem(i18n("None"));
widget->comboHotkey->insertItem(i18n("Other..."));
// display KXKB switching options
@ -243,7 +246,7 @@ void LayoutConfig::initUI() {
widget->radFlagOnly->setChecked( showFlag && !showLabel );
widget->radLabelOnly->setChecked( !showFlag && showLabel );
widget->checkResetOld->setChecked(m_kxkbConfig.m_resetOldOptions);
widget->xkbOptsMode->setButton(m_kxkbConfig.m_resetOldOptions ? 0 : 1);
widget->grpLabel->setButton( ( m_kxkbConfig.m_useThemeColors ? 0 : 1 ) );
widget->bgColor->setColor( m_kxkbConfig.m_colorBackground );
@ -280,15 +283,22 @@ void LayoutConfig::initUI() {
widget->chkEnable->setChecked( m_kxkbConfig.m_useKxkb );
widget->grpLayouts->setEnabled( m_kxkbConfig.m_useKxkb );
widget->optionsFrame->setEnabled( m_kxkbConfig.m_useKxkb );
widget->swOptsFrame->setEnabled( m_kxkbConfig.m_useKxkb );
widget->indOptsFrame->setEnabled( m_kxkbConfig.m_useKxkb );
// display xkb options
TQStringList activeOptions = TQStringList::split(',', m_kxkbConfig.m_options);
bool foundGrp = false;
for (TQStringList::ConstIterator it = activeOptions.begin(); it != activeOptions.end(); ++it)
{
TQString option = *it;
TQString optionKey = option.mid(0, option.find(':'));
TQString optionName = m_rules->options()[option];
if (optionKey == "grp") {
foundGrp = true;
}
OptionListItem *item = m_optionGroups[i18n(optionKey.latin1())];
if (item != NULL) {
@ -304,8 +314,15 @@ void LayoutConfig::initUI() {
}
}
if (!foundGrp) {
OptionListItem *grpNone = itemForOption("grp:none");
if (grpNone) {
grpNone->setOn(true);
}
}
updateOptionsCommand();
updateHotkeyCombo();
updateHotkeyCombo(true);
emit TDECModule::changed( false );
}
@ -315,7 +332,7 @@ void LayoutConfig::save()
TQString model = lookupLocalized(m_rules->models(), widget->comboModel->currentText());
m_kxkbConfig.m_model = model;
m_kxkbConfig.m_resetOldOptions = widget->checkResetOld->isChecked();
m_kxkbConfig.m_resetOldOptions = widget->radXkbOverwrite->isOn();
m_kxkbConfig.m_options = createOptionString();
m_kxkbConfig.m_useThemeColors = widget->radLabelUseTheme->isChecked();
@ -373,6 +390,35 @@ void LayoutConfig::save()
m_kxkbConfig.save();
// We might need to unset previous hotkey options
if (m_forceGrpOverwrite)
{
// First get all the server's options
TQStringList srvOptions = TQStringList::split(",", XKBExtension::getServerOptions());
TQStringList newOptions;
// Then remove all grp: options
for (TQStringList::Iterator it = srvOptions.begin(); it != srvOptions.end(); ++it)
{
TQString opt(*it);
if (!opt.startsWith("grp:"))
{
newOptions << opt;
}
}
XkbOptions xkbOptions;
xkbOptions.options = newOptions.join(",");
xkbOptions.resetOld = true;
if (!XKBExtension::setXkbOptions(xkbOptions))
{
kdWarning() << "[LayoutConfig::save] Could not overwrite previous grp: options!" << endl;
}
m_forceGrpOverwrite = false;
}
// Get current layout from Kxkb
if (!kapp->dcopClient()->isAttached())
kapp->dcopClient()->attach();
@ -405,6 +451,8 @@ void LayoutConfig::save()
}
}
updateHotkeyCombo();
emit TDECModule::changed( false );
}
@ -518,6 +566,7 @@ void LayoutConfig::variantChanged()
if( selectedVariant == DEFAULT_VARIANT_NAME )
selectedVariant = "";
selLayout->setText(LAYOUT_COLUMN_VARIANT, selectedVariant);
updateLayoutCommand();
}
// helper
@ -598,12 +647,12 @@ TQWidget* LayoutConfig::makeOptionsTab()
listView->clear();
connect(listView, TQT_SIGNAL(clicked(TQListViewItem *)), TQT_SLOT(changed()));
connect(listView, TQT_SIGNAL(clicked(TQListViewItem *)), TQT_SLOT(updateOptionsCommand()));
connect(listView, TQT_SIGNAL(clicked(TQListViewItem *)), TQT_SLOT(resolveConflicts(TQListViewItem *)));
connect(listView, TQT_SIGNAL(clicked(TQListViewItem *)), TQT_SLOT(updateHotkeyCombo()));
connect(widget->checkResetOld, TQT_SIGNAL(toggled(bool)), TQT_SLOT(changed()));
connect(widget->checkResetOld, TQT_SIGNAL(toggled(bool)), TQT_SLOT(updateOptionsCommand()));
connect(widget->checkResetOld, TQT_SIGNAL(toggled(bool)), TQT_SLOT(updateHotkeyCombo()));
connect(widget->xkbOptsMode, TQT_SIGNAL(released(int)), TQT_SLOT(changed()));
connect(widget->xkbOptsMode, TQT_SIGNAL(released(int)), TQT_SLOT(updateOptionsCommand()));
connect(widget->xkbOptsMode, TQT_SIGNAL(released(int)), TQT_SLOT(updateHotkeyCombo()));
//Create controllers for all options
TQDictIterator<char> it(m_rules->options());
@ -613,13 +662,20 @@ TQWidget* LayoutConfig::makeOptionsTab()
if (!it.currentKey().contains(':'))
{
if( it.currentKey() == "ctrl" || it.currentKey() == "caps"
|| it.currentKey() == "altwin" || it.currentKey() == "grp") {
|| it.currentKey() == "altwin") {
parent = new OptionListItem(listView, i18n( it.current() ),
TQCheckListItem::RadioButtonController, it.currentKey());
OptionListItem *item = new OptionListItem(parent, i18n( "None" ),
TQCheckListItem::RadioButton, "none");
item->setState(TQCheckListItem::On);
}
else if (it.currentKey() == "grp") {
parent = new OptionListItem(listView, i18n(it.current()),
TQCheckListItem::RadioButtonController, it.currentKey());
parent->setSelectable(false);
OptionListItem *item = new OptionListItem(parent, i18n("None"),
TQCheckListItem::CheckBox, "grp:none");
}
else {
parent = new OptionListItem(listView, i18n( it.current() ),
TQCheckListItem::CheckBoxController, it.currentKey());
@ -643,12 +699,13 @@ TQWidget* LayoutConfig::makeOptionsTab()
// workaroung for mistake in rules file for xkb options in XFree 4.2.0
TQString text(it.current());
text = text.replace( "Cap$", "Caps." );
if( parent->type() == TQCheckListItem::RadioButtonController )
if ( parent->type() == TQCheckListItem::CheckBoxController
|| key.startsWith("grp:"))
new OptionListItem(parent, i18n(text.utf8()),
TQCheckListItem::RadioButton, key);
TQCheckListItem::CheckBox, key);
else
new OptionListItem(parent, i18n(text.utf8()),
TQCheckListItem::CheckBox, key);
TQCheckListItem::RadioButton, key);
}
}
}
@ -662,14 +719,19 @@ void LayoutConfig::updateOptionsCommand()
{
TQString setxkbmap;
TQString options = createOptionString();
bool overwrite = widget->radXkbOverwrite->isOn();
if( !options.isEmpty() ) {
setxkbmap = "setxkbmap -option "; //-rules " + m_rule
if( widget->checkResetOld->isChecked() )
if (overwrite)
setxkbmap += "-option ";
setxkbmap += options;
}
else if (overwrite) {
setxkbmap = "setxkbmap -option";
}
widget->editCmdLineOpt->setText(setxkbmap);
widget->editCmdLineOpt->setDisabled(setxkbmap.isEmpty());
}
void LayoutConfig::updateLayoutCommand()
@ -726,38 +788,268 @@ void LayoutConfig::updateLayoutCommand()
widget->editDisplayName->setText(selDisplayName);
}
void LayoutConfig::checkConflicts(OptionListItem *current,
TQStringList conflicting,
TQStringList &conflicts)
{
if (!current || conflicting.count() < 2 ||
!conflicting.contains(current->optionName()))
{
return;
}
TQStringList::Iterator it;
for (it = conflicting.begin(); it != conflicting.end(); ++it) {
TQString option(*it);
if (option == current->optionName()) {
continue;
}
OptionListItem *item = itemForOption(option);
if (item && item->isOn()) {
conflicts << item->text();
}
}
}
void LayoutConfig::resolveConflicts(TQListViewItem *lvi) {
OptionListItem *current = (OptionListItem*)lvi;
kdDebug() << "resolveConflicts : " << current->optionName() << endl;
if (current->optionName().startsWith("grp:")) {
OptionListItem *grpItem = m_optionGroups[i18n("grp")];
if (grpItem == NULL) {
kdWarning() << "LayoutConfig: cannot find grp item group" << endl;
return;
}
OptionListItem *noneItem = grpItem->findChildItem("grp:none");
if (!noneItem) {
kdDebug() << "LayoutConfig: unable to find None item for grp!" << endl;
}
else {
// Option "none" selected, uncheck all other options immediately
if (current->optionName() == "grp:none") {
if (current->isOn()) {
OptionListItem *child = (OptionListItem*)grpItem->firstChild();
while (child) {
if (child != current) {
child->setOn(false);
}
child = (OptionListItem*)child->nextSibling();
}
}
else {
current->setOn(true);
}
updateOptionsCommand();
return;
}
// If no options are selected then select "none"
bool notNone = false;
OptionListItem *child = (OptionListItem*)grpItem->firstChild();
while (child) {
if (child->isOn() && child->optionName() != "none") {
notNone = true;
break;
}
child = (OptionListItem*)child->nextSibling();
}
noneItem->setOn(!notNone);
}
}
TQStringList conflicts;
OptionListItem *conflict;
TQStringList conflicting;
/* Might be incomplete */
// Space
conflicting << "grp:win_space_toggle"
<< "grp:alt_space_toggle"
<< "grp:ctrl_space_toggle";
checkConflicts(current, conflicting, conflicts);
// Shift
conflicting.clear();
conflicting << "grp:ctrl_shift_toggle"
<< "grp:alt_shift_toggle";
checkConflicts(current, conflicting, conflicts);
// Control
conflicting.clear();
conflicting << "grp:ctrl_select"
<< "grp:ctrl_alt_toggle"
<< "grp:ctrl_shift_toggle";
checkConflicts(current, conflicting, conflicts);
// Left Ctrl
conflicting.clear();
conflicting << "grp:lctrl_toggle"
<< "grp:lctrl_lshift_toggle";
checkConflicts(current, conflicting, conflicts);
// Right Ctrl
conflicting.clear();
conflicting << "grp:rctrl_toggle"
<< "grp:rctrl_rshift_toggle";
checkConflicts(current, conflicting, conflicts);
// Win
conflicting.clear();
conflicting << "grp:win_space_toggle"
<< "grp:win_switch"
<< "win_menu_select";
checkConflicts(current, conflicting, conflicts);
// Left Alt
conflicting.clear();
conflicting << "grp:lalt_toggle"
<< "grp:lalt_lshift_toggle";
checkConflicts(current, conflicting, conflicts);
// Right Alt
conflicting.clear();
conflicting << "grp:ralt_toggle"
<< "grp:ralt_rshift_toggle";
checkConflicts(current, conflicting, conflicts);
// Caps Lock
conflicting.clear();
conflicting << "grp:caps_toggle"
<< "grp:caps_select"
<< "grp:caps_switch"
<< "grp:alt_caps_toggle";
checkConflicts(current, conflicting, conflicts);
if (conflicts.count()) {
TQString curText = current->text();
int confirm = KMessageBox::warningYesNoList(this,
i18n("<qt>The option <b>%1</b> might conflict with "
"other options that you have already enabled.<br>"
"Are you sure that you really want to enable "
"<b>%2</b>?</qt>")
.arg(curText).arg(curText),
conflicts,
i18n("Conflicting options"));
if (confirm == KMessageBox::No) {
current->setOn(false);
}
}
updateOptionsCommand();
}
// Synchronizes Xkb grp options --> hotkeys combobox
void LayoutConfig::updateHotkeyCombo() {
updateHotkeyCombo(false);
}
void LayoutConfig::updateHotkeyCombo(bool initial) {
OptionListItem *grpItem = m_optionGroups[i18n("grp")];
if (grpItem == NULL) {
kdWarning() << "LayoutConfig: cannot find grp item group" << endl;
return;
}
OptionListItem *child = (OptionListItem*)grpItem->firstChild();
while (child) {
if (child->isOn()) {
bool found = false;
for (int i = 0; i < widget->comboHotkey->count(); ++i) {
if (child->text() == widget->comboHotkey->text(i)) {
widget->comboHotkey->setCurrentItem(i);
found = true;
}
TQStringList hotkeys;
// Get server options first
if (initial || widget->xkbOptsMode->selectedId() == 1)
{
TQStringList opts = TQStringList::split(",", XKBExtension::getServerOptions());
for (TQStringList::Iterator it = opts.begin(); it != opts.end(); ++it)
{
TQString option(*it);
if (!option.startsWith("grp:"))
{
continue;
}
if (!found) {
int other = widget->comboHotkey->count() - 1;
widget->comboHotkey->changeItem(i18n("Other (%1)...").arg(child->text()),
other);
widget->comboHotkey->setCurrentItem(other);
// Get description from existing item
// This spares us the trouble of querying Xkb rules second time
OptionListItem *item = itemForOption(option);
if (!item)
{
kdWarning() << "[updateHotkeyCombo] server has set unknown option: "
<< option << endl;
continue;
}
TQString optionName = item->text();
if (!hotkeys.contains(optionName) && option != "grp:none")
{
hotkeys << optionName;
}
}
}
OptionListItem *child = (OptionListItem*)grpItem->firstChild();
while (child) {
TQString optionText = child->text();
if (child->isOn() && !hotkeys.contains(optionText) && child->optionName() != "grp:none") {
hotkeys << optionText;
}
child = (OptionListItem*)child->nextSibling();
}
if (!hotkeys.count()) {
OptionListItem *noneItem = itemForOption("grp:none");
if (noneItem)
{
hotkeys << noneItem->text();
}
else
{
kdWarning() << "[updateHotkeyCombo] cannot find grp:none item!" << endl;
hotkeys << widget->comboHotkey->text(0); // fallback
}
}
int other = widget->comboHotkey->count() - 1;
widget->comboHotkey->changeItem(i18n("Custom..."), other);
if (hotkeys.count() < 2) {
bool found = false;
for (int i = 0; i < widget->comboHotkey->count(); ++i) {
if (hotkeys[0] == widget->comboHotkey->text(i)) {
widget->comboHotkey->setCurrentItem(i);
found = true;
}
}
if (!found) {
widget->comboHotkey->changeItem(i18n("Other (%1)").arg(hotkeys[0]),
other);
widget->comboHotkey->setCurrentItem(other);
}
}
else {
widget->comboHotkey->changeItem(i18n("Multiple (%1)").arg(hotkeys.join("; ")),
other);
widget->comboHotkey->setCurrentItem(other);
}
}
// Synchronizes hotkeys combobox --> Xkb grp options
void LayoutConfig::hotkeyComboChanged() {
TQString hkDesc = widget->comboHotkey->currentText();
TQStringList hotkeys;
int other = widget->comboHotkey->count() - 1;
if (widget->comboHotkey->currentItem() != other) {
hotkeys << widget->comboHotkey->currentText();
}
else {
TQString otherStr = widget->comboHotkey->text(other);
int i1 = otherStr.find("(");
if (i1 != -1) { // custom hotkey(s) set
++i1;
int i2 = otherStr.findRev(")");
if (i2 != -1) {
hotkeys = TQStringList::split("; ", otherStr.mid(i1, i2-i1));
}
}
}
OptionListItem *grpItem = m_optionGroups[i18n("grp")];
if (grpItem == NULL) {
@ -767,21 +1059,17 @@ void LayoutConfig::hotkeyComboChanged() {
OptionListItem *child = (OptionListItem*)grpItem->firstChild();
while (child) {
child->setOn(child->text() == hkDesc);
child->setOn(hotkeys.contains(child->text()));
child = (OptionListItem*)child->nextSibling();
}
// Other...
if (widget->comboHotkey->count() - 1 == widget->comboHotkey->currentItem()) {
OptionListItem *none = grpItem->findChildItem("none");
if (none) {
none->setOn(true);
}
widget->tabWidget->setCurrentPage(2);
widget->listOptions->setCurrentItem(none);
widget->listOptions->ensureItemVisible(none);
if (widget->comboHotkey->currentItem() == other) {
widget->tabWidget->setCurrentPage(3);
widget->listOptions->ensureItemVisible(grpItem);
widget->listOptions->setFocus();
}
m_forceGrpOverwrite = true;
}
void LayoutConfig::changed()
@ -831,6 +1119,20 @@ void LayoutConfig::loadRules()
//TODO: reset options and xkb options
}
OptionListItem* LayoutConfig::itemForOption(TQString option) {
if (!option.contains(':')) {
return nullptr;
}
TQString optionKey = option.mid(0, option.find(':'));
OptionListItem *item = m_optionGroups[optionKey];
if( !item ) {
kdDebug() << "WARNING: skipping empty group for " << option << endl;
return nullptr;
}
return (OptionListItem*)item->findChildItem(option);
}
TQString LayoutConfig::createOptionString()
{
@ -838,32 +1140,17 @@ TQString LayoutConfig::createOptionString()
for (TQDictIterator<char> it(m_rules->options()); it.current(); ++it)
{
TQString option(it.currentKey());
if (option.contains(':')) {
TQString 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() == TQCheckListItem::On ) {
TQString selectedName = child->optionName();
if ( !selectedName.isEmpty() && selectedName != "none" ) {
if (!options.isEmpty())
options.append(',');
options.append(selectedName);
}
}
OptionListItem *child = itemForOption(option);
if (!child) {
continue;
}
if ( child->state() == TQCheckListItem::On ) {
TQString 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;
@ -931,7 +1218,7 @@ extern "C"
kapp->startServiceByDesktopName("kxkb");
}
else {
if (!XKBExtension::setXkbOptions(m_kxkbConfig.getXkbOptions())) {
if (!XKBExtension::setXkbOptions(m_kxkbConfig.getKXkbOptions())) {
kdDebug() << "Setting XKB options failed!" << endl;
}
}

@ -31,6 +31,7 @@ public:
protected:
TQString createOptionString();
void updateIndicator(TQListViewItem* selLayout);
OptionListItem* itemForOption(TQString option);
protected slots:
void moveUp();
@ -43,8 +44,10 @@ protected slots:
void updateLayoutCommand();
void updateOptionsCommand();
void updateHotkeyCombo();
void updateHotkeyCombo(bool initial);
void add();
void remove();
void resolveConflicts(TQListViewItem *lvi);
void changed();
@ -54,10 +57,13 @@ private:
XkbRules *m_rules;
KxkbConfig m_kxkbConfig;
TQDict<OptionListItem> m_optionGroups;
bool m_forceGrpOverwrite;
TQWidget* makeOptionsTab();
void updateStickyLimit();
static LayoutUnit getLayoutUnitKey(TQListViewItem *sel);
void checkConflicts(OptionListItem *current, TQStringList conflicting,
TQStringList &conflicts);
};

@ -116,7 +116,7 @@
</sizepolicy>
</property>
<property name="whatsThis" stdset="0">
<string>Here you can choose the key combination you want to use to switch to the next layout. This list includes only the most common variants. If you choose "Other...", then you will be redirected to the "Options" tab where you can pick from all the available variants.</string>
<string>Here you can choose the key combination you want to use to switch to the next layout. This list includes only the most common variants. If you choose "Other...", then you will be redirected to the "Options" tab where you can pick from all the available variants. Note that if you have selected Append Mode in the Xkb Options tab this option is not available; you have to use the Xkb Options tab instead.</string>
</property>
</widget>
<widget class="TQLabel" row="0" column="2" rowspan="1" colspan="3">
@ -430,13 +430,166 @@
<attribute name="title">
<string>Switching Options</string>
</attribute>
<grid>
<property name="name">
<cstring>unnamed</cstring>
</property>
<spacer row="1" column="0">
<vbox>
<widget class="TQFrame">
<property name="name">
<cstring>spacer1</cstring>
<cstring>swOptsFrame</cstring>
</property>
<property name="frameShape">
<enum>StyledPanel</enum>
</property>
<property name="frameShadow">
<enum>Raised</enum>
</property>
<property name="lineWidth">
<number>0</number>
</property>
<vbox>
<widget class="TQButtonGroup">
<property name="name">
<cstring>grpSwitching</cstring>
</property>
<property name="title">
<string>Switching Policy</string>
</property>
<property name="exclusive">
<bool>true</bool>
</property>
<property name="whatsThis" stdset="0">
<string>If you select "Application" or "Window" switching policy, changing the keyboard layout will only affect the current application or window.</string>
</property>
<vbox>
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1</cstring>
</property>
<property name="text">
<string>&amp;Global</string>
</property>
<property name="checked">
<bool>true</bool>
</property>
</widget>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1_3</cstring>
</property>
<property name="text">
<string>Application</string>
</property>
</widget>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1_2</cstring>
</property>
<property name="text">
<string>&amp;Window</string>
</property>
</widget>
</vbox>
</widget>
<widget class="TQGroupBox">
<property name="name">
<cstring>grpBoxStickySwitching</cstring>
</property>
<property name="title">
<string>Sticky Switching</string>
</property>
<grid>
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQCheckBox" row="0" column="0" rowspan="1" colspan="3">
<property name="name">
<cstring>chkEnableSticky</cstring>
</property>
<property name="text">
<string>Enable sticky switching</string>
</property>
<property name="whatsThis" stdset="0">
<string>If you have more than two layouts and turn this option on, switching with the keyboard shortcut or clicking on the kxkb indicator will only cycle through the last few layouts. You can specify the number of layouts to rotate below. You can still access all layouts by right-clicking on the kxkb indicator.</string>
</property>
</widget>
<spacer row="1" column="0">
<property name="name">
<cstring>spacer2</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Fixed</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>20</height>
</size>
</property>
</spacer>
<spacer row="1" column="3">
<property name="name">
<cstring>spacer2</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Expanding</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>20</height>
</size>
</property>
</spacer>
<widget class="TQLabel" row="1" column="1">
<property name="name">
<cstring>textLabel1_5</cstring>
</property>
<property name="enabled">
<bool>false</bool>
</property>
<property name="text">
<string>Number of layouts to rotate:</string>
</property>
<property name="buddy" stdset="0">
<cstring>spinBox1</cstring>
</property>
</widget>
<widget class="TQSpinBox" row="1" column="2">
<property name="name">
<cstring>spinStickyDepth</cstring>
</property>
<property name="enabled">
<bool>false</bool>
</property>
<property name="maxValue">
<number>10</number>
</property>
<property name="minValue">
<number>2</number>
</property>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>5</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
</grid>
</widget>
</vbox>
</widget>
<spacer>
<property name="name">
<cstring>spacer2</cstring>
</property>
<property name="orientation">
<enum>Vertical</enum>
@ -451,9 +604,19 @@
</size>
</property>
</spacer>
<widget class="TQFrame" row="0" column="0">
</vbox>
</widget>
<widget class="TQWidget">
<property name="name">
<cstring>tab</cstring>
</property>
<attribute name="title">
<string>Indicator Options</string>
</attribute>
<vbox>
<widget class="TQFrame">
<property name="name">
<cstring>optionsFrame</cstring>
<cstring>indOptsFrame</cstring>
</property>
<property name="frameShape">
<enum>StyledPanel</enum>
@ -465,10 +628,7 @@
<number>0</number>
</property>
<grid>
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQButtonGroup" row="1" column="0">
<widget class="TQButtonGroup" row="0" column="0">
<property name="name">
<cstring>grpStyle</cstring>
</property>
@ -514,7 +674,7 @@
</widget>
</vbox>
</widget>
<widget class="TQButtonGroup" row="1" column="1">
<widget class="TQButtonGroup" row="1" column="0" colspan="2">
<property name="name">
<cstring>grpLabel</cstring>
</property>
@ -558,6 +718,9 @@
<enum>Plain</enum>
</property>
<grid>
<property name="margin">
<number>0</number>
</property>
<spacer row="0" column="0">
<property name="name">
<cstring>spacer2</cstring>
@ -593,7 +756,32 @@
<property name="whatsThis" stdset="0">
<string>This color will be used as the indicator's background unless the indicator was set to display a flag.</string>
</property>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>4</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
<spacer row="0" column="3">
<property name="name">
<cstring>spacer23</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Expanding</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>20</height>
</size>
</property>
</spacer>
<spacer row="1" column="0">
<property name="name">
<cstring>spacer2</cstring>
@ -626,29 +814,71 @@
<property name="name">
<cstring>fgColor</cstring>
</property>
<property name="whatsThis" stdset="0">
<string>This color will be used to draw the language label on the indicator.</string>
</property>
<property name="whatsThis" stdset="0">
<string>This color will be used to draw the language label on the indicator.</string>
</property>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>4</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
<spacer row="1" column="3">
<property name="name">
<cstring>spacer23</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Expanding</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>20</height>
</size>
</property>
</spacer>
<spacer row="4" column="0">
<property name="name">
<cstring>spacer22</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Fixed</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>20</height>
</size>
</property>
</spacer>
<widget class="TQCheckBox" row="4" column="1" rowspan="1" colspan="3">
<property name="name">
<cstring>chkBgTransparent</cstring>
</property>
<property name="text">
<string>Transparent background</string>
</property>
<property name="whatsThis" stdset="0">
<string>Check this to remove the indicator's background. Only applicable in "Label only" mode.</string>
</property>
</widget>
</grid>
</widget>
<widget class="TQCheckBox" row="4" column="0" rowspan="1" colspan="3">
<property name="name">
<cstring>chkBgTransparent</cstring>
</property>
<property name="text">
<string>Transparent background</string>
</property>
<property name="whatsThis" stdset="0">
<string>Check this to remove the indicator's background. Only applicable in "Label only" mode.</string>
</property>
</widget>
<widget class="KSeparator" row="5" column="0" rowspan="1" colspan="3">
<property name="name">
<cstring>separator1</cstring>
</property>
</widget>
<widget class="TQLabel" row="6" column="0" rowspan="1">
<widget class="TQLabel" row="6" column="0">
<property name="name">
<cstring>labelFontRequester</cstring>
</property>
@ -659,12 +889,12 @@
<string>This is the font which will be used by the layout indicator to draw the label.</string>
</property>
</widget>
<widget class="TDEFontRequester" row="6" column="2">
<widget class="TDEFontRequester" row="6" column="1" colspan="2">
<property name="name">
<cstring>labelFont</cstring>
</property>
</widget>
<widget class="TQCheckBox" row="7" column="0" rowspan="1" colspan="2">
<widget class="TQCheckBox" row="7" column="0">
<property name="name">
<cstring>chkLabelShadow</cstring>
</property>
@ -674,8 +904,16 @@
<property name="whatsThis" stdset="0">
<string>Draw a drop shadow behind the language label. In some cases this option can improve readability.</string>
</property>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>4</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
<widget class="KColorButton" row="7" column="2">
<widget class="KColorButton" row="7" column="1">
<property name="name">
<cstring>shColor</cstring>
</property>
@ -685,86 +923,23 @@
<property name="enabled">
<bool>false</bool>
</property>
</widget>
</grid>
</widget>
<widget class="TQButtonGroup" row="2" column="0">
<property name="name">
<cstring>grpSwitching</cstring>
</property>
<property name="title">
<string>Switching Policy</string>
</property>
<property name="exclusive">
<bool>true</bool>
</property>
<property name="whatsThis" stdset="0">
<string>If you select "Application" or "Window" switching policy, changing the keyboard layout will only affect the current application or window.</string>
</property>
<vbox>
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1</cstring>
</property>
<property name="text">
<string>&amp;Global</string>
</property>
<property name="checked">
<bool>true</bool>
</property>
</widget>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1_3</cstring>
</property>
<property name="text">
<string>Application</string>
</property>
</widget>
<widget class="TQRadioButton">
<property name="name">
<cstring>radioButton1_2</cstring>
</property>
<property name="text">
<string>&amp;Window</string>
</property>
</widget>
</vbox>
</widget>
<widget class="TQGroupBox" row="2" column="1">
<property name="name">
<cstring>grpBoxStickySwitching</cstring>
</property>
<property name="title">
<string>Sticky Switching</string>
</property>
<grid>
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQCheckBox" row="0" column="0" rowspan="1" colspan="3">
<property name="name">
<cstring>chkEnableSticky</cstring>
</property>
<property name="text">
<string>Enable sticky switching</string>
</property>
<property name="whatsThis" stdset="0">
<string>If you have more than two layouts and turn this option on, switching with the keyboard shortcut or clicking on the kxkb indicator will only cycle through the last few layouts. You can specify the number of layouts to rotate below. You can still access all layouts by right-clicking on the kxkb indicator.</string>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>4</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
<spacer row="1" column="0">
</widget> <spacer row="7" column="2">
<property name="name">
<cstring>spacer2</cstring>
<cstring>spacer23</cstring>
</property>
<property name="orientation">
<enum>Horizontal</enum>
</property>
<property name="sizeType">
<enum>Fixed</enum>
<enum>Expanding</enum>
</property>
<property name="sizeHint">
<size>
@ -773,55 +948,46 @@
</size>
</property>
</spacer>
<widget class="TQLabel" row="1" column="1">
<property name="name">
<cstring>textLabel1_5</cstring>
</property>
<property name="enabled">
<bool>false</bool>
</property>
<property name="text">
<string>Number of layouts to rotate:</string>
</property>
<property name="buddy" stdset="0">
<cstring>spinBox1</cstring>
</property>
</widget>
<widget class="TQSpinBox" row="1" column="2">
<property name="name">
<cstring>spinStickyDepth</cstring>
</property>
<property name="enabled">
<bool>false</bool>
</property>
<property name="maxValue">
<number>10</number>
</property>
<property name="minValue">
<number>2</number>
</property>
<property name="sizePolicy">
<sizepolicy>
<hsizetype>7</hsizetype>
<vsizetype>0</vsizetype>
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
</grid>
</widget>
<widget class="TQCheckBox" row="0" column="0" rowspan="1" colspan="2">
<widget class="TQButtonGroup" row="0" column="1">
<property name="name">
<cstring>chkShowSingle</cstring>
<cstring>grpMisc</cstring>
</property>
<property name="text">
<string>Show indicator for single layout</string>
<property name="title">
<string>Miscellaneous</string>
</property>
<vbox>
<widget class="TQCheckBox">
<property name="name">
<cstring>chkShowSingle</cstring>
</property>
<property name="text">
<string>Show indicator for single layout</string>
</property>
</widget>
</vbox>
</widget>
</grid>
</widget>
</grid>
<spacer>
<property name="name">
<cstring>spacer1</cstring>
</property>
<property name="orientation">
<enum>Vertical</enum>
</property>
<property name="sizeType">
<enum>Expanding</enum>
</property>
<property name="sizeHint">
<size>
<width>20</width>
<height>40</height>
</size>
</property>
</spacer>
</vbox>
</widget>
<widget class="TQWidget">
<property name="name">
@ -848,15 +1014,7 @@
<property name="name">
<cstring>unnamed</cstring>
</property>
<widget class="TQCheckBox" row="0" column="0" rowspan="1" colspan="2">
<property name="name">
<cstring>checkResetOld</cstring>
</property>
<property name="text">
<string>&amp;Reset old options</string>
</property>
</widget>
<widget class="TQLabel" row="2" column="0">
<widget class="TQLabel" row="1" column="0">
<property name="name">
<cstring>textLabel1_3_2</cstring>
</property>
@ -864,7 +1022,7 @@
<string>Command:</string>
</property>
</widget>
<widget class="TQLineEdit" row="2" column="1">
<widget class="TQLineEdit" row="1" column="1">
<property name="name">
<cstring>editCmdLineOpt</cstring>
</property>
@ -872,7 +1030,7 @@
<bool>true</bool>
</property>
</widget>
<widget class="TQListView" row="1" column="0" rowspan="1" colspan="2">
<widget class="TQListView" row="0" column="0" rowspan="1" colspan="2">
<column>
<property name="text">
<string>Options</string>
@ -890,6 +1048,41 @@
</widget>
</grid>
</widget>
<widget class="TQButtonGroup">
<property name="name">
<cstring>xkbOptsMode</cstring>
</property>
<property name="title">
<string>Options Mode</string>
</property>
<property name="whatsThis" stdset="0">
<string>Here you can choose how the options you select here will be applied: in addition to, or instead of existing options.</string>
</property>
<vbox>
<widget class="TQRadioButton">
<property name="name">
<cstring>radXkbOverwrite</cstring>
</property>
<property name="text">
<string>&amp;Overwrite existing options (recommended)</string>
</property>
<property name="whatsThis" stdset="0">
<string>Overwrite any existing Xkb options that might have been previously set by another program or from a script (e.g. via setxkbmap). This is the recommended option.</string>
</property>
</widget>
<widget class="TQRadioButton">
<property name="name">
<cstring>radXkbAppend</cstring>
</property>
<property name="text">
<string>&amp;Append to existing options</string>
</property>
<property name="whatsThis" stdset="0">
<string>Append the selected options to any existing Xkb options that might have been previously set by another program or from a script (e.g. via setxkbmap). Only use this if you really need to.</string>
</property>
</widget>
</vbox>
</widget>
</vbox>
</widget>
</widget>
@ -902,6 +1095,18 @@
<receiver>grpLayouts</receiver>
<slot>setEnabled(bool)</slot>
</connection>
<connection>
<sender>chkEnable</sender>
<signal>toggled(bool)</signal>
<receiver>swOptsFrame</receiver>
<slot>setEnabled(bool)</slot>
</connection>
<connection>
<sender>chkEnable</sender>
<signal>toggled(bool)</signal>
<receiver>indOptsFrame</receiver>
<slot>setEnabled(bool)</slot>
</connection>
<connection>
<sender>chkEnableSticky</sender>
<signal>toggled(bool)</signal>
@ -926,12 +1131,6 @@
<receiver>shColor</receiver>
<slot>setEnabled(bool)</slot>
</connection>
<connection>
<sender>chkEnable</sender>
<signal>toggled(bool)</signal>
<receiver>optionsFrame</receiver>
<slot>setEnabled(bool)</slot>
</connection>
<connection>
<sender>radFlagOnly</sender>
<signal>toggled(bool)</signal>

@ -99,7 +99,7 @@ int KXKBApp::newInstance()
bool KXKBApp::settingsRead()
{
XkbOptions options = kxkbConfig.getXkbOptions();
XkbOptions options = kxkbConfig.getKXkbOptions();
if( !m_extension->setXkbOptions(options) ) {
kdDebug() << "Setting XKB options failed!" << endl;
}

@ -58,8 +58,8 @@ public:
virtual int newInstance();
bool setLayout(const LayoutUnit& layoutUnit);
bool setLayout(const uint group);
bool setLayout(const LayoutUnit& layoutUnit);
bool setLayout(const uint group);
k_dcop:
bool setLayout(const TQString& layoutPair);
TQString getCurrentLayout() { return m_currentLayout.toPair(); }
@ -72,7 +72,7 @@ protected slots:
void menuActivated(int id);
void windowChanged(WId winId);
void layoutApply();
void slotGroupChanged(uint group);
void slotGroupChanged(uint group);
void slotSettingsChanged(int category);

@ -266,7 +266,7 @@ TQString KxkbConfig::getDefaultDisplayName(const LayoutUnit& layoutUnit, bool si
return displayName;
}
const XkbOptions KxkbConfig::getXkbOptions() {
const XkbOptions KxkbConfig::getKXkbOptions() {
load(LOAD_ALL);
XkbOptions options;
@ -281,7 +281,7 @@ const XkbOptions KxkbConfig::getXkbOptions() {
options.variants = variants.join(",");
options.model = m_model;
options.options = m_options;
kdDebug() << "[getXkbOptions] options: " << m_options << endl;
kdDebug() << "[getKXkbOptions] options: " << m_options << endl;
options.resetOld = m_resetOldOptions;
return options;
}

@ -128,7 +128,7 @@ public:
static TQString getDefaultDisplayName(const TQString& code_);
static TQString getDefaultDisplayName(const LayoutUnit& layoutUnit, bool single=false);
const XkbOptions getXkbOptions();
const XkbOptions getKXkbOptions();
private:
static const TQMap<TQString, TQString> parseIncludesMap(const TQStringList& pairList);

@ -88,7 +88,7 @@ LayoutState& LayoutMap::getNextLayout() {
layoutQueue.enqueue(layoutState);
kdDebug() << "map: Next layout: " << layoutQueue.head()->layoutUnit.toPair()
<< " for " << m_currentWinId << endl;
<< " for " << m_currentWinId << endl;
return *layoutQueue.head();
}
@ -96,7 +96,7 @@ LayoutState& LayoutMap::getNextLayout() {
void LayoutMap::setCurrentLayout(const LayoutUnit& layoutUnit) {
LayoutQueue& layoutQueue = getCurrentLayoutQueue(m_currentWinId);
kdDebug() << "map: Storing layout: " << layoutUnit.toPair()
<< " for " << m_currentWinId << endl;
<< " for " << m_currentWinId << endl;
int queueSize = (int)layoutQueue.count();
for(int ii=0; ii<queueSize; ii++) {

Loading…
Cancel
Save